1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
|
use log::debug;
use std::{
env::args,
fs::File,
io::{BufReader, Read},
};
use unity_tools::{
helper::{Endianness, ReadExt},
unityfs::UnityFS,
};
fn main() -> anyhow::Result<()> {
env_logger::init_from_env("LOG");
let file = BufReader::new(File::open(args().nth(1).unwrap())?);
let mut fs = UnityFS::open(file)?;
for node in fs.nodes().to_vec() {
if node.name.ends_with(".resource") || node.name.ends_with("resS") {
continue;
}
let mut cab = fs.read(&node)?;
// let mut writer = File::create(format!("/tmp/{}", node.name))?;
// copy(&mut cab, &mut writer)?;
// continue;
let mut metadata_size = cab.read_u32_be()?;
let mut file_size = cab.read_u32_be()? as u64;
let format = cab.read_u32_be()?;
let mut data_offset = cab.read_u32_be()? as u64;
debug!("format={format}");
assert!(format >= 9);
let e = match cab.read_u32_be()? {
0 => Endianness::Little,
_ => Endianness::Big,
};
debug!("endianess={e:?}");
if format >= 22 {
metadata_size = cab.read_u32(e)?;
file_size = cab.read_u64(e)?;
data_offset = cab.read_u64(e)?;
cab.read_u64(e)?;
}
debug!("metadata_size={metadata_size}");
debug!("file_size={file_size}");
debug!("data_offset={data_offset}");
let generator_version = cab.read_cstr()?;
let target_platform = cab.read_u32_le()?;
debug!("generator_version={generator_version:?}");
debug!("target_platform={target_platform}");
let has_type_trees = cab.read_u8()? != 0;
let num_types = cab.read_u32(e)?;
for _ in 0..num_types {
let mut class_id = cab.read_i32(e)?;
let stripped_type = cab.read_u8()? != 0;
let script_id = cab.read_i16(e)?;
if class_id == 114 {
if script_id >= 0 {
class_id = -2 - script_id as i32;
} else {
class_id = -1;
}
}
eprintln!("class_id={class_id}");
eprintln!("stripped_type={stripped_type}");
eprintln!("script_id={script_id}");
let hash = if class_id < 0 {
(cab.read_u128_be()?, cab.read_u128_be()?)
} else {
(cab.read_u128_be()?, 0)
};
eprintln!("{hash:032x?}");
if has_type_trees {
let num_nodes = cab.read_u32(e)?;
eprintln!("tree:num_nodes={num_nodes}");
let size = cab.read_u32(e)?;
assert!(format >= 19);
for _ in 0..num_nodes {
cab.read_u32(e)?;
}
let mut data = vec![0u8; size as usize];
cab.read_exact(&mut data)?;
}
if format > 21 {
cab.read_u32_be()?;
}
}
// let num_objects = cab.read_u32_le()?;
// debug!("num_objects={num_objects}");
// for _ in 0..num_objects {}
}
Ok(())
}
|