aboutsummaryrefslogtreecommitdiff
path: root/src/main.rs
blob: c1dcfd20e00302234434785ef18d258039293888 (plain)
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(())
}