Add MadoCrypt

This commit is contained in:
2026-04-11 16:15:21 +08:00
parent 2873780797
commit 9b983728ef
2 changed files with 121 additions and 0 deletions

View File

@@ -231,6 +231,10 @@ enum CryptType {
RhapsodyCrypt {
file_list_name: String,
},
#[serde(rename_all = "PascalCase")]
MadoCrypt {
seed: u32,
},
}
#[derive(Clone, Debug, Deserialize)]
@@ -359,6 +363,7 @@ impl Schema {
&file_list_name,
config.xp3_file_list_path.as_ref().map(|s| s.as_str()),
)?),
CryptType::MadoCrypt { seed } => Box::new(MadoCrypt::new(self.base.clone(), *seed)),
})
}
}
@@ -543,6 +548,20 @@ macro_rules! base_schema_impl {
};
}
macro_rules! base_schema2_impl {
() => {
fn hash_after_crypt(&self) -> bool {
AsRef::<BaseSchema>::as_ref(self).hash_after_crypt
}
fn startup_tjs_not_encrypted(&self) -> bool {
AsRef::<BaseSchema>::as_ref(self).startup_tjs_not_encrypted
}
fn obfuscated_index(&self) -> bool {
AsRef::<BaseSchema>::as_ref(self).obfuscated_index
}
};
}
macro_rules! seek_crypt_base_impl {
($crypt:ident, $reader:ident) => {
#[derive(Debug)]
@@ -1667,6 +1686,73 @@ impl<R: Read> Read for RhapsodyCryptReader<R> {
}
}
#[derive(Debug)]
pub struct MadoCrypt {
base: AkabeiCrypt,
}
impl MadoCrypt {
pub fn new(base: BaseSchema, seed: u32) -> Self {
Self {
base: AkabeiCrypt::new(base, seed),
}
}
}
impl AsRef<BaseSchema> for MadoCrypt {
fn as_ref(&self) -> &BaseSchema {
&self.base.base
}
}
impl Crypt for MadoCrypt {
base_schema2_impl!();
fn decrypt_supported(&self) -> bool {
true
}
fn decrypt_seek_supported(&self) -> bool {
true
}
fn decrypt<'a>(
&self,
entry: &Xp3Entry,
cur_seg: &Segment,
stream: Box<dyn Read + 'a>,
) -> Result<Box<dyn ReadDebug + 'a>> {
Ok(Box::new(MadoCryptReader::new(
stream,
cur_seg,
self.base.get_key(entry.file_hash),
)))
}
fn decrypt_with_seek<'a>(
&self,
entry: &Xp3Entry,
cur_seg: &Segment,
stream: Box<dyn ReadSeek + 'a>,
) -> Result<Box<dyn ReadSeek + 'a>> {
Ok(Box::new(MadoCryptReader::new(
stream,
cur_seg,
self.base.get_key(entry.file_hash),
)))
}
}
seek_reader_key_impl!(MadoCryptReader<T>, [u8; 0x20]);
impl<R: Read> Read for MadoCryptReader<R> {
fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> {
let readed = self.inner.read(buf)?;
for (i, t) in (&mut buf[..readed]).iter_mut().enumerate() {
let offset = self.seg_start + self.pos + i as u64;
*t ^= self.key[(offset % 0x1F) as usize];
}
self.pos += readed as u64;
Ok(readed)
}
}
#[test]
fn test_deserialize_crypt() {
for (key, schema) in CRYPT_SCHEMA.iter() {