Compare commits

..

No commits in common. "490ae80a60b5fcb447cf56b73370b252f270abbd" and "b8b06dcd5cd90d361309ed0dc398ab75a9f9c0bd" have entirely different histories.

8 changed files with 68 additions and 84 deletions

26
Cargo.lock generated
View File

@ -393,15 +393,6 @@ dependencies = [
"serde", "serde",
] ]
[[package]]
name = "higher-kinded-types"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "561985554c8b8d4808605c90a5f1979cc6c31a5d20b78465cd59501233c6678e"
dependencies = [
"never-say-never",
]
[[package]] [[package]]
name = "hybrid-array" name = "hybrid-array"
version = "0.2.1" version = "0.2.1"
@ -589,12 +580,6 @@ dependencies = [
"adler2", "adler2",
] ]
[[package]]
name = "never-say-never"
version = "6.6.666"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cf5a574dadd7941adeaa71823ecba5e28331b8313fb2e1c6a5c7e5981ea53ad6"
[[package]] [[package]]
name = "nod" name = "nod"
version = "2.0.0-alpha.1" version = "2.0.0-alpha.1"
@ -620,7 +605,6 @@ dependencies = [
"memmap2", "memmap2",
"miniz_oxide", "miniz_oxide",
"openssl", "openssl",
"polonius-the-crab",
"rand", "rand",
"rayon", "rayon",
"sha1", "sha1",
@ -767,16 +751,6 @@ version = "0.3.31"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2" checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2"
[[package]]
name = "polonius-the-crab"
version = "0.4.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e97ca2c89572ae41bbec1c99498251f87dd5a94e500c5ec19c382dd593dd5ce9"
dependencies = [
"higher-kinded-types",
"never-say-never",
]
[[package]] [[package]]
name = "portable-atomic" name = "portable-atomic"
version = "1.9.0" version = "1.9.0"

View File

@ -45,7 +45,6 @@ md-5 = { workspace = true }
memmap2 = "0.9" memmap2 = "0.9"
miniz_oxide = { version = "0.8", optional = true } miniz_oxide = { version = "0.8", optional = true }
openssl = { version = "0.10", optional = true } openssl = { version = "0.10", optional = true }
polonius-the-crab = "0.4"
rand = "0.8" rand = "0.8"
rayon = "1.10" rayon = "1.10"
sha1 = { workspace = true } sha1 = { workspace = true }

View File

@ -9,7 +9,7 @@ use zerocopy::FromBytes;
use crate::{ use crate::{
disc::{ disc::{
preloader::{fetch_sector_group, Preloader, SectorGroup, SectorGroupRequest}, preloader::{Preloader, SectorGroup, SectorGroupRequest},
ApploaderHeader, DiscHeader, DolHeader, PartitionHeader, BI2_SIZE, BOOT_SIZE, ApploaderHeader, DiscHeader, DolHeader, PartitionHeader, BI2_SIZE, BOOT_SIZE,
SECTOR_GROUP_SIZE, SECTOR_SIZE, SECTOR_GROUP_SIZE, SECTOR_SIZE,
}, },
@ -77,9 +77,14 @@ impl BufRead for PartitionReaderGC {
mode: PartitionEncryption::Original, mode: PartitionEncryption::Original,
}; };
// Load sector group let sector_group = if matches!(&self.sector_group, Some(sector_group) if sector_group.request == request)
let (sector_group, _updated) = {
fetch_sector_group(request, max_groups, &mut self.sector_group, &self.preloader)?; // We can improve this in Rust 2024 with `if_let_rescope`
// https://github.com/rust-lang/rust/issues/124085
self.sector_group.as_ref().unwrap()
} else {
self.sector_group.insert(self.preloader.fetch(request, max_groups)?)
};
// Calculate the number of consecutive sectors in the group // Calculate the number of consecutive sectors in the group
let group_sector = abs_sector - abs_group_sector; let group_sector = abs_sector - abs_group_sector;
@ -141,9 +146,18 @@ pub(crate) fn read_dol(
let mut raw_dol: Vec<u8> = let mut raw_dol: Vec<u8> =
read_vec(reader, size_of::<DolHeader>()).context("Reading DOL header")?; read_vec(reader, size_of::<DolHeader>()).context("Reading DOL header")?;
let dol_header = DolHeader::ref_from_bytes(raw_dol.as_slice()).unwrap(); let dol_header = DolHeader::ref_from_bytes(raw_dol.as_slice()).unwrap();
let dol_size = (dol_header.text_offs.iter().zip(&dol_header.text_sizes)) let dol_size = dol_header
.chain(dol_header.data_offs.iter().zip(&dol_header.data_sizes)) .text_offs
.iter()
.zip(&dol_header.text_sizes)
.map(|(offs, size)| offs.get() + size.get()) .map(|(offs, size)| offs.get() + size.get())
.chain(
dol_header
.data_offs
.iter()
.zip(&dol_header.data_sizes)
.map(|(offs, size)| offs.get() + size.get()),
)
.max() .max()
.unwrap_or(size_of::<DolHeader>() as u32); .unwrap_or(size_of::<DolHeader>() as u32);
raw_dol.resize(dol_size as usize, 0); raw_dol.resize(dol_size as usize, 0);
@ -189,10 +203,12 @@ pub(crate) fn read_part_meta(
let mut raw_apploader: Vec<u8> = let mut raw_apploader: Vec<u8> =
read_vec(reader, size_of::<ApploaderHeader>()).context("Reading apploader header")?; read_vec(reader, size_of::<ApploaderHeader>()).context("Reading apploader header")?;
let apploader_header = ApploaderHeader::ref_from_bytes(raw_apploader.as_slice()).unwrap(); let apploader_header = ApploaderHeader::ref_from_bytes(raw_apploader.as_slice()).unwrap();
let apploader_size = size_of::<ApploaderHeader>() raw_apploader.resize(
+ apploader_header.size.get() as usize size_of::<ApploaderHeader>()
+ apploader_header.trailer_size.get() as usize; + apploader_header.size.get() as usize
raw_apploader.resize(apploader_size, 0); + apploader_header.trailer_size.get() as usize,
0,
);
reader reader
.read_exact(&mut raw_apploader[size_of::<ApploaderHeader>()..]) .read_exact(&mut raw_apploader[size_of::<ApploaderHeader>()..])
.context("Reading apploader")?; .context("Reading apploader")?;

View File

@ -67,7 +67,7 @@ pub struct DiscHeader {
pub audio_stream_buf_size: u8, pub audio_stream_buf_size: u8,
/// Padding /// Padding
_pad1: [u8; 14], _pad1: [u8; 14],
/// If this is a Wii disc, this will be 0x5D1C9EA3 /// If this is a Wii disc, this will bPartitionKinde 0x5D1C9EA3
pub wii_magic: MagicBytes, pub wii_magic: MagicBytes,
/// If this is a GameCube disc, this will be 0xC2339F3D /// If this is a GameCube disc, this will be 0xC2339F3D
pub gcn_magic: MagicBytes, pub gcn_magic: MagicBytes,
@ -126,7 +126,7 @@ pub struct PartitionHeader {
pub debug_mon_offset: U32, pub debug_mon_offset: U32,
/// Debug monitor load address /// Debug monitor load address
pub debug_load_address: U32, pub debug_load_address: U32,
/// Padding /// PaddingPartitionKind
_pad1: [u8; 0x18], _pad1: [u8; 0x18],
/// Offset to main DOL (Wii: >> 2) /// Offset to main DOL (Wii: >> 2)
pub dol_offset: U32, pub dol_offset: U32,

View File

@ -11,7 +11,6 @@ use bytes::{Bytes, BytesMut};
use crossbeam_channel::{Receiver, Sender}; use crossbeam_channel::{Receiver, Sender};
use crossbeam_utils::sync::WaitGroup; use crossbeam_utils::sync::WaitGroup;
use lru::LruCache; use lru::LruCache;
use polonius_the_crab::{polonius, polonius_return};
use simple_moving_average::{SingleSumSMA, SMA}; use simple_moving_average::{SingleSumSMA, SMA};
use tracing::{debug, error, instrument, span, Level}; use tracing::{debug, error, instrument, span, Level};
use zerocopy::FromZeros; use zerocopy::FromZeros;
@ -557,22 +556,3 @@ impl SectorGroupLoader {
Ok((sector_bitmap, io_duration)) Ok((sector_bitmap, io_duration))
} }
} }
/// Fetch a sector group from the cache or from the preloader.
/// Returns a boolean indicating if the group was updated.
pub fn fetch_sector_group<'a>(
request: SectorGroupRequest,
max_groups: u32,
mut cached: &'a mut Option<SectorGroup>,
preloader: &Preloader,
) -> io::Result<(&'a SectorGroup, bool)> {
polonius!(|cached| -> io::Result<(&'polonius SectorGroup, bool)> {
if let Some(sector_group) = cached {
if sector_group.request == request {
polonius_return!(Ok((sector_group, false)));
}
}
});
let sector_group = preloader.fetch(request, max_groups)?;
Ok((cached.insert(sector_group), true))
}

View File

@ -14,9 +14,7 @@ use crate::{
direct::{DirectDiscReader, DirectDiscReaderMode}, direct::{DirectDiscReader, DirectDiscReaderMode},
fst::{Fst, NodeKind}, fst::{Fst, NodeKind},
gcn::{read_fst, PartitionReaderGC}, gcn::{read_fst, PartitionReaderGC},
preloader::{ preloader::{Preloader, SectorGroup, SectorGroupLoader, SectorGroupRequest},
fetch_sector_group, Preloader, SectorGroup, SectorGroupLoader, SectorGroupRequest,
},
wii::{ wii::{
PartitionReaderWii, WiiPartEntry, WiiPartGroup, WiiPartitionHeader, REGION_OFFSET, PartitionReaderWii, WiiPartEntry, WiiPartGroup, WiiPartitionHeader, REGION_OFFSET,
REGION_SIZE, WII_PART_GROUP_OFF, REGION_SIZE, WII_PART_GROUP_OFF,
@ -34,7 +32,6 @@ use crate::{
pub struct DiscReader { pub struct DiscReader {
io: Box<dyn BlockReader>, io: Box<dyn BlockReader>,
preloader: Arc<Preloader>,
pos: u64, pos: u64,
size: u64, size: u64,
mode: PartitionEncryption, mode: PartitionEncryption,
@ -42,6 +39,7 @@ pub struct DiscReader {
partitions: Arc<[PartitionInfo]>, partitions: Arc<[PartitionInfo]>,
region: Option<[u8; REGION_SIZE]>, region: Option<[u8; REGION_SIZE]>,
sector_group: Option<SectorGroup>, sector_group: Option<SectorGroup>,
preloader: Arc<Preloader>,
alt_disc_header: Option<Arc<DiscHeader>>, alt_disc_header: Option<Arc<DiscHeader>>,
alt_partitions: Option<Arc<[PartitionInfo]>>, alt_partitions: Option<Arc<[PartitionInfo]>>,
} }
@ -50,7 +48,6 @@ impl Clone for DiscReader {
fn clone(&self) -> Self { fn clone(&self) -> Self {
Self { Self {
io: self.io.clone(), io: self.io.clone(),
preloader: self.preloader.clone(),
pos: 0, pos: 0,
size: self.size, size: self.size,
mode: self.mode, mode: self.mode,
@ -58,6 +55,7 @@ impl Clone for DiscReader {
partitions: self.partitions.clone(), partitions: self.partitions.clone(),
region: self.region, region: self.region,
sector_group: None, sector_group: None,
preloader: self.preloader.clone(),
alt_disc_header: self.alt_disc_header.clone(), alt_disc_header: self.alt_disc_header.clone(),
alt_partitions: self.alt_partitions.clone(), alt_partitions: self.alt_partitions.clone(),
} }
@ -126,7 +124,6 @@ impl DiscReader {
); );
Ok(Self { Ok(Self {
io, io,
preloader,
pos: 0, pos: 0,
size, size,
mode: options.partition_encryption, mode: options.partition_encryption,
@ -134,6 +131,7 @@ impl DiscReader {
partitions, partitions,
region, region,
sector_group: None, sector_group: None,
preloader,
alt_disc_header, alt_disc_header,
alt_partitions, alt_partitions,
}) })
@ -248,8 +246,14 @@ impl DiscReader {
}; };
// Load sector group // Load sector group
let (sector_group, _updated) = let sector_group = if matches!(&self.sector_group, Some(sector_group) if sector_group.request == request)
fetch_sector_group(request, max_groups, &mut self.sector_group, &self.preloader)?; {
// We can improve this in Rust 2024 with `if_let_rescope`
// https://github.com/rust-lang/rust/issues/124085
self.sector_group.as_ref().unwrap()
} else {
self.sector_group.insert(self.preloader.fetch(request, max_groups)?)
};
// Calculate the number of consecutive sectors in the group // Calculate the number of consecutive sectors in the group
let group_sector = abs_sector - abs_group_sector; let group_sector = abs_sector - abs_group_sector;
@ -303,8 +307,14 @@ impl BufRead for DiscReader {
}; };
// Load sector group // Load sector group
let (sector_group, _updated) = let sector_group = if matches!(&self.sector_group, Some(sector_group) if sector_group.request == request)
fetch_sector_group(request, max_groups, &mut self.sector_group, &self.preloader)?; {
// We can improve this in Rust 2024 with `if_let_rescope`
// https://github.com/rust-lang/rust/issues/124085
self.sector_group.as_ref().unwrap()
} else {
self.sector_group.insert(self.preloader.fetch(request, max_groups)?)
};
// Calculate the number of consecutive sectors in the group // Calculate the number of consecutive sectors in the group
let group_sector = abs_sector - abs_group_sector; let group_sector = abs_sector - abs_group_sector;

View File

@ -15,7 +15,7 @@ use crate::{
disc::{ disc::{
gcn::{read_part_meta, PartitionReaderGC}, gcn::{read_part_meta, PartitionReaderGC},
hashes::sha1_hash, hashes::sha1_hash,
preloader::{fetch_sector_group, Preloader, SectorGroup, SectorGroupRequest}, preloader::{Preloader, SectorGroup, SectorGroupRequest},
SECTOR_GROUP_SIZE, SECTOR_SIZE, SECTOR_GROUP_SIZE, SECTOR_SIZE,
}, },
io::block::BlockReader, io::block::BlockReader,
@ -378,19 +378,24 @@ impl BufRead for PartitionReaderWii {
PartitionEncryption::ForceDecryptedNoHashes PartitionEncryption::ForceDecryptedNoHashes
}, },
}; };
let sector_group = if matches!(&self.sector_group, Some(sector_group) if sector_group.request == request)
// Load sector group {
let (sector_group, updated) = // We can improve this in Rust 2024 with `if_let_rescope`
fetch_sector_group(request, max_groups, &mut self.sector_group, &self.preloader)?; // https://github.com/rust-lang/rust/issues/124085
if updated && self.options.validate_hashes { self.sector_group.as_ref().unwrap()
if let Some(h3_table) = self.meta.as_ref().and_then(|m| m.raw_h3_table.as_deref()) { } else {
verify_hashes( let sector_group = self.preloader.fetch(request, max_groups)?;
array_ref![sector_group.data, 0, SECTOR_GROUP_SIZE], if self.options.validate_hashes {
group_idx, if let Some(h3_table) = self.meta.as_ref().and_then(|m| m.raw_h3_table.as_deref()) {
h3_table, verify_hashes(
)?; array_ref![sector_group.data, 0, SECTOR_GROUP_SIZE],
group_idx,
h3_table,
)?;
}
} }
} self.sector_group.insert(sector_group)
};
// Read from sector group buffer // Read from sector group buffer
let consecutive_sectors = sector_group.consecutive_sectors(group_sector); let consecutive_sectors = sector_group.consecutive_sectors(group_sector);

View File

@ -65,7 +65,7 @@ impl BlockReaderCISO {
} }
// Build block map // Build block map
let mut block_map = <[u16; CISO_MAP_SIZE]>::new_box_zeroed()?; let mut block_map = [0u16; CISO_MAP_SIZE];
let mut block = 0u16; let mut block = 0u16;
for (presence, out) in header.block_present.iter().zip(block_map.iter_mut()) { for (presence, out) in header.block_present.iter().zip(block_map.iter_mut()) {
if *presence == 1 { if *presence == 1 {
@ -92,7 +92,7 @@ impl BlockReaderCISO {
None None
}; };
Ok(Box::new(Self { inner, header, block_map: Arc::from(block_map), nkit_header })) Ok(Box::new(Self { inner, header, block_map: Arc::new(block_map), nkit_header }))
} }
} }