2 FUJITSU FM-Towns Emulator 'eFMTowns'
4 Author : Kyuma Ohta <whatisthis.sowhat _at_ gmail.com>
9 2016-12-28 Copy from eFMR-50.
13 #include "../../emu.h"
14 #include "../device.h"
17 //#include "../hd46505.h"
22 #include "../i386_np21.h"
23 //#include "../i386.h"
26 #include "../mb8877.h"
27 #include "../msm58321.h"
29 #include "../pcm1bit.h"
30 #include "../harddisk.h"
31 #include "../scsi_hdd.h"
32 #include "./towns_scsi_host.h"
33 #include "../upd71071.h"
37 #include "./dictionary.h"
39 #include "./towns_memory.h"
45 //#include "mb87078.h"
47 //#include "../ym2612.h"
56 #include "../debugger.h"
62 #include "./fontroms.h"
63 #include "./joystick.h"
65 #include "./keyboard.h"
67 #include "./msdosrom.h"
69 #include "./serialrom.h"
73 #include "./planevram.h"
75 // ----------------------------------------------------------------------------
77 // ----------------------------------------------------------------------------
80 using FMTOWNS::DICTIONARY;
81 using FMTOWNS::FLOPPY;
82 using FMTOWNS::FONT_ROMS;
83 using FMTOWNS::JOYSTICK;
84 using FMTOWNS::JOYPAD;
85 using FMTOWNS::KEYBOARD;
87 using FMTOWNS::MSDOSROM;
89 using FMTOWNS::SERIAL_ROM;
90 using FMTOWNS::SYSROM;
92 using FMTOWNS::TOWNS_ICCARD;
94 using FMTOWNS::TOWNS_CDROM;
95 using FMTOWNS::TOWNS_CRTC;
96 using FMTOWNS::TOWNS_DMAC;
97 using FMTOWNS::TOWNS_MEMORY;
98 using FMTOWNS::TOWNS_SCSI_HOST;
99 using FMTOWNS::TOWNS_SPRITE;
100 using FMTOWNS::TOWNS_VRAM;
101 using FMTOWNS::PLANEVRAM;
104 VM::VM(EMU_TEMPLATE* parent_emu) : VM_TEMPLATE(parent_emu)
125 first_device = last_device = NULL;
126 dummy = new DEVICE(this, emu); // must be 1st device
127 event = new EVENT(this, emu); // must be 2nd device
129 dummy->set_device_name(_T("1st Dummy"));
130 event->set_device_name(_T("EVENT"));
133 cpu = new I386(this, emu);
135 #if defined(HAS_I386)
136 cpu->set_device_name(_T("CPU(i386)"));
137 #elif defined(HAS_I486)
138 cpu->set_device_name(_T("CPU(i486)"));
139 #elif defined(HAS_PENTIUM)
140 cpu->set_device_name(_T("CPU(Pentium)"));
144 io = new IO(this, emu);
146 crtc = new TOWNS_CRTC(this, emu);
147 cdrom = new TOWNS_CDROM(this, emu);
149 memory = new TOWNS_MEMORY(this, emu);
150 vram = new TOWNS_VRAM(this, emu);
151 sprite = new TOWNS_SPRITE(this, emu);
152 sysrom = new SYSROM(this, emu);
153 msdosrom = new MSDOSROM(this, emu);
154 fontrom = new FONT_ROMS(this, emu);
155 dictionary = new DICTIONARY(this, emu);
156 #if defined(HAS_20PIX_FONTS)
157 fontrom_20pix = new FONT_ROM_20PIX(this, emu);
159 serialrom = new SERIAL_ROM(this, emu);
161 adpcm = new ADPCM(this, emu);
162 // mixer = new MIXER(this, emu); // Pseudo mixer.
164 planevram = new PLANEVRAM(this, emu);
166 adc = new AD7820KR(this, emu);
167 rf5c68 = new RF5C68(this, emu);
168 // e_volume[0] = new MB87878(this, emu);
169 // e_volume[1] = new MB87878(this, emu);
171 sio = new I8251(this, emu);
172 pit0 = new I8253(this, emu);
173 pit1 = new I8253(this, emu);
174 pic = new I8259(this, emu);
175 fdc = new MB8877(this, emu);
176 rtc = new MSM58321(this, emu);
177 beep = new PCM1BIT(this, emu);
178 opn2 = new YM2612(this, emu);
180 seek_sound = new NOISE(this, emu);
181 head_up_sound = new NOISE(this, emu);
182 head_down_sound = new NOISE(this, emu);
184 // scsi_host = new TOWNS_SCSI_HOST(this, emu);
185 scsi_host = new SCSI_HOST(this, emu);
187 for(int i = 0; i < 7; i++) {
190 #if defined(USE_HARD_DISK)
191 for(int i = 0; i < USE_HARD_DISK; i++) {
192 scsi_hdd[i] = new SCSI_HDD(this, emu);
193 scsi_hdd[i]->set_device_name(_T("SCSI Hard Disk Drive #%d"), i + 1);
194 scsi_hdd[i]->scsi_id = i ;
195 scsi_hdd[i]->set_disk_handler(0, new HARDDISK(emu));
196 scsi_hdd[i]->set_context_interface(scsi_host);
197 my_sprintf_s(scsi_hdd[i]->vendor_id, 9, "FUJITSU");
198 my_sprintf_s(scsi_hdd[i]->product_id, 17, "SCSI-HDD");
199 scsi_host->set_context_target(scsi_hdd[i]);
202 dma = new TOWNS_DMAC(this, emu);
203 extra_dma = new TOWNS_DMAC(this, emu);
205 floppy = new FLOPPY(this, emu);
206 keyboard = new KEYBOARD(this, emu);
207 joystick = new JOYSTICK(this, emu);
208 scsi = new SCSI(this, emu);
209 timer = new TIMER(this, emu);
211 iccard1 = new TOWNS_ICCARD(this, emu);
213 iccard2 = new TOWNS_ICCARD(this, emu);
217 joypad[0] = new JOYPAD(this, emu);
218 joypad[1] = new JOYPAD(this, emu);
219 mouse = new MOUSE(this, emu);
221 uint16_t machine_id = 0x0100; // FM-Towns1
222 uint16_t cpu_id = 0x0001; // i386DX
223 uint32_t cpu_clock = 16000 * 1000; // 16MHz
224 #if defined(_FMTOWNS1_2ND_GEN)
225 machine_id = 0x0200; // 1F/2F/1H/2H
226 #elif defined(_FMTOWNS_UX_VARIANTS)
227 machine_id = 0x0300; // UX10/20/40
228 cpu_id = 0x0003; // i386SX
229 #elif defined(_FMTOWNS1_3RD_GEN)
230 machine_id = 0x0400; // 10F/20F.40H/80H
231 #elif defined(_FMTOWNS2_CX_VARIANTS)
232 machine_id = 0x0500; // CX10/20/30/40
233 #elif defined(_FMTOWNS_UG_VARIANTS)
234 machine_id = 0x0600; // UG10/20/40/80
235 cpu_id = 0x0003; // i386SX
236 cpu_clock = 20000 * 1000; // 20MHz
237 #elif defined(_FMTOWNS_HR_VARIANTS)
239 cpu_id = 0x0002; // i486SX
240 cpu_clock = 20000 * 1000; // 20MHz
241 #elif defined(_FMTOWNS_HG_VARIANTS)
243 cpu_clock = 20000 * 1000; // 20MHz
244 #elif defined(_FMTOWNS_SG_VARIANTS)
245 machine_id = 0x0800; // OK?
246 #elif defined(_FMTOWNS_SR_VARIANTS)
247 machine_id = 0x0700; // OK?
248 cpu_id = 0x0002; // i486SX
249 cpu_clock = 20000 * 1000; // 20MHz
250 #elif defined(_FMTOWNS_UR_VARIANTS)
251 machine_id = 0x0900; // UR10/20/40/80
252 cpu_id = 0x0002; // i486DX
253 cpu_clock = 20000 * 1000; // ToDo: Correct frequency.
254 #elif defined(_FMTOWNS_MA_VARIANTS)
255 machine_id = 0x0b00; // OK?
256 cpu_id = 0x0002; // i486SX
257 cpu_clock = 33000 * 1000; // 33MHz
258 #elif defined(_FMTOWNS_ME_VARIANTS)
259 machine_id = 0x0d00; // OK?
260 cpu_id = 0x0002; // i486SX
261 cpu_clock = 25000 * 1000; // 25MHz
262 #elif defined(_FMTOWNS_MF_VARIANTS)
263 machine_id = 0x0f00; // OK?
264 cpu_id = 0x0002; // i486SX
265 cpu_clock = 33000 * 1000; // 33MHz
266 #elif defined(_FMTOWNS_MX_VARIANTS)
267 machine_id = 0x0c00; // OK?
268 cpu_id = 0x0002; // i486DX (With FPU?)
269 cpu_clock = 66000 * 1000; // 66MHz
270 #elif defined(_FMTOWNS_HC_VARIANTS)
272 // From FMTowns::MachineID() of TSUGARU,
273 // git 83d4ec2309ac9fcbb8c01f26061ff0d49c5321e4.
274 machine_id = 0x1100; // OK?
275 cpu_id = 0x0002; // Pentium (With FPU?)
276 cpu_clock = 50000 * 1000; // ToDo: Correctness frequency.
278 // ToDo: Pentium Model (After HB).
282 event->set_frames_per_sec(FRAMES_PER_SEC);
283 event->set_lines_per_frame(LINES_PER_FRAME);
285 set_machine_type(machine_id, cpu_id);
287 event->set_context_cpu(cpu, cpu_clock);
294 // Use pseudo mixer instead of event.Due to using ADC.
299 //line_mix_ch = mixer->set_context_sound(line_in);
300 //modem_mix_ch = mixer->set_context_sound(modem_in);
301 //mic_mix_ch = mixer->set_context_sound(mic_in);
302 beep_mix_ch = mixer->set_context_sound(beep);
303 pcm_mix_ch = mixer->set_context_sound(rf5c68);
304 opn2_mix_ch = mixer->set_context_sound(opn2);
305 cdc_mix_ch = mixer->set_context_sound(cdrom);
306 mixer->set_interpolate_filter_freq(pcm_mix_ch, 4000); // channel, freq; disable if freq <= 0.
307 event->set_context_sound(mixer);
309 // Temporally not use mixer.
310 event->set_context_sound(beep);
311 event->set_context_sound(opn2);
312 event->set_context_sound(rf5c68);
313 event->set_context_sound(cdrom);
315 fdc->set_context_noise_seek(seek_sound);
316 fdc->set_context_noise_head_down(head_down_sound);
317 fdc->set_context_noise_head_up(head_up_sound);
318 event->set_context_sound(seek_sound);
319 event->set_context_sound(head_down_sound);
320 event->set_context_sound(head_up_sound);
323 pit0->set_context_debugger(new DEBUGGER(this, emu));
324 pit1->set_context_debugger(new DEBUGGER(this, emu));
326 pit0->set_context_ch0(timer, SIG_TIMER_CH0, 1);
327 pit0->set_context_ch1(timer, SIG_TIMER_CH1, 1);
328 pit0->set_context_ch2(beep, SIG_PCM1BIT_SIGNAL, 1);
329 pit0->set_constant_clock(0, 307200);
330 pit0->set_constant_clock(1, 307200);
331 pit0->set_constant_clock(2, 307200);
332 pit1->set_constant_clock(0, 1229900);
333 pit1->set_constant_clock(1, 1229900);
334 pit1->set_constant_clock(2, 1229900);
335 // pic->set_context_cpu(cpu);
336 pic->set_context_cpu(memory);
337 fdc->set_context_irq(floppy, SIG_FLOPPY_IRQ, 1);
338 rtc->set_context_data(timer, SIG_TIMER_RTC, 0x0f, 0);
339 rtc->set_context_busy(timer, SIG_TIMER_RTC_BUSY, 0x80);
340 scsi_host->set_context_irq(scsi, SIG_SCSI_IRQ, 1);
341 scsi_host->set_context_drq(scsi, SIG_SCSI_DRQ, 1);
342 scsi_host->set_context_drq(keyboard, SIG_KEYBOARD_BOOTSEQ_END, 1);
344 dma->set_context_memory(memory);
345 dma->set_context_ch0(fdc);
346 dma->set_context_ch1(scsi_host);
347 //dma->set_context_ch2(printer);
348 dma->set_context_ch3(cdrom);
349 dma->set_context_tc1(scsi, SIG_SCSI_EOT, 0xffffffff);
350 dma->set_context_tc3(cdrom, SIG_TOWNS_CDROM_DMAINT, 0xffffffff);
351 dma->set_context_ack3(cdrom, SIG_TOWNS_CDROM_DMAACK, 0xffffffff);
353 dma->set_context_ube1(scsi_host, SIG_SCSI_16BIT_BUS, 0x02);
355 dma->set_context_child_dma(extra_dma);
357 floppy->set_context_fdc(fdc);
359 sprite->set_context_vram(vram);
360 sprite->set_context_font(fontrom);
361 sprite->set_context_crtc(crtc);
363 sprite->set_context_debugger(new DEBUGGER(this, emu));
366 vram->set_context_sprite(sprite);
367 vram->set_context_crtc(crtc);
369 planevram->set_context_vram(vram);
370 planevram->set_context_sprite(sprite);
371 planevram->set_context_crtc(crtc);
373 crtc->set_context_sprite(sprite);
374 crtc->set_context_vram(vram);
375 crtc->set_context_font(fontrom);
377 //e_volume[0]->set_context_ch0(line_in, MB87878_VOLUME_LEFT);
378 //e_volume[0]->set_context_ch1(line_in, MB87878_VOLUME_RIGHT);
379 //e_volume[0]->set_context_ch2(NULL, MB87878_VOLUME_LEFT);
380 //e_volume[0]->set_context_ch3(NULL, MB87878_VOLUME_RIGHT);
381 // e_volume[1]->set_context_ch0(cdrom, MB87878_VOLUME_LEFT);
382 // e_volume[1]->set_context_ch1(cdrom, MB87878_VOLUME_RIGHT);
383 //e_volume[1]->set_context_ch2(mic, MB87878_VOLUME_LEFT | MB87878_VOLUME_RIGHT);
384 //e_volume[1]->set_context_ch3(modem, MB87878_VOLUME_LEFT | MB87878_VOLUME_RIGHT);
386 memory->set_context_cpu(cpu);
387 memory->set_context_dmac(dma);
388 memory->set_context_vram(vram);
389 memory->set_context_planevram(planevram);
390 memory->set_context_crtc(crtc);
391 memory->set_context_system_rom(sysrom);
392 memory->set_context_msdos(msdosrom);
393 memory->set_context_dictionary(dictionary);
394 memory->set_context_font_rom(fontrom);
395 memory->set_context_timer(timer);
396 memory->set_context_serial_rom(serialrom);
397 memory->set_context_sprite(sprite);
398 memory->set_context_pcm(rf5c68);
399 memory->set_context_iccard(iccard1, 0);
400 memory->set_context_iccard(iccard2, 1);
402 adpcm->set_context_opn2(opn2);
403 adpcm->set_context_rf5c68(rf5c68);
404 adpcm->set_context_adc(adc);
406 rf5c68->set_context_interrupt_boundary(adpcm, SIG_ADPCM_WRITE_INTERRUPT, 0xffffffff);
408 rf5c68->set_context_debugger(new DEBUGGER(this, emu));
410 opn2->set_context_irq(adpcm, SIG_ADPCM_OPX_INTR, 0xffffffff);
412 adc->set_sample_rate(19200);
413 adc->set_sound_bank(-1);
414 adc->set_context_interrupt(adpcm, SIG_ADPCM_ADC_INTR, 0xffffffff);
416 scsi->set_context_dma(dma);
417 scsi->set_context_host(scsi_host);
418 scsi->set_context_pic(pic);
419 timer->set_context_pcm(beep);
420 timer->set_context_rtc(rtc);
421 timer->set_context_halt_line(cpu, SIG_CPU_HALTREQ, 0xffffffff);
423 joystick->set_context_mouse(mouse, SIG_MOUSE_STROBE, 0xff);
425 joystick->set_context_enable0(joypad[0], SIG_JOYPAD_ENABLE, 0xffffffff);
426 joystick->set_context_enable1(joypad[1], SIG_JOYPAD_ENABLE, 0xffffffff);
427 joystick->set_context_mask(joypad[0], SIG_JOYPAD_SELECT_BUS, 0x10); // Mouse0 or joypad0
428 joystick->set_context_mask(joypad[1], SIG_JOYPAD_SELECT_BUS, 0x20); // Mouse1 or joypad1
429 joystick->set_context_query(joypad[0], SIG_JOYPAD_QUERY, 0x1);
430 joystick->set_context_query(joypad[1], SIG_JOYPAD_QUERY, 0x2);
432 joypad[0]->set_context_port_num(0);
433 joypad[1]->set_context_port_num(1);
434 joypad[0]->set_context_data(joystick, SIG_JOYPORT_CH0 | SIG_JOYPORT_TYPE_2BUTTONS | SIG_JOYPORT_DATA, 0xffffffff);
435 joypad[1]->set_context_data(joystick, SIG_JOYPORT_CH1 | SIG_JOYPORT_TYPE_2BUTTONS | SIG_JOYPORT_DATA, 0xffffffff);
437 joypad[0]->set_context_com(joystick, SIG_JOYPORT_CH0 | SIG_JOYPORT_TYPE_2BUTTONS | SIG_JOYPORT_COM, 0xffffffff);
438 joypad[1]->set_context_com(joystick, SIG_JOYPORT_CH1 | SIG_JOYPORT_TYPE_2BUTTONS | SIG_JOYPORT_COM, 0xffffffff);
439 mouse->set_context_com(joystick);
442 cpu->set_context_mem(memory);
443 cpu->set_context_io(io);
444 cpu->set_context_intr(pic);
445 cpu->set_context_dma(dma);
446 cpu->set_context_bios(NULL);
447 cpu->set_context_extreset(memory, SIG_FMTOWNS_NOTIFY_RESET, 0xffffffff);
449 cpu->set_context_debugger(new DEBUGGER(this, emu));
454 // IRQ2 : USART (ToDo)
455 // IRQ3 : EXTRA USART (ToDo)
456 // IRQ4 : EXTRA I/O (Maybe not implement)
457 // IRQ5 : EXTRA I/O (Maybe not implement)
459 // IRQ7 : Deisy chain (to IRQ8 - 15)
460 timer->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR0, 0xffffffff);
461 keyboard->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR1, 0xffffffff);
462 floppy->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR6, 0xffffffff);
464 // IRQ8 : SCSI (-> scsi.cpp)
466 // IRQ10 : EXTRA I/O (Maybe not implement)
468 // IRQ12 : PRINTER (ToDo)
469 // IRQ13 : ADPCM AND OPN2 (Route to adpcm.cpp)
470 // IRQ14 : EXTRA I/O (Maybe not implement)
472 cdrom->set_context_mpuint_line(pic, SIG_I8259_CHIP1 | SIG_I8259_IR1, 0xffffffff);
473 crtc->set_context_vsync(pic, SIG_I8259_CHIP1 | SIG_I8259_IR3, 0xffffffff);
474 adpcm->set_context_intr_line(pic, SIG_I8259_CHIP1 | SIG_I8259_IR5, 0xffffffff);
477 // DMA1 : SCSI (-> scsi.cpp)
478 // DMA2 : PRINTER (ToDo)
480 // EXTRA DMA0 : EXTRA SLOT (Maybe not implement)
481 // EXTRA DMA1 : Reserved
482 // EXTRA DMA2 : Reserved
483 // EXTRA DMA3 : Reserved
484 fdc->set_context_drq(dma, SIG_UPD71071_CH0, 1);
485 fdc->set_context_drq(keyboard, SIG_KEYBOARD_BOOTSEQ_END, 1);
486 cdrom->set_context_drq_line(dma, SIG_UPD71071_CH3, 0xff);
487 cdrom->set_context_drq_line(keyboard, SIG_KEYBOARD_BOOTSEQ_END, 1);
489 // NMI0 : KEYBOARD (RAS)
490 // NMI1 : Extra SLOT (Maybe not implement)
491 keyboard->set_context_nmi_line(memory, SIG_CPU_NMI, 0xffffffff);
493 cdrom->set_context_dmac(dma);
494 // For Debugging, will remove 20200822 K.O
495 cdrom->set_context_cpu(cpu);
498 io->set_iowait_range_rw(0x0000, 0xffff, 6); // ToDo: May variable wait.
500 io->set_iomap_alias_rw (0x0000, pic, I8259_ADDR_CHIP0 | 0);
501 io->set_iomap_alias_rw (0x0002, pic, I8259_ADDR_CHIP0 | 1);
502 io->set_iomap_alias_rw (0x0010, pic, I8259_ADDR_CHIP1 | 0);
503 io->set_iomap_alias_rw (0x0012, pic, I8259_ADDR_CHIP1 | 1);
505 io->set_iomap_range_rw (0x0020, 0x0025, memory);
506 io->set_iomap_range_rw (0x0026, 0x0027, timer); // Freerun counter
507 io->set_iomap_single_rw(0x0028, memory);
509 io->set_iomap_range_r (0x0030, 0x0031, memory); // cpu id / machine id
510 io->set_iomap_single_rw(0x0032, memory); // serial rom (routed from memory)
511 io->set_iomap_single_r (0x0034, scsi); // ENABLE/ UNABLE to WORD DMA for SCSI
513 io->set_iomap_alias_rw(0x0040, pit0, 0);
514 io->set_iomap_alias_rw(0x0042, pit0, 1);
515 io->set_iomap_alias_rw(0x0044, pit0, 2);
516 io->set_iomap_alias_rw(0x0046, pit0, 3);
517 io->set_iomap_alias_rw(0x0050, pit1, 0);
518 io->set_iomap_alias_rw(0x0052, pit1, 1);
519 io->set_iomap_alias_rw(0x0054, pit1, 2);
520 io->set_iomap_alias_rw(0x0056, pit1, 3);
522 io->set_iomap_single_rw(0x0060, timer); // Beep and interrupts register
523 io->set_iomap_single_rw(0x0068, timer); // Interval timer register2 (after Towns 10F).
524 io->set_iomap_single_rw(0x006a, timer); // Interval timer register2 (after Towns 10F).
525 io->set_iomap_single_rw(0x006b, timer); // Interval timer register2 (after Towns 10F).
526 io->set_iomap_single_rw(0x006c, timer); // 1uS wait register (after Towns 10F).
528 io->set_iomap_single_rw(0x0070, timer); // RTC DATA
529 io->set_iomap_single_w (0x0080, timer); // RTC COMMAND
531 io->set_iomap_range_rw (0x00a0, 0x00af, dma);
532 io->set_iomap_range_rw (0x00b0, 0x00bf, extra_dma);
534 io->set_iomap_single_rw(0x00c0, memory); // CACHE CONTROLLER
535 io->set_iomap_single_rw(0x00c2, memory); // CACHE CONTROLLER
537 io->set_iomap_alias_rw (0x0200, fdc, 0); // STATUS/COMMAND
538 io->set_iomap_alias_rw (0x0202, fdc, 1); // TRACK
539 io->set_iomap_alias_rw (0x0204, fdc, 2); // SECTOR
540 io->set_iomap_alias_rw (0x0206, fdc, 3); // DATA
541 io->set_iomap_single_rw(0x0208, floppy); // DRIVE STATUS / DRIVE CONTROL
542 io->set_iomap_single_rw(0x020c, floppy); // DRIVE SELECT
543 io->set_iomap_single_r (0x020d, floppy); // FDDVEXT (after HG/HR).
544 io->set_iomap_single_rw(0x020e, floppy); // Towns drive SW
546 io->set_iomap_range_rw (0x0400, 0x0404, memory); // System Status
547 // io->set_iomap_range_rw (0x0406, 0x043f, memory); // Reserved
549 io->set_iomap_range_rw(0x0440, 0x0443, crtc); // CRTC
550 io->set_iomap_range_rw(0x0448, 0x044f, crtc); // VIDEO OUT (CRTC)
552 io->set_iomap_range_rw(0x0450, 0x0452, sprite); // SPRITE
554 io->set_iomap_single_rw(0x0458, vram); // VRAM ACCESS CONTROLLER (ADDRESS)
555 io->set_iomap_range_rw (0x045a, 0x045b, vram); // VRAM ACCESS CONTROLLER (DATA)
557 io->set_iomap_single_rw(0x0480, memory); // MEMORY REGISTER
558 io->set_iomap_single_rw(0x0484, dictionary); // Dictionary
560 io->set_iomap_alias_r(0x48a, iccard1, 0); //
561 //io->set_iomap_alias_rw(0x490, memory_card); // After Towns2
562 //io->set_iomap_alias_rw(0x491, memory_card); // After Towns2
564 io->set_iomap_range_rw(0x04c0, 0x04c6, cdrom); // CDROM
565 io->set_iomap_range_r (0x04cc, 0x04cd, cdrom); // CDROM
568 io->set_iomap_single_r(0x04d0, joystick); // Pad1
569 io->set_iomap_single_r(0x04d2, joystick); // Pad 2
570 io->set_iomap_single_w(0x04d6, joystick); // Pad out
572 io->set_iomap_single_rw(0x04d5, adpcm); // mute
574 io->set_iomap_alias_rw(0x04d8, opn2, 0); // STATUS(R)/Addrreg 0(W)
575 io->set_iomap_alias_w (0x04da, opn2, 1); // Datareg 0(W)
576 io->set_iomap_alias_w (0x04dc, opn2, 2); // Addrreg 1(W)
577 io->set_iomap_alias_w (0x04de, opn2, 3); // Datareg 1(W)
579 // io->set_iomap_alias_rw(0x04e0, e_volume[0], 0);
580 // io->set_iomap_alias_rw(0x04e1, e_volume[0], 1);
581 // io->set_iomap_alias_rw(0x04e2, e_volume[1], 0);
582 // io->set_iomap_alias_rw(0x04e3, e_volume[1], 1);
585 io->set_iomap_range_rw(0x04e7, 0x04ec, adpcm); // A/D SAMPLING DATA REG
586 io->set_iomap_range_rw(0x04f0, 0x04f8, rf5c68); // A/D SAMPLING DATA REG
588 io->set_iomap_single_rw(0x05c0, memory); // NMI MASK
589 io->set_iomap_single_r (0x05c2, memory); // NMI STATUS
590 io->set_iomap_single_r (0x05c8, sprite); // TVRAM EMULATION
591 io->set_iomap_single_w (0x05ca, crtc); // VSYNC INTERRUPT
593 io->set_iomap_single_rw(0x05e0, memory); // Hidden MEMORY WAIT REGISTER from AB.COM (Towns 1/2)
594 io->set_iomap_single_rw(0x05e2, memory); // Hidden MEMORY WAIT REGISTER from AB.COM (After Towns 1H/1F/2H/2F )
595 io->set_iomap_single_rw(0x05e8, memory); // RAM capacity register.(later Towns1H/2H/1F/2F).
596 io->set_iomap_single_rw(0x05ec, memory); // RAM Wait register , ofcially after Towns2, but exists after Towns1H.
597 io->set_iomap_single_r (0x05ed, memory); // Maximum clock register (After HR/HG).
598 io->set_iomap_single_rw(0x05ee, vram); // VRAM CACHE CONTROLLER
600 io->set_iomap_single_rw(0x0600, keyboard);
601 io->set_iomap_single_rw(0x0602, keyboard);
602 io->set_iomap_single_rw(0x0604, keyboard);
604 //io->set_iomap_single_rw(0x0800, printer);
605 //io->set_iomap_single_rw(0x0802, printer);
606 //io->set_iomap_single_rw(0x0804, printer);
608 io->set_iomap_alias_rw (0x0a00, sio, 0);
609 io->set_iomap_alias_rw (0x0a02, sio, 1);
610 // io->set_iomap_single_r (0x0a04, serial);
611 // io->set_iomap_single_r (0x0a06, serial);
612 // io->set_iomap_single_w (0x0a08, serial);
613 // io->set_iomap_single_rw(0x0a0a, modem);
615 io->set_iomap_single_rw(0x0c30, scsi);
616 io->set_iomap_single_rw(0x0c32, scsi);
617 io->set_iomap_single_r (0x0c34, scsi);
619 io->set_iomap_range_rw (0x3000, 0x3fff, dictionary); // CMOS
621 io->set_iomap_range_rw (0xfd90, 0xfda0, crtc); // Palette and CRTC
622 io->set_iomap_single_r (0xfda2, crtc); // CRTC
623 io->set_iomap_single_rw(0xfda4, memory); // memory
625 io->set_iomap_range_rw (0xff80, 0xff83, planevram); // MMIO
626 io->set_iomap_single_r (0xff84, planevram); // MMIO
627 io->set_iomap_single_rw(0xff86, planevram); // MMIO
628 io->set_iomap_single_rw(0xff88, memory); // MMIO
629 io->set_iomap_range_rw (0xff94, 0xff99, memory); // MMIO
630 io->set_iomap_range_r (0xff9c, 0xff9d, memory); // MMIO
631 io->set_iomap_single_rw(0xff9e, memory); // MMIO
632 io->set_iomap_single_rw(0xffa0, planevram); // MMIO
634 // io->set_iomap_range_w (0xff94, 0xff95, fontrom);
635 // io->set_iomap_range_r (0xff96, 0xff97, fontrom);
637 // Vram allocation may be before initialize().
638 // initialize all devices
639 #if defined(__GIT_REPO_VERSION)
640 strncpy(_git_revision, __GIT_REPO_VERSION, sizeof(_git_revision) - 1);
642 // ToDo : Use config framework
643 int exram_size = config.current_ram_size;
645 if(machine_id < 0x0200) { // Model1 - 2H
647 } else if(machine_id == 0x0500) { // CX
649 } else if(machine_id < 0x0700) { // 10F,20H
651 } else if(machine_id == 0x0800) { // HG
657 if(exram_size < MIN_RAM_SIZE) {
658 exram_size = MIN_RAM_SIZE;
661 memory->set_extra_ram_size(exram_size);
663 #if defined(WITH_I386SX)
664 cpu->device_model = INTEL_80386;
665 #elif defined(WITH_I486SX)
666 cpu->device_model = INTEL_I486SX;
667 #elif defined(WITH_I486DX)
668 cpu->device_model = INTEL_I486DX;
669 #elif defined(WITH_PENTIUM)
670 cpu->device_model = INTEL_PENTIUM;
673 cpu->device_model = INTEL_80386;
676 for(DEVICE* device = first_device; device; device = device->next_device) {
677 device->initialize();
679 // cpu->set_address_mask(0xffffffff);
684 // delete all devices
685 for(DEVICE* device = first_device; device;) {
686 DEVICE *next_device = device->next_device;
687 // printf("DEVID=%d\n", device->this_device_id);
690 device = next_device;
694 DEVICE* VM::get_device(int id)
696 for(DEVICE* device = first_device; device; device = device->next_device) {
697 if(device->this_device_id == id) {
704 void VM::set_machine_type(uint16_t machine_id, uint16_t cpu_id)
707 memory->set_cpu_id(cpu_id);
708 memory->set_machine_id(machine_id);
711 crtc->set_cpu_id(cpu_id);
712 crtc->set_machine_id(machine_id);
715 timer->set_cpu_id(cpu_id);
716 timer->set_machine_id(machine_id);
719 cdrom->set_cpu_id(cpu_id);
720 cdrom->set_machine_id(machine_id);
723 scsi->set_cpu_id(cpu_id);
724 scsi->set_machine_id(machine_id);
726 if(serialrom != NULL) {
727 serialrom->set_cpu_id(cpu_id);
728 serialrom->set_machine_id(machine_id);
731 floppy->set_cpu_id(cpu_id);
732 floppy->set_machine_id(machine_id);
734 #if defined(HAS_20PIX_FONTS)
735 if(fontrom_20pix != NULL) {
736 fontrom_20pix->set_cpu_id(cpu_id);
737 fontrom_20pix->set_machine_id(machine_id);
741 vram->set_cpu_id(cpu_id);
742 vram->set_machine_id(machine_id);
748 // ----------------------------------------------------------------------------
749 // drive virtual machine
750 // ----------------------------------------------------------------------------
756 for(DEVICE* device = first_device; device; device = device->next_device) {
759 // cpu->set_address_mask(0xffffffff);
762 void VM::special_reset(int num)
767 for(DEVICE* device = first_device; device; device = device->next_device) {
770 keyboard->special_reset(num);
772 // cpu->set_address_mask(0xffffffff);
780 // ----------------------------------------------------------------------------
782 // ----------------------------------------------------------------------------
785 DEVICE *VM::get_cpu(int index)
794 // ----------------------------------------------------------------------------
796 // ----------------------------------------------------------------------------
798 void VM::draw_screen()
803 uint32_t VM::is_floppy_disk_accessed()
805 uint32_t val = fdc->read_signal(0);
808 keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
815 // ----------------------------------------------------------------------------
817 // ----------------------------------------------------------------------------
819 void VM::initialize_sound(int rate, int samples)
822 // init sound manager
823 event->initialize_sound(rate, samples);
826 beep->initialize_sound(rate, 8000);
829 // MASTER CLOCK MAYBE 600KHz * 12 = 7200KHz .
830 // From FM-Towns Technical Databook (Rev.2), Page 201
831 opn2->initialize_sound(rate, (int)(600.0e3 * 12.0) , samples, 0.0, 0.0);
832 //opn2->initialize_sound(rate, (int)(8000.0e3) , samples, 0.0, 0.0);
833 //opn2->initialize_sound(rate, (int)(600.0e3 * 6.0) , samples, 0.0, 0.0);
836 rf5c68->initialize_sound(rate, samples);
838 // add_sound_in_source() must add after per initialize_sound().
839 adc_in_ch = event->add_sound_in_source(rate, samples, 2);
840 adc->set_sample_rate(19200);
841 adc->set_sound_bank(adc_in_ch);
843 mixer->set_context_out_line(adc_in_ch);
844 mixer->set_context_sample_out(adc_in_ch, rate, samples); // Must be 2ch.
845 // ToDo: Check recording sample rate & channels.
846 mic_in_ch = event->add_sound_in_source(rate, samples, 2);
847 mixer->set_context_mic_in(mic_in_ch, rate, samples);
849 line_in_ch = event->add_sound_in_source(rate, samples, 2);
850 mixer->set_context_line_in(line_in_ch, rate, samples);
855 uint16_t* VM::create_sound(int* extra_frames)
857 return event->create_sound(extra_frames);
860 int VM::get_sound_buffer_ptr()
862 return event->get_sound_buffer_ptr();
865 void VM::clear_sound_in()
867 event->clear_sound_in_source(adc_in_ch);
868 event->clear_sound_in_source(mic_in_ch);
869 event->clear_sound_in_source(line_in_ch);
873 int VM::get_sound_in_data(int ch, int32_t* dst, int expect_samples, int expect_rate, int expect_channels)
875 if(dst == NULL) return 0;
876 if(expect_samples <= 0) return 0;
889 if(n_ch < 0) return 0;
890 int samples = event->get_sound_in_data(n_ch, dst, expect_samples, expect_rate, expect_channels);
894 // Write to event's buffer
895 int VM::sound_in(int ch, int32_t* src, int samples)
898 if(ch >= 2) return 0;
901 case 0x100: // ADC in from MIXER, not connected.
910 if(n_ch < 0) return 0;
915 ss = event->write_sound_in_buffer(n_ch, src, samples);
922 #if defined(USE_HARD_DISK)
923 void VM::open_hard_disk(int drv, const _TCHAR* file_path)
925 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
926 if(scsi_hdd[drv] != NULL) {
927 scsi_hdd[drv]->open(0, file_path, 512);
932 void VM::close_hard_disk(int drv)
934 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
935 if(scsi_hdd[drv] != NULL) {
936 scsi_hdd[drv]->close(0);
941 bool VM::is_hard_disk_inserted(int drv)
943 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
944 if(scsi_hdd[drv] != NULL) {
945 return scsi_hdd[drv]->mounted(0);
951 uint32_t VM::is_hard_disk_accessed()
955 for(int drv = 0; drv < USE_HARD_DISK; drv++) {
956 if(scsi_hdd[drv] != NULL) {
957 if(scsi_hdd[drv]->accessed(0)) {
964 keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
970 #endif // USE_HARD_DISK
972 void VM::open_compact_disc(int drv, const _TCHAR* file_path)
974 cdrom->open(file_path);
977 void VM::close_compact_disc(int drv)
982 bool VM::is_compact_disc_inserted(int drv)
984 return cdrom->mounted();
987 uint32_t VM::is_compact_disc_accessed()
989 uint32_t status = cdrom->accessed();
992 keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
999 #ifdef USE_SOUND_VOLUME
1000 void VM::set_sound_device_volume(int ch, int decibel_l, int decibel_r)
1002 #ifndef HAS_LINEIN_SOUND
1003 // if(ch >= 7) ch++;
1005 #ifndef HAS_MIC_SOUND
1006 // if(ch >= 8) ch++;
1008 #ifndef HAS_MODEM_SOUND
1009 // if(ch >= 9) ch++;
1011 #ifndef HAS_2ND_ADPCM
1012 // if(ch >= 10) ch++;
1015 if(ch == 0) { // BEEP
1016 mixer->set_volume(beep_mix_ch, decibel_l, decibel_r);
1018 else if(ch == 1) { // CD-ROM
1019 //e_volume[1]->set_volume(0, decibel_l);
1020 //e_volume[1]->set_volume(1, decibel_r);
1021 mixer->set_volume(cdc_mix_ch, decibel_l, decibel_r);
1023 else if(ch == 2) { // OPN2
1024 mixer->set_volume(opn2_mix_ch, decibel_l, decibel_r);
1026 else if(ch == 3) { // ADPCM
1027 mixer->set_volume(pcm_mix_ch, decibel_l, decibel_r);
1029 else if(ch == 4) { // LINE IN
1030 //mixer->set_volume(line_mix_ch, decibel_l, decibel_r);
1032 else if(ch == 5) { // MIC
1033 //mic->set_volume(0, (decibel_l + decibel_r) / 2);
1035 else if(ch == 6) { // MODEM
1036 //modem->set_volume(0, (decibel_l + decibel_r) / 2);
1038 #ifdef HAS_2ND_ADPCM
1039 else if(ch == 7) { // ADPCM
1040 adpcm2->set_volume(0, decibel_l, decibel_r);
1043 else if(ch == 8) { // FDD
1044 fdc->set_volume(0, decibel_l, decibel_r);
1046 else if(ch == 9) { // HDD(ToDo)
1047 fdc->set_volume(0, decibel_l, decibel_r);
1050 if(ch == 0) { // BEEP
1051 beep->set_volume(0, decibel_l, decibel_r);
1053 else if(ch == 1) { // CD-ROM
1054 cdrom->set_volume(0, decibel_l, decibel_r);
1056 else if(ch == 2) { // OPN2
1057 opn2->set_volume(0, decibel_l, decibel_r);
1059 else if(ch == 3) { // ADPCM
1060 rf5c68->set_volume(0, decibel_l, decibel_r);
1062 else if(ch == 4) { // SEEK, HEAD UP / DOWN
1063 seek_sound->set_volume(0, decibel_l, decibel_r);
1064 head_up_sound->set_volume(0, decibel_l, decibel_r);
1065 head_down_sound->set_volume(0, decibel_l, decibel_r);
1072 // ----------------------------------------------------------------------------
1074 // ----------------------------------------------------------------------------
1076 void VM::key_down(int code, bool repeat)
1078 keyboard->key_down(code);
1081 void VM::key_up(int code)
1083 keyboard->key_up(code);
1086 // ----------------------------------------------------------------------------
1088 // ----------------------------------------------------------------------------
1089 void VM::open_cart(int drv, const _TCHAR* file_path)
1093 if(iccard1 != NULL) {
1094 iccard1->open_cart(file_path);
1098 if(iccard2 != NULL) {
1099 iccard2->open_cart(file_path);
1105 void VM::close_cart(int drv)
1109 if(iccard1 != NULL) {
1110 iccard1->close_cart();
1114 if(iccard2 != NULL) {
1115 iccard2->close_cart();
1121 bool VM::is_cart_inserted(int drv)
1125 if(iccard1 != NULL) {
1126 return iccard1->is_cart_inserted();
1130 if(iccard2 != NULL) {
1131 return iccard2->is_cart_inserted();
1138 void VM::open_floppy_disk(int drv, const _TCHAR* file_path, int bank)
1141 fdc->open_disk(drv, file_path, bank);
1142 floppy->change_disk(drv);
1145 void VM::close_floppy_disk(int drv)
1147 fdc->close_disk(drv);
1148 // floppy->change_disk(drv);
1151 bool VM::is_floppy_disk_inserted(int drv)
1153 return fdc->is_disk_inserted(drv);
1156 void VM::is_floppy_disk_protected(int drv, bool value)
1158 fdc->is_disk_protected(drv, value);
1161 bool VM::is_floppy_disk_protected(int drv)
1163 return fdc->is_disk_protected(drv);
1166 bool VM::is_frame_skippable()
1168 return event->is_frame_skippable();
1171 void VM::update_config()
1173 for(DEVICE* device = first_device; device; device = device->next_device) {
1174 device->update_config();
1178 double VM::get_current_usec()
1180 if(event == NULL) return 0.0;
1181 return event->get_current_usec();
1184 uint64_t VM::get_current_clock_uint64()
1186 if(event == NULL) return (uint64_t)0;
1187 return event->get_current_clock_uint64();
1190 #define STATE_VERSION 3
1192 bool VM::process_state(FILEIO* state_fio, bool loading)
1194 if(!state_fio->StateCheckUint32(STATE_VERSION)) {
1197 for(DEVICE* device = first_device; device; device = device->next_device) {
1198 // Note: typeid(foo).name is fixed by recent ABI.Not decr. 6.
1199 // const char *name = typeid(*device).name();
1200 // But, using get_device_name() instead of typeid(foo).name() 20181008 K.O
1201 const char *name = device->get_device_name();
1202 int len = (int)strlen(name);
1203 if(!state_fio->StateCheckInt32(len)) {
1206 if(!state_fio->StateCheckBuffer(name, len, 1)) {
1209 if(!device->process_state(state_fio, loading)) {
1211 printf("Data loading Error: DEVID=%d\n", device->this_device_id);
1216 // Machine specified.
1217 state_fio->StateValue(beep_mix_ch);
1218 state_fio->StateValue(cdc_mix_ch);
1219 state_fio->StateValue(opn2_mix_ch);
1220 state_fio->StateValue(pcm_mix_ch);
1221 state_fio->StateValue(line_mix_ch);
1222 state_fio->StateValue(modem_mix_ch);
1223 state_fio->StateValue(mic_mix_ch);
1225 state_fio->StateValue(boot_seq);