2 CANON BX-1 Emulator 'eBX-1'
4 Author : Takeda.Toshiya
11 #include "../../emu.h"
12 #include "../device.h"
15 #include "../mc6800.h"
17 #include "../memory.h"
19 #include "../mc6843.h"
20 #include "../mc6844.h"
24 #include "../debugger.h"
37 // ----------------------------------------------------------------------------
39 // ----------------------------------------------------------------------------
41 VM::VM(EMU_TEMPLATE* parent_emu) : VM_TEMPLATE(parent_emu)
44 first_device = last_device = NULL;
45 dummy = new DEVICE(this, emu); // must be 1st device
46 event = new EVENT(this, emu); // must be 2nd device
48 cpu = new MC6800(this, emu);
49 io = new IO(this, emu);
50 memory = new MEMORY(this, emu);
51 fdc = new MC6843(this, emu); // HD46503
52 fdc->set_context_noise_seek(new NOISE(this, emu));
53 fdc->set_context_noise_head_down(new NOISE(this, emu));
54 fdc->set_context_noise_head_up(new NOISE(this, emu));
55 dma = new MC6844(this, emu); // HD46504
57 display = new DISPLAY(this, emu);
58 floppy = new FLOPPY(this, emu);
59 keyboard = new KEYBOARD(this, emu);
60 printer = new PRINTER(this, emu);
63 event->set_context_cpu(cpu);
64 event->set_context_sound(fdc->get_context_noise_seek());
65 event->set_context_sound(fdc->get_context_noise_head_down());
66 event->set_context_sound(fdc->get_context_noise_head_up());
68 fdc->set_context_drq(dma, SIG_MC6844_TX_RQ_0, 1);
69 dma->set_context_memory(memory);
70 dma->set_context_ch0(fdc);
71 dma->set_context_ch1(display);
73 display->set_context_dma(dma);
74 floppy->set_context_fdc(fdc);
75 printer->set_context_ram(ram);
78 cpu->set_context_mem(memory);
80 DEBUGGER *debugger = new DEBUGGER(this, emu);
81 cpu->set_context_debugger(debugger);
83 debugger->add_symbol(0x015c, _T("VRAM_TOP"));
85 debugger->add_symbol(0xe121, _T("KEY_DOWN"));
86 debugger->add_symbol(0xe122, _T("KEY_UP"));
87 debugger->add_symbol(0xe140, _T("DMA[0].ADDR_HI"));
88 debugger->add_symbol(0xe141, _T("DMA[0].ADDR_LO"));
89 debugger->add_symbol(0xe142, _T("DMA[0].COUNT_HI"));
90 debugger->add_symbol(0xe143, _T("DMA[0].COUNT_LO"));
91 debugger->add_symbol(0xe144, _T("DMA[1].ADDR_HI"));
92 debugger->add_symbol(0xe145, _T("DMA[1].ADDR_LO"));
93 debugger->add_symbol(0xe146, _T("DMA[1].COUNT_HI"));
94 debugger->add_symbol(0xe147, _T("DMA[1].COUNT_LO"));
95 debugger->add_symbol(0xe148, _T("DMA[2].ADDR_HI"));
96 debugger->add_symbol(0xe149, _T("DMA[2].ADDR_LO"));
97 debugger->add_symbol(0xe14a, _T("DMA[2].COUNT_HI"));
98 debugger->add_symbol(0xe14b, _T("DMA[2].COUNT_LO"));
99 debugger->add_symbol(0xe14c, _T("DMA[3].ADDR_HI"));
100 debugger->add_symbol(0xe14d, _T("DMA[3].ADDR_LO"));
101 debugger->add_symbol(0xe14e, _T("DMA[3].COUNT_HI"));
102 debugger->add_symbol(0xe14f, _T("DMA[3].COUNT_LO"));
103 debugger->add_symbol(0xe150, _T("DMA[0].CH_CTRL"));
104 debugger->add_symbol(0xe151, _T("DMA[1].CH_CTRL"));
105 debugger->add_symbol(0xe152, _T("DMA[2].CH_CTRL"));
106 debugger->add_symbol(0xe153, _T("DMA[3].CH_CTRL"));
107 debugger->add_symbol(0xe154, _T("DMA.PRIORITY_CTRL"));
108 debugger->add_symbol(0xe155, _T("DMA.INTERRUPT_CTRL"));
109 debugger->add_symbol(0xe156, _T("DMA.DATA_CHAIN"));
110 debugger->add_symbol(0xe180, _T("FDC.DATA"));
111 debugger->add_symbol(0xe181, _T("FDC.CUR_TRACK"));
112 debugger->add_symbol(0xe182, _T("FDC.INTSTAT_CMD"));
113 debugger->add_symbol(0xe183, _T("FDC.STATA_SETUP"));
114 debugger->add_symbol(0xe184, _T("FDC.STATB_SECTOR"));
115 debugger->add_symbol(0xe185, _T("FDC.GEN_COUNT"));
116 debugger->add_symbol(0xe186, _T("FDC.CRC_CTRL"));
117 debugger->add_symbol(0xe187, _T("FDC.LOGIC_TRACK"));
118 debugger->add_symbol(0xe18a, _T("FDD.MOTOR_ON")); // ???
122 memset(ram, 0x00, sizeof(ram));
123 memset(cart_5000, 0xff, sizeof(cart_5000));
124 memset(cart_6000, 0xff, sizeof(cart_6000));
125 memset(cart_7000, 0xff, sizeof(cart_7000));
126 memset(cart_8000, 0xff, sizeof(cart_8000));
127 memset(bios_9000, 0xff, sizeof(bios_9000));
128 memset(bios_f000, 0xff, sizeof(bios_f000));
130 memory->read_bios(_T("CART_5000.ROM"), cart_5000, sizeof(cart_5000));
131 memory->read_bios(_T("CART_6000.ROM"), cart_6000, sizeof(cart_6000));
132 memory->read_bios(_T("CART_7000.ROM"), cart_7000, sizeof(cart_7000));
133 memory->read_bios(_T("CART_8000.ROM"), cart_8000, sizeof(cart_8000));
134 memory->read_bios(_T("BIOS_9000.ROM"), bios_9000, sizeof(bios_9000));
135 memory->read_bios(_T("BIOS_F000.ROM"), bios_f000, sizeof(bios_f000));
138 memory->set_memory_rw(0x0000, 0x07ff, ram + 0x0000);
140 memory->set_memory_rw(0x0000, 0x03ff, ram + 0x0000);
142 memory->set_memory_rw(0x1000, 0x4fff, ram + 0x1000);
143 memory->set_memory_r(0x5000, 0x5fff, cart_5000);
144 memory->set_memory_r(0x6000, 0x6fff, cart_6000);
145 memory->set_memory_r(0x7000, 0x7fff, cart_7000);
146 memory->set_memory_r(0x8000, 0x8fff, cart_8000);
147 memory->set_memory_r(0x9000, 0xdfff, bios_9000);
148 memory->set_memory_mapped_io_rw(0xe000, 0xefff, io);
149 memory->set_memory_r(0xf000, 0xffff, bios_f000);
152 io->set_iomap_range_r (0xe121, 0xe122, keyboard);
153 io->set_iomap_range_rw(0xe140, 0xe156, dma);
154 io->set_iomap_range_rw(0xe180, 0xe187, fdc);
155 io->set_iomap_range_rw(0xe188, 0xe18f, floppy);
156 io->set_iomap_range_rw(0xe210, 0xe212, printer); // ?????
158 //! @note Not remove below comment. 20210511 K.O
166 chain regでdma[3]→dma[1]に設定をコピー
168 dma[2] bfd5hから09hバイト メモリ→
170 c7h c7h c7h c7h e2h d9h d9h d9h d9h
172 dma[1] 15chから10hバイト メモリ→DISPLAY?
204 // io->set_iovalue_single_r(0xe189, 0);
205 // io->set_iovalue_single_r(0xe212, 0);
216 UNKNOWN: 00c0d0 IN8 e182 = ff
217 UNKNOWN: 00b618 IN8 e184 = ff
218 UNKNOWN: 00b61b IN8 e180 = ff
219 UNKNOWN: 00b61e OUT8 e180,ff
220 UNKNOWN: 00b667 OUT8 e18c,00
224 // initialize all devices
225 for(DEVICE* device = first_device; device; device = device->next_device) {
226 device->initialize();
232 // delete all devices
233 for(DEVICE* device = first_device; device;) {
234 DEVICE *next_device = device->next_device;
237 device = next_device;
241 DEVICE* VM::get_device(int id)
243 for(DEVICE* device = first_device; device; device = device->next_device) {
244 if(device->this_device_id == id) {
251 // ----------------------------------------------------------------------------
252 // drive virtual machine
253 // ----------------------------------------------------------------------------
258 for(DEVICE* device = first_device; device; device = device->next_device) {
266 cpu->write_signal(SIG_CPU_IRQ, 1, 1);
269 // ----------------------------------------------------------------------------
271 // ----------------------------------------------------------------------------
274 DEVICE *VM::get_cpu(int index)
283 // ----------------------------------------------------------------------------
285 // ----------------------------------------------------------------------------
287 void VM::draw_screen()
289 display->draw_screen();
292 // ----------------------------------------------------------------------------
294 // ----------------------------------------------------------------------------
296 void VM::initialize_sound(int rate, int samples)
298 // init sound manager
299 event->initialize_sound(rate, samples);
302 uint16_t* VM::create_sound(int* extra_frames)
304 return event->create_sound(extra_frames);
307 int VM::get_sound_buffer_ptr()
309 return event->get_sound_buffer_ptr();
312 #ifdef USE_SOUND_VOLUME
313 void VM::set_sound_device_volume(int ch, int decibel_l, int decibel_r)
316 fdc->get_context_noise_seek()->set_volume(0, decibel_l, decibel_r);
317 fdc->get_context_noise_head_down()->set_volume(0, decibel_l, decibel_r);
318 fdc->get_context_noise_head_up()->set_volume(0, decibel_l, decibel_r);
323 // ----------------------------------------------------------------------------
325 // ----------------------------------------------------------------------------
327 void VM::key_down(int code, bool repeat)
330 keyboard->key_down(code);
334 void VM::key_up(int code)
338 // ----------------------------------------------------------------------------
340 // ----------------------------------------------------------------------------
342 void VM::open_floppy_disk(int drv, const _TCHAR* file_path, int bank)
344 fdc->open_disk(drv, file_path, bank);
346 // unformatted disk image is inserted
347 if(fdc->is_disk_inserted(drv)) {
348 DISK *disk = fdc->get_disk_handler(drv);
349 bool formatted = false;
351 for(int trk = 0; trk < 35; trk++) {
352 if(disk->get_track(trk, 0)) {
359 for(int trk = 0; trk < 35; trk++) {
360 disk->format_track(trk, 0);
361 disk->track_mfm = false;
363 for(int sec = 1; sec <= 16; sec++) {
364 disk->insert_sector(trk, 0, sec, 0, false, false, 0x00, 128);
371 void VM::close_floppy_disk(int drv)
373 fdc->close_disk(drv);
376 bool VM::is_floppy_disk_inserted(int drv)
378 return fdc->is_disk_inserted(drv);
381 void VM::is_floppy_disk_protected(int drv, bool value)
383 fdc->is_disk_protected(drv, value);
386 bool VM::is_floppy_disk_protected(int drv)
388 return fdc->is_disk_protected(drv);
391 uint32_t VM::is_floppy_disk_accessed()
393 return fdc->read_signal(0);
396 bool VM::is_frame_skippable()
398 return event->is_frame_skippable();
401 void VM::update_config()
403 for(DEVICE* device = first_device; device; device = device->next_device) {
404 device->update_config();
408 #define STATE_VERSION 2
410 bool VM::process_state(FILEIO* state_fio, bool loading)
412 if(!state_fio->StateCheckUint32(STATE_VERSION)) {
415 for(DEVICE* device = first_device; device; device = device->next_device) {
416 const char *name = typeid(*device).name() + 6; // skip "class "
417 int len = (int)strlen(name);
419 if(!state_fio->StateCheckInt32(len)) {
422 if(!state_fio->StateCheckBuffer(name, len, 1)) {
425 if(!device->process_state(state_fio, loading)) {
429 state_fio->StateArray(ram, sizeof(ram), 1);