aboutsummaryrefslogtreecommitdiff
path: root/src/desktop/load_save.rs
blob: 5c3a1235671ed93a62513cc4cd990bbf4c93552e (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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use crate::io::{Audio, LoadSave, Serial};
use crate::state::GBState;
use std::fs::File;
use std::io::{Read, Write};

#[derive(Debug)]
pub struct FSLoadSave {
    rom_file: String,
    save_file: String,
    state_file: Option<String>,
}

impl FSLoadSave {
    pub fn new(rom_file: impl Into<String>, save_file: impl Into<String>) -> Self {
        Self {
            rom_file: rom_file.into(),
            save_file: save_file.into(),
            state_file: None,
        }
    }

    pub fn state_file(mut self, state_file: impl Into<String>) -> Self {
        self.state_file = Some(state_file.into());
        self
    }
}

impl LoadSave for FSLoadSave {
    type Error = std::io::Error;

    fn load_rom(&self, rom: &mut [u8]) -> Result<(), std::io::Error> {
        let mut f = File::open(&self.rom_file)?;

        f.read(rom)?;

        return Ok(());
    }

    fn load_bootrom(&self, boot_rom: &mut [u8]) -> Result<(), std::io::Error> {
        println!("MBC: {:02x}", boot_rom[0x147]);
        println!("CGB: {:02x}", boot_rom[0x143]);

        if boot_rom[0x143] == 0x80 || boot_rom[0x143] == 0xc0 {
            unimplemented!("CGB Boot rom is not implemented");
            // let bytes = include_bytes!("../assets/cgb_boot.bin");

            // self.boot_rom[..0x900].copy_from_slice(bytes);
            // self.cgb_mode = true;
            // self.display.cgb_mode = true;
        } else {
            let bytes = include_bytes!("../../assets/dmg_boot.bin");

            boot_rom[..0x100].copy_from_slice(bytes);
        }

        Ok(())
    }

    fn load_external_ram(&self, external_ram: &mut [u8]) -> Result<(), std::io::Error> {
        let mut f = File::open(&self.save_file)?;

        f.read(external_ram)?;

        println!("Save file loaded from \"{}\"!", self.save_file);

        Ok(())
    }

    fn save_external_ram(&self, external_ram: &[u8]) -> Result<(), std::io::Error> {
        let mut f = File::create(&self.save_file)?;

        f.write_all(&external_ram)?;

        println!("Save written to \"{}\"!", self.save_file);

        Ok(())
    }

    fn save_state<S: Serial, A: Audio>(&self, state: &GBState<S, A>) {
        if let Some(state_file) = &self.state_file {
            {
                let mut vram_dump_file = File::create(format!("{}.vram", state_file)).unwrap();

                for addr in 0x8000..0xa000 {
                    vram_dump_file
                        .write_all(format!("{:02X} ", state.mem.r(addr).unwrap()).as_bytes());
                }
            }

            {
                let mut wram_dump_file = File::create(format!("{}.wram", state_file)).unwrap();

                for addr in 0xc000..0xe000 {
                    wram_dump_file
                        .write_all(format!("{:02X} ", state.mem.r(addr).unwrap()).as_bytes());
                }
            }

            {
                let mut io_dump_file = File::create(format!("{}.io", state_file)).unwrap();

                for addr in 0xff00..0xff80 {
                    io_dump_file
                        .write_all(format!("{:02X} ", state.mem.r(addr).unwrap()).as_bytes());
                }
            }

            {
                let mut hram_dump_file = File::create(format!("{}.hram", state_file)).unwrap();

                for addr in 0xff80..=0xffff {
                    hram_dump_file
                        .write_all(format!("{:02X} ", state.mem.r(addr).unwrap()).as_bytes());
                }
            }
        } else {
            panic!("{:?}", self)
        }
    }
}