mirror of
https://github.com/lifegpc/msg-tool.git
synced 2026-06-16 01:54:19 +08:00
Add SyangrilaSmartCrypt
This commit is contained in:
@@ -256,6 +256,7 @@ enum CryptType {
|
||||
key2: u64,
|
||||
key3: u64,
|
||||
},
|
||||
SyangrilaSmartCrypt,
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, Deserialize)]
|
||||
@@ -404,6 +405,7 @@ impl Schema {
|
||||
*key2,
|
||||
*key3,
|
||||
)),
|
||||
CryptType::SyangrilaSmartCrypt => Box::new(SyangrilaSmartCrypt::new(self.base.clone())),
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -2127,6 +2129,80 @@ impl<R: Read> Read for NinkiSeiyuuCryptReader<R> {
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct SyangrilaSmartCrypt {
|
||||
base: BaseSchema,
|
||||
}
|
||||
|
||||
impl SyangrilaSmartCrypt {
|
||||
pub fn new(base: BaseSchema) -> Self {
|
||||
Self { base }
|
||||
}
|
||||
|
||||
fn get_key(hash: u32) -> [u8; 5] {
|
||||
[
|
||||
(hash >> 5) as u8,
|
||||
(hash >> 5) as u8,
|
||||
(hash >> 7) as u8,
|
||||
(hash >> 1) as u8,
|
||||
(hash >> 4) as u8,
|
||||
]
|
||||
}
|
||||
}
|
||||
|
||||
impl Crypt for SyangrilaSmartCrypt {
|
||||
base_schema_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 + Send + Sync + 'a>,
|
||||
) -> Result<Box<dyn ReadDebug + Send + Sync + 'a>> {
|
||||
Ok(Box::new(SyangrilaSmartCryptReader::new(
|
||||
stream,
|
||||
cur_seg,
|
||||
Self::get_key(entry.file_hash),
|
||||
)))
|
||||
}
|
||||
fn decrypt_with_seek<'a>(
|
||||
&self,
|
||||
entry: &Xp3Entry,
|
||||
cur_seg: &Segment,
|
||||
stream: Box<dyn ReadSeek + Send + Sync + 'a>,
|
||||
) -> Result<Box<dyn ReadSeek + Send + Sync + 'a>> {
|
||||
Ok(Box::new(SyangrilaSmartCryptReader::new(
|
||||
stream,
|
||||
cur_seg,
|
||||
Self::get_key(entry.file_hash),
|
||||
)))
|
||||
}
|
||||
}
|
||||
|
||||
seek_reader_key_impl!(SyangrilaSmartCryptReader<T>, [u8; 5]);
|
||||
|
||||
impl<R: Read> Read for SyangrilaSmartCryptReader<R> {
|
||||
fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> {
|
||||
let readed = self.inner.read(buf)?;
|
||||
let offset = self.seg_start + self.pos;
|
||||
for (i, t) in buf[..readed].iter_mut().enumerate() {
|
||||
let tpos = offset + i as u64;
|
||||
if tpos <= 0x64 {
|
||||
*t ^= self.key[4];
|
||||
} else {
|
||||
*t ^= self.key[(tpos & 3) as usize];
|
||||
}
|
||||
}
|
||||
self.pos += readed as u64;
|
||||
Ok(readed)
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_deserialize_crypt() {
|
||||
for (key, schema) in CRYPT_SCHEMA.iter() {
|
||||
|
||||
Reference in New Issue
Block a user