Don't store oodle as a global

This commit is contained in:
Truman Kilen 2023-12-16 18:47:31 -06:00
parent 95b4f35b0d
commit c84be27763
5 changed files with 37 additions and 30 deletions

View file

@ -22,11 +22,11 @@ type OodleLZ_Decompress = unsafe extern "win64" fn(
threadPhase: u32, threadPhase: u32,
) -> i32; ) -> i32;
pub fn decompress() -> OodleDecompress { pub fn decompress() -> Result<OodleDecompress, Box<dyn std::error::Error>> {
#[cfg(windows)] #[cfg(windows)]
return windows_oodle::decompress_wrapper_windows; return Ok(windows_oodle::decompress_wrapper_windows);
#[cfg(unix)] #[cfg(unix)]
return linux_oodle::oodle_loader_linux(); return Ok(linux_oodle::oodle_loader_linux());
} }
fn call_decompress(comp_buf: &[u8], raw_buf: &mut [u8], decompress: OodleLZ_Decompress) -> i32 { fn call_decompress(comp_buf: &[u8], raw_buf: &mut [u8], decompress: OodleLZ_Decompress) -> i32 {

View file

@ -10,6 +10,7 @@ edition.workspace = true
default = ["compression", "encryption"] default = ["compression", "encryption"]
compression = ["dep:flate2", "dep:zstd"] compression = ["dep:flate2", "dep:zstd"]
oodle = [] oodle = []
oodle_loader = ["dep:oodle_loader"]
encryption = ["dep:aes"] encryption = ["dep:aes"]
[dependencies] [dependencies]
@ -17,10 +18,10 @@ byteorder = "1.4"
aes = { workspace = true, optional = true } aes = { workspace = true, optional = true }
flate2 = { version = "1.0", optional = true } flate2 = { version = "1.0", optional = true }
zstd = { version = "0.12", optional = true } zstd = { version = "0.12", optional = true }
oodle_loader = { path = "../oodle_loader", optional = true}
thiserror = "1.0" thiserror = "1.0"
sha1 = { workspace = true } sha1 = { workspace = true }
strum = { workspace = true } strum = { workspace = true }
once_cell = "1.18"
[dev-dependencies] [dev-dependencies]
base64 = { workspace = true } base64 = { workspace = true }

View file

@ -306,6 +306,7 @@ impl Entry {
version: Version, version: Version,
compression: &[Compression], compression: &[Compression],
#[allow(unused)] key: &super::Key, #[allow(unused)] key: &super::Key,
#[allow(unused)] oodle: &super::Oodle,
buf: &mut W, buf: &mut W,
) -> Result<(), super::Error> { ) -> Result<(), super::Error> {
reader.seek(io::SeekFrom::Start(self.offset))?; reader.seek(io::SeekFrom::Start(self.offset))?;
@ -375,10 +376,11 @@ impl Entry {
} }
} }
#[cfg(feature = "oodle")] #[cfg(feature = "oodle")]
Some(Compression::Oodle) => unsafe { Some(Compression::Oodle) => {
let Some(ref oodle) = super::OODLE else { let oodle = match oodle {
return Err(super::Error::OodleFailed); crate::Oodle::Some(getter) => getter().map_err(|_| super::Error::OodleFailed),
}; crate::Oodle::None => Err(super::Error::OodleFailed),
}?;
let mut decompressed = vec![0; self.uncompressed as usize]; let mut decompressed = vec![0; self.uncompressed as usize];
let mut compress_offset = 0; let mut compress_offset = 0;
@ -408,7 +410,7 @@ impl Entry {
"Oodle decompression length mismatch" "Oodle decompression length mismatch"
); );
buf.write_all(&decompressed)?; buf.write_all(&decompressed)?;
}, }
#[cfg(not(feature = "oodle"))] #[cfg(not(feature = "oodle"))]
Some(Compression::Oodle) => return Err(super::Error::Oodle), Some(Compression::Oodle) => return Err(super::Error::Oodle),
#[cfg(not(feature = "compression"))] #[cfg(not(feature = "compression"))]

View file

@ -10,10 +10,13 @@ pub use {error::*, pak::*};
pub const MAGIC: u32 = 0x5A6F12E1; pub const MAGIC: u32 = 0x5A6F12E1;
#[cfg(feature = "oodle")] #[cfg(feature = "oodle")]
static mut OODLE: Option<once_cell::sync::Lazy<OodleDecompress>> = None; mod oodle {
pub type OodleGetter = fn() -> Result<OodleDecompress, Box<dyn std::error::Error>>;
pub type OodleDecompress = fn(comp_buf: &[u8], raw_buf: &mut [u8]) -> i32;
}
#[cfg(feature = "oodle")] #[cfg(feature = "oodle_loader")]
type OodleDecompress = fn(comp_buf: &[u8], raw_buf: &mut [u8]) -> i32; pub use oodle_loader;
#[derive( #[derive(
Clone, Clone,
@ -127,10 +130,11 @@ pub enum Compression {
} }
#[allow(clippy::large_enum_variant)] #[allow(clippy::large_enum_variant)]
#[derive(Debug)] #[derive(Debug, Default)]
pub(crate) enum Key { pub(crate) enum Key {
#[cfg(feature = "encryption")] #[cfg(feature = "encryption")]
Some(aes::Aes256), Some(aes::Aes256),
#[default]
None, None,
} }
@ -141,13 +145,10 @@ impl From<aes::Aes256> for Key {
} }
} }
#[cfg(feature = "oodle")] #[derive(Debug, Default)]
pub(crate) enum Oodle<'func> {
Some(&'func OodleDecompress),
None,
}
#[cfg(not(feature = "oodle"))]
pub(crate) enum Oodle { pub(crate) enum Oodle {
#[cfg(feature = "oodle")]
Some(oodle::OodleGetter),
#[default]
None, None,
} }

View file

@ -4,16 +4,15 @@ use byteorder::{ReadBytesExt, WriteBytesExt, LE};
use std::collections::BTreeMap; use std::collections::BTreeMap;
use std::io::{self, Read, Seek, Write}; use std::io::{self, Read, Seek, Write};
#[derive(Debug)] #[derive(Debug, Default)]
pub struct PakBuilder { pub struct PakBuilder {
key: super::Key, key: super::Key,
oodle: super::Oodle,
} }
impl PakBuilder { impl PakBuilder {
pub fn new() -> Self { pub fn new() -> Self {
Self { Self::default()
key: super::Key::None,
}
} }
#[cfg(feature = "encryption")] #[cfg(feature = "encryption")]
pub fn key(mut self, key: aes::Aes256) -> Self { pub fn key(mut self, key: aes::Aes256) -> Self {
@ -21,19 +20,19 @@ impl PakBuilder {
self self
} }
#[cfg(feature = "oodle")] #[cfg(feature = "oodle")]
pub fn oodle(self, oodle: fn() -> super::OodleDecompress) -> Self { pub fn oodle(mut self, oodle_getter: super::oodle::OodleGetter) -> Self {
unsafe { super::OODLE = Some(once_cell::sync::Lazy::new(oodle)) } self.oodle = super::Oodle::Some(oodle_getter);
self self
} }
pub fn reader<R: Read + Seek>(self, reader: &mut R) -> Result<PakReader, super::Error> { pub fn reader<R: Read + Seek>(self, reader: &mut R) -> Result<PakReader, super::Error> {
PakReader::new_any_inner(reader, self.key) PakReader::new_any_inner(reader, self.key, self.oodle)
} }
pub fn reader_with_version<R: Read + Seek>( pub fn reader_with_version<R: Read + Seek>(
self, self,
reader: &mut R, reader: &mut R,
version: super::Version, version: super::Version,
) -> Result<PakReader, super::Error> { ) -> Result<PakReader, super::Error> {
PakReader::new_inner(reader, version, self.key) PakReader::new_inner(reader, version, self.key, self.oodle)
} }
pub fn writer<W: Write + Seek>( pub fn writer<W: Write + Seek>(
self, self,
@ -50,6 +49,7 @@ impl PakBuilder {
pub struct PakReader { pub struct PakReader {
pak: Pak, pak: Pak,
key: super::Key, key: super::Key,
oodle: super::Oodle,
} }
#[derive(Debug)] #[derive(Debug)]
@ -128,13 +128,14 @@ impl PakReader {
fn new_any_inner<R: Read + Seek>( fn new_any_inner<R: Read + Seek>(
reader: &mut R, reader: &mut R,
key: super::Key, key: super::Key,
oodle: super::Oodle,
) -> Result<Self, super::Error> { ) -> Result<Self, super::Error> {
use std::fmt::Write; use std::fmt::Write;
let mut log = "\n".to_owned(); let mut log = "\n".to_owned();
for ver in Version::iter() { for ver in Version::iter() {
match Pak::read(&mut *reader, ver, &key) { match Pak::read(&mut *reader, ver, &key) {
Ok(pak) => return Ok(Self { pak, key }), Ok(pak) => return Ok(Self { pak, key, oodle }),
Err(err) => writeln!(log, "trying version {} failed: {}", ver, err)?, Err(err) => writeln!(log, "trying version {} failed: {}", ver, err)?,
} }
} }
@ -145,8 +146,9 @@ impl PakReader {
reader: &mut R, reader: &mut R,
version: super::Version, version: super::Version,
key: super::Key, key: super::Key,
oodle: super::Oodle,
) -> Result<Self, super::Error> { ) -> Result<Self, super::Error> {
Pak::read(reader, version, &key).map(|pak| Self { pak, key }) Pak::read(reader, version, &key).map(|pak| Self { pak, key, oodle })
} }
pub fn version(&self) -> super::Version { pub fn version(&self) -> super::Version {
@ -183,6 +185,7 @@ impl PakReader {
self.pak.version, self.pak.version,
&self.pak.compression, &self.pak.compression,
&self.key, &self.key,
&self.oodle,
writer, writer,
), ),
None => Err(super::Error::MissingEntry(path.to_owned())), None => Err(super::Error::MissingEntry(path.to_owned())),