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 "../scsi_host.h"
33 //#include "./towns_scsi_host.h"
34 #include "../upd71071.h"
38 #include "./dictionary.h"
40 #include "./towns_memory.h"
48 //#include "../ym2612.h"
57 #include "../debugger.h"
63 #include "./fontroms.h"
64 #include "./joystick.h"
65 #include "./joypad_2btn.h"
66 #include "./joypad_6btn.h"
67 #include "./keyboard.h"
69 #include "./msdosrom.h"
71 #include "./serialrom.h"
75 #include "./planevram.h"
77 // ----------------------------------------------------------------------------
79 // ----------------------------------------------------------------------------
82 using FMTOWNS::DICTIONARY;
83 using FMTOWNS::FLOPPY;
84 using FMTOWNS::FONT_ROMS;
85 using FMTOWNS::JOYSTICK;
86 using FMTOWNS::JOYPAD_2BTN;
87 using FMTOWNS::JOYPAD_6BTN;
89 using FMTOWNS::KEYBOARD;
91 using FMTOWNS::MSDOSROM;
93 using FMTOWNS::SERIAL_ROM;
94 using FMTOWNS::SYSROM;
96 using FMTOWNS::TOWNS_ICCARD;
98 using FMTOWNS::TOWNS_CDROM;
99 using FMTOWNS::TOWNS_CRTC;
100 using FMTOWNS::TOWNS_DMAC;
101 using FMTOWNS::TOWNS_MEMORY;
102 //using FMTOWNS::TOWNS_SCSI_HOST;
103 using FMTOWNS::TOWNS_SPRITE;
104 using FMTOWNS::TOWNS_VRAM;
105 using FMTOWNS::PLANEVRAM;
108 VM::VM(EMU_TEMPLATE* parent_emu) : VM_TEMPLATE(parent_emu)
129 //first_device = last_device = nullptr;
130 dummy = new DEVICE(this, emu); // must be 1st device
131 event = new EVENT(this, emu); // must be 2nd device
133 dummy->set_device_name(_T("1st Dummy"));
134 event->set_device_name(_T("EVENT"));
137 cpu = new I386(this, emu);
139 #if defined(HAS_I386)
140 cpu->set_device_name(_T("CPU(i386)"));
141 #elif defined(HAS_I486)
142 cpu->set_device_name(_T("CPU(i486)"));
143 #elif defined(HAS_PENTIUM)
144 cpu->set_device_name(_T("CPU(Pentium)"));
148 io = new IO(this, emu);
149 io->space = _IO_SPACE;
150 io->bus_width = _IO_BUS_WIDTH;
152 crtc = new TOWNS_CRTC(this, emu);
153 cdrom = new TOWNS_CDROM(this, emu);
155 memory = new TOWNS_MEMORY(this, emu);
156 //memory->space = _MEMORY_SPACE;
157 //memory->bank_size = _MEMORY_BANK_SIZE;
158 //memory->bus_width = _MEMORY_BUS_WIDTH;
160 vram = new TOWNS_VRAM(this, emu);
161 sprite = new TOWNS_SPRITE(this, emu);
162 sysrom = new SYSROM(this, emu);
163 msdosrom = new MSDOSROM(this, emu);
164 fontrom = new FONT_ROMS(this, emu);
165 dictionary = new DICTIONARY(this, emu);
166 #if defined(HAS_20PIX_FONTS)
167 fontrom_20pix = new FONT_ROM_20PIX(this, emu);
169 serialrom = new SERIAL_ROM(this, emu);
171 adpcm = new ADPCM(this, emu);
172 // mixer = new MIXER(this, emu); // Pseudo mixer.
174 planevram = new PLANEVRAM(this, emu);
176 adc = new AD7820KR(this, emu);
177 rf5c68 = new RF5C68(this, emu);
178 e_volumes[0] = new MB87078(this, emu);
179 e_volumes[1] = new MB87078(this, emu);
181 sio = new I8251(this, emu);
182 pit0 = new I8253(this, emu);
183 pit0->device_model = INTEL_8253;
185 pit1 = new I8253(this, emu);
186 pit1->device_model = INTEL_8253;
188 pic = new I8259(this, emu);
190 fdc = new MB8877(this, emu);
191 rtc = new MSM58321(this, emu);
192 beep = new PCM1BIT(this, emu);
193 opn2 = new YM2612(this, emu);
195 seek_sound = new NOISE(this, emu);
196 head_up_sound = new NOISE(this, emu);
197 head_down_sound = new NOISE(this, emu);
199 // scsi_host = new TOWNS_SCSI_HOST(this, emu);
200 scsi_host = new SCSI_HOST(this, emu);
202 for(int i = 0; i < 7; i++) {
203 scsi_hdd[i] = nullptr;
205 #if defined(USE_HARD_DISK)
206 for(int i = 0; i < USE_HARD_DISK; i++) {
207 scsi_hdd[i] = new SCSI_HDD(this, emu);
208 scsi_hdd[i]->set_device_name(_T("SCSI Hard Disk Drive #%d"), i + 1);
209 scsi_hdd[i]->scsi_id = i ;
210 scsi_hdd[i]->set_disk_handler(0, new HARDDISK(emu));
211 scsi_hdd[i]->set_context_interface(scsi_host);
212 my_sprintf_s(scsi_hdd[i]->vendor_id, 9, "FUJITSU");
213 my_sprintf_s(scsi_hdd[i]->product_id, 17, "SCSI-HDD");
214 scsi_host->set_context_target(scsi_hdd[i]);
217 dma = new TOWNS_DMAC(this, emu);
218 extra_dma = new TOWNS_DMAC(this, emu);
220 floppy = new FLOPPY(this, emu);
221 keyboard = new KEYBOARD(this, emu);
222 joystick = new JOYSTICK(this, emu);
223 scsi = new SCSI(this, emu);
224 timer = new TIMER(this, emu);
226 iccard1 = new TOWNS_ICCARD(this, emu);
228 iccard2 = new TOWNS_ICCARD(this, emu);
232 for(int i = 0; i < 2; i++) {
233 joypad_2btn[i] = new JOYPAD_2BTN(this, emu);
234 joypad_6btn[i] = new JOYPAD_6BTN(this, emu);
235 mouse[i] = new MOUSE(this, emu);
238 uint16_t machine_id = 0x0100; // FM-Towns1
239 uint16_t cpu_id = 0x0001; // i386DX
240 uint32_t cpu_clock = 16000 * 1000; // 16MHz
241 #if defined(_FMTOWNS1_2ND_GEN)
242 machine_id = 0x0200; // 1F/2F/1H/2H
243 #elif defined(_FMTOWNS_UX_VARIANTS)
244 machine_id = 0x0300; // UX10/20/40
245 cpu_id = 0x0003; // i386SX
246 #elif defined(_FMTOWNS1_3RD_GEN)
247 machine_id = 0x0400; // 10F/20F.40H/80H
248 #elif defined(_FMTOWNS2_CX_VARIANTS)
249 machine_id = 0x0500; // CX10/20/30/40
250 #elif defined(_FMTOWNS_UG_VARIANTS)
251 machine_id = 0x0600; // UG10/20/40/80
252 cpu_id = 0x0003; // i386SX
253 cpu_clock = 20000 * 1000; // 20MHz
254 #elif defined(_FMTOWNS_HR_VARIANTS)
256 cpu_id = 0x0002; // i486SX
257 cpu_clock = 20000 * 1000; // 20MHz
258 #elif defined(_FMTOWNS_HG_VARIANTS)
260 cpu_clock = 20000 * 1000; // 20MHz
261 #elif defined(_FMTOWNS_SG_VARIANTS)
262 machine_id = 0x0800; // OK?
263 #elif defined(_FMTOWNS_SR_VARIANTS)
264 machine_id = 0x0700; // OK?
265 cpu_id = 0x0002; // i486SX
266 cpu_clock = 20000 * 1000; // 20MHz
267 #elif defined(_FMTOWNS_UR_VARIANTS)
268 machine_id = 0x0900; // UR10/20/40/80
269 cpu_id = 0x0002; // i486DX
270 cpu_clock = 20000 * 1000; // ToDo: Correct frequency.
271 #elif defined(_FMTOWNS_MA_VARIANTS)
272 machine_id = 0x0b00; // OK?
273 cpu_id = 0x0002; // i486SX
274 cpu_clock = 33000 * 1000; // 33MHz
275 #elif defined(_FMTOWNS_ME_VARIANTS)
276 machine_id = 0x0d00; // OK?
277 cpu_id = 0x0002; // i486SX
278 cpu_clock = 25000 * 1000; // 25MHz
279 #elif defined(_FMTOWNS_MF_VARIANTS)
280 machine_id = 0x0f00; // OK?
281 cpu_id = 0x0002; // i486SX
282 cpu_clock = 33000 * 1000; // 33MHz
283 #elif defined(_FMTOWNS_MX_VARIANTS)
284 machine_id = 0x0c00; // OK?
285 cpu_id = 0x0002; // i486DX (With FPU?)
286 cpu_clock = 66000 * 1000; // 66MHz
287 #elif defined(_FMTOWNS_HC_VARIANTS)
289 // From FMTowns::MachineID() of TSUGARU,
290 // git 83d4ec2309ac9fcbb8c01f26061ff0d49c5321e4.
291 machine_id = 0x1100; // OK?
292 cpu_id = 0x0002; // Pentium (With FPU?)
293 cpu_clock = 50000 * 1000; // ToDo: Correctness frequency.
295 // ToDo: Pentium Model (After HB).
299 event->set_frames_per_sec(FRAMES_PER_SEC);
300 event->set_lines_per_frame(LINES_PER_FRAME);
302 set_machine_type(machine_id, cpu_id);
304 event->set_context_cpu(cpu, cpu_clock);
311 // Use pseudo mixer instead of event.Due to using ADC.
312 // Temporally not use mixer.
313 event->set_context_sound(beep);
314 event->set_context_sound(opn2);
315 event->set_context_sound(rf5c68);
316 event->set_context_sound(cdrom);
318 fdc->set_context_noise_seek(seek_sound);
319 fdc->set_context_noise_head_down(head_down_sound);
320 fdc->set_context_noise_head_up(head_up_sound);
321 event->set_context_sound(seek_sound);
322 event->set_context_sound(head_down_sound);
323 event->set_context_sound(head_up_sound);
326 pit0->set_context_debugger(new DEBUGGER(this, emu));
327 pit1->set_context_debugger(new DEBUGGER(this, emu));
329 pit0->set_context_ch0(timer, SIG_TIMER_CH0, 1);
330 pit0->set_context_ch1(timer, SIG_TIMER_CH1, 1);
331 pit0->set_context_ch2(beep, SIG_PCM1BIT_SIGNAL, 1);
332 pit0->set_constant_clock(0, 307200);
333 pit0->set_constant_clock(1, 307200);
334 pit0->set_constant_clock(2, 307200);
335 pit1->set_constant_clock(0, 1229900);
336 pit1->set_constant_clock(1, 1229900);
337 pit1->set_constant_clock(2, 1229900);
338 pic->set_context_cpu(cpu);
339 //pic->set_context_cpu(memory);
341 fdc->set_context_irq(floppy, SIG_FLOPPY_IRQ, 1);
342 //fdc->set_context_irq(dma, SIG_TOWNS_DMAC_EOT_CH0, 1);
344 rtc->set_context_data(timer, SIG_TIMER_RTC, 0x0f, 0);
345 rtc->set_context_busy(timer, SIG_TIMER_RTC_BUSY, 0x80);
346 scsi_host->set_context_irq(scsi, SIG_SCSI_IRQ, 1);
347 scsi_host->set_context_drq(scsi, SIG_SCSI_DRQ, 1);
348 scsi_host->set_context_drq(keyboard, SIG_KEYBOARD_BOOTSEQ_END, 1);
351 dma->set_context_debugger(new DEBUGGER(this, emu));
352 extra_dma->set_context_debugger(new DEBUGGER(this, emu));
354 // Note: DMAC may set wait value to CPU. 20230409 K.O
355 dma->set_context_cpu(NULL);
356 //dma->set_context_cpu(cpu);
357 dma->set_context_memory(memory);
358 dma->set_context_ch0(fdc);
359 // This is workaround for FM-Towns's SCSI.
360 dma->set_force_16bit_transfer(1, false);
361 dma->set_context_ch1(scsi_host);
362 //dma->set_context_ch2(printer);
363 dma->set_context_ch3(cdrom);
365 //extra_dma->set_context_cpu(cpu);
366 extra_dma->set_context_cpu(NULL);
367 extra_dma->set_context_memory(memory);
369 //dma->set_context_tc1(scsi, SIG_SCSI_EOT, 0xffffffff);
370 dma->set_context_tc3(cdrom, SIG_TOWNS_CDROM_DMAINT, 0xffffffff);
372 dma->set_context_ube(1, scsi_host, SIG_SCSI_16BIT_BUS, 0x02);
373 //dma->set_context_ack(1, scsi_host, SIG_SCSI_ACK, 0xffffffff);
374 //dma->set_context_ack(3, cdrom, SIG_TOWNS_CDROM_DMAACK, 0xffffffff);
375 dma->set_context_child_dma(extra_dma);
377 floppy->set_context_fdc(fdc);
379 sprite->set_context_vram(vram);
380 // sprite->set_context_font(fontrom);
381 // sprite->set_context_crtc(crtc);
383 sprite->set_context_debugger(new DEBUGGER(this, emu));
386 planevram->set_context_vram(vram);
387 planevram->set_context_sprite(sprite);
388 planevram->set_context_crtc(crtc);
390 crtc->set_context_sprite(sprite);
391 crtc->set_context_vram(vram);
392 crtc->set_context_font(fontrom);
394 // e_volumes[0]->set_context_device(0, line_in, 0,
395 // MB87078_TYPE_MASK_LEFT);
396 // e_volumes[0]->set_context_device(1, line_in, 0,
397 // MB87078_TYPE_MASK_RIGHT);
398 e_volumes[1]->set_context_device(0, cdrom, 0,
399 MB87078_TYPE_SET_LEFT, SIG_TOWNS_CDROM_MUTE_L,
404 e_volumes[1]->set_context_device(1, cdrom, 0,
405 MB87078_TYPE_SET_RIGHT, SIG_TOWNS_CDROM_MUTE_R,
411 memory->set_context_cpu(cpu);
412 memory->set_context_dmac(dma);
413 memory->set_context_vram(vram);
414 memory->set_context_planevram(planevram);
415 memory->set_context_crtc(crtc);
416 memory->set_context_system_rom(sysrom);
417 memory->set_context_msdos(msdosrom);
418 memory->set_context_dictionary(dictionary);
419 memory->set_context_font_rom(fontrom);
420 memory->set_context_timer(timer);
421 memory->set_context_serial_rom(serialrom);
422 memory->set_context_sprite(sprite);
423 memory->set_context_pcm(rf5c68);
424 memory->set_context_iccard(iccard1, 0);
425 memory->set_context_iccard(iccard2, 1);
427 adpcm->set_context_opn2(opn2);
428 adpcm->set_context_rf5c68(rf5c68);
429 adpcm->set_context_adc(adc);
431 rf5c68->set_context_interrupt_boundary(adpcm, SIG_ADPCM_WRITE_INTERRUPT, 0xffffffff);
433 rf5c68->set_context_debugger(new DEBUGGER(this, emu));
435 opn2->set_context_irq(adpcm, SIG_ADPCM_OPX_INTR, 0xffffffff);
437 adc->set_sample_rate(19200);
438 adc->set_sound_bank(-1);
439 adc->set_context_interrupt(adpcm, SIG_ADPCM_ADC_INTR, 0xffffffff);
441 scsi->set_context_dma(dma);
442 scsi->set_context_host(scsi_host);
443 scsi->set_context_pic(pic);
444 timer->set_context_pcm(beep);
445 timer->set_context_rtc(rtc);
446 //timer->set_context_halt_line(cpu, SIG_CPU_HALTREQ, 0xffffffff);
447 timer->set_context_halt_line(cpu, SIG_CPU_BUSREQ, 0xffffffff);
449 for(int i = 0; i < 2; i++) {
451 // 0: Towns PAD 2buttons
452 // 1: Towns PAD 6buttons
454 // 3: Analog Pad (reserved)
455 // 4: Libble Rabble stick (reserved)
456 joystick->set_context_joystick(i, joypad_2btn[i]);
457 joystick->set_context_joystick(i, joypad_6btn[i]);
458 joystick->set_context_joystick(i, mouse[i]);
461 joystick->set_context_debugger(new DEBUGGER(this, emu));
463 // ToDo: Selective by config.
464 for(int i = 0; i < 2; i++) {
465 joypad_2btn[i]->set_context_pad_num(i);
466 joypad_2btn[i]->set_context_parent_port(i, joystick, 0, 0xff);
467 joypad_2btn[i]->set_negative_logic(true);
468 joypad_2btn[i]->set_enable(true);
470 for(int i = 0; i < 2; i++) {
471 joypad_6btn[i]->set_context_pad_num(i);
472 joypad_6btn[i]->set_context_parent_port(i, joystick, 0, 0xff);
473 joypad_6btn[i]->set_negative_logic(true);
474 joypad_6btn[i]->set_enable(false);
476 for(int i = 0; i < 2; i++) {
477 mouse[i]->set_context_pad_num(i);
478 mouse[i]->set_context_parent_port(i, joystick, 0, 0xff);
479 mouse[i]->set_negative_logic(true);
480 mouse[i]->set_enable(false);
482 joystick->set_using_pad(0, -1);
483 joystick->set_using_pad(1, -1);
486 cpu->set_context_mem(memory);
487 cpu->set_context_io(io);
488 cpu->set_context_intr(pic);
489 #ifdef SINGLE_MODE_DMA
490 cpu->set_context_dma(dma);
492 cpu->set_context_bios(nullptr);
493 cpu->set_context_extreset(memory, SIG_FMTOWNS_NOTIFY_RESET, 0xffffffff);
495 cpu->set_context_debugger(new DEBUGGER(this, emu));
500 // IRQ2 : USART (ToDo)
501 // IRQ3 : EXTRA USART (ToDo)
502 // IRQ4 : EXTRA I/O (Maybe not implement)
503 // IRQ5 : EXTRA I/O (Maybe not implement)
505 // IRQ7 : Deisy chain (to IRQ8 - 15)
506 timer->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR0, 0xffffffff);
507 keyboard->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR1, 0xffffffff);
508 floppy->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR6, 0xffffffff);
510 // IRQ8 : SCSI (-> scsi.cpp)
512 // IRQ10 : EXTRA I/O (Maybe not implement)
514 // IRQ12 : PRINTER (ToDo)
515 // IRQ13 : ADPCM AND OPN2 (Route to adpcm.cpp)
516 // IRQ14 : EXTRA I/O (Maybe not implement)
518 cdrom->set_context_mpuint_line(pic, SIG_I8259_CHIP1 | SIG_I8259_IR1, 0xffffffff);
519 crtc->set_context_vsync(pic, SIG_I8259_CHIP1 | SIG_I8259_IR3, 0xffffffff);
520 adpcm->set_context_intr_line(pic, SIG_I8259_CHIP1 | SIG_I8259_IR5, 0xffffffff);
523 // DMA1 : SCSI (-> scsi.cpp)
524 // DMA2 : PRINTER (ToDo)
526 // EXTRA DMA0 : EXTRA SLOT (Maybe not implement)
527 // EXTRA DMA1 : Reserved
528 // EXTRA DMA2 : Reserved
529 // EXTRA DMA3 : Reserved
530 fdc->set_context_drq(dma, SIG_UPD71071_CH0, 1);
531 fdc->set_context_drq(keyboard, SIG_KEYBOARD_BOOTSEQ_END, 1);
532 cdrom->set_context_drq_line(dma, SIG_UPD71071_CH3, 0xff);
533 cdrom->set_context_drq_line(keyboard, SIG_KEYBOARD_BOOTSEQ_END, 1);
535 // NMI0 : KEYBOARD (RAS)
536 // NMI1 : Extra SLOT (Maybe not implement)
537 keyboard->set_context_nmi_line(memory, SIG_CPU_NMI, 0xffffffff);
539 cdrom->set_context_dmac(dma);
540 // For Debugging, will remove 20200822 K.O
541 cdrom->set_context_cpu(cpu);
542 cdrom->set_context_eot_line(dma, SIG_TOWNS_DMAC_EOT_CH3, 0xffffffff);
545 io->set_iowait_range_rw(0x0000, 0xffff, 6); // ToDo: May variable wait.
547 io->set_iomap_alias_rw (0x0000, pic, I8259_ADDR_CHIP0 | 0);
548 io->set_iomap_alias_rw (0x0002, pic, I8259_ADDR_CHIP0 | 1);
549 io->set_iomap_alias_rw (0x0010, pic, I8259_ADDR_CHIP1 | 0);
550 io->set_iomap_alias_rw (0x0012, pic, I8259_ADDR_CHIP1 | 1);
552 io->set_iomap_single_rw(0x0020, memory); // RESET REASON / POWER CONTROL (by software)
553 io->set_iomap_single_rw(0x0022, memory); // POWER CONTROL
554 io->set_iomap_single_rw(0x0024, memory); // MISC3 / DMA WRAP (AFTER MA/MX/ME)
555 io->set_iomap_single_r (0x0025, memory); // MISC4
556 io->set_iomap_range_r (0x0026, 0x0027, timer); // Freerun counter
557 io->set_iomap_single_rw(0x0028, memory); // NMI MASK
559 io->set_iomap_range_r (0x0030, 0x0031, memory); // cpu id / machine id
560 io->set_iomap_single_rw(0x0032, memory); // serial rom (routed from memory)
561 io->set_iomap_single_r (0x0034, scsi); // ENABLE/ UNABLE to WORD DMA for SCSI
563 io->set_iomap_alias_rw(0x0040, pit0, 0);
564 io->set_iomap_alias_rw(0x0042, pit0, 1);
565 io->set_iomap_alias_rw(0x0044, pit0, 2);
566 io->set_iomap_alias_rw(0x0046, pit0, 3);
567 io->set_iomap_alias_rw(0x0050, pit1, 0);
568 io->set_iomap_alias_rw(0x0052, pit1, 1);
569 io->set_iomap_alias_rw(0x0054, pit1, 2);
570 io->set_iomap_alias_rw(0x0056, pit1, 3);
572 io->set_iomap_single_rw(0x0060, timer); // Beep and interrupts register
573 io->set_iomap_single_rw(0x0068, timer); // Interval timer register2 : CONTROL (after Towns 10F).
574 io->set_iomap_range_rw (0x006a, 0x006b, timer); // Interval timer register2 : DATA (after Towns 10F).
575 io->set_iomap_single_rw(0x006c, timer); // 1uS wait register (after Towns 10F).
577 io->set_iomap_single_rw(0x0070, timer); // RTC DATA
578 io->set_iomap_single_w (0x0080, timer); // RTC COMMAND
580 io->set_iomap_range_rw (0x00a0, 0x00af, dma);
581 io->set_iomap_range_rw (0x00b0, 0x00bf, extra_dma);
583 io->set_iomap_single_rw(0x00c0, memory); // CACHE CONTROLLER (after HR, i486)
584 io->set_iomap_single_rw(0x00c2, memory); // CACHE DIAGNNOSTICS (after HR, i486)
586 io->set_iomap_alias_rw (0x0200, fdc, 0); // STATUS/COMMAND
587 io->set_iomap_alias_rw (0x0202, fdc, 1); // TRACK
588 io->set_iomap_alias_rw (0x0204, fdc, 2); // SECTOR
589 io->set_iomap_alias_rw (0x0206, fdc, 3); // DATA
590 io->set_iomap_single_rw(0x0208, floppy); // DRIVE STATUS / DRIVE CONTROL
591 io->set_iomap_single_rw(0x020c, floppy); // DRIVE SELECT
592 io->set_iomap_single_r (0x020d, floppy); // FDDVEXT (after HG/HR).
593 io->set_iomap_single_rw(0x020e, floppy); // Towns drive SW
595 io->set_iomap_single_r (0x0400, memory); // RESOLUTION
596 io->set_iomap_single_rw(0x0402, memory); // RESERVED (??)
597 io->set_iomap_single_rw(0x0404, memory); // SYSTEM STATUS
598 // io->set_iomap_range_rw (0x0406, 0x043f, memory); // Reserved
600 io->set_iomap_single_rw(0x0440, crtc); // CRTC ADDRESS INDEX
601 io->set_iomap_range_rw (0x0442, 0x0443, crtc); // CRTC DATA
602 io->set_iomap_single_rw(0x0448, crtc); // VIDEO OUT ADDRESS INDEX
603 io->set_iomap_single_rw(0x044a, crtc); // VIDEO OUT DATA
604 io->set_iomap_single_r (0x044c, crtc); // DIGITAL PALLETTE STATUS, SPRITE STATUS
607 io->set_iomap_single_rw(0x0450, sprite); // SPRITE
608 io->set_iomap_single_rw(0x0452, sprite); // SPRITE
610 io->set_iomap_single_rw(0x0458, vram); // VRAM ACCESS CONTROLLER (ADDRESS)
611 io->set_iomap_range_rw (0x045a, 0x045b, vram); // VRAM ACCESS CONTROLLER (DATA)
613 //io->set_iomap_single_r (0x0470, crtc); // HIGH RESOLUTION (after MX)
614 //io->set_iomap_single_r (0x0471, vram); // VRAM CAPACITY (after MX, by MBytes)
615 //io->set_iomap_range_rw (0x0472, 0x0473, crtc); // VIDEO OUT REGS ADDRESS (after MX, by MBytes)
616 //io->set_iomap_range_rw (0x0474, 0x0477, crtc); // VIDEO OUT REGS DATA (after MX, by MBytes)
618 io->set_iomap_single_rw(0x0480, memory); // MEMORY REGISTER
619 io->set_iomap_single_rw(0x0484, dictionary); // Dictionary BANK
621 io->set_iomap_alias_r(0x48a, iccard1, 0); //
622 //io->set_iomap_alias_rw(0x490, memory_card); // After Towns2
623 //io->set_iomap_alias_rw(0x491, memory_card); // After Towns2
625 //io->set_iomap_single_r(0x4b0, cdrom); // CDROM functions (after MX)
626 io->set_iomap_single_rw(0x04c0, cdrom); // CDROM: MASTER STATUS/MASTER CONTROL
627 io->set_iomap_single_rw(0x04c2, cdrom); // CDROM: STATUS(FIFO) / COMMAND
628 io->set_iomap_single_rw(0x04c4, cdrom); // CDROM: DATA (BUFFER~ / PARAMETER (FIFO)
629 io->set_iomap_single_w (0x04c6, cdrom); // CDROM TRANSFER CONTROL
630 io->set_iomap_single_rw(0x04c8, cdrom); // CDROM CACHE CONTROL (after HR)
631 io->set_iomap_range_r (0x04cc, 0x04cd, cdrom); // CDROM SUBCODE DATA
634 io->set_iomap_single_r (0x04d0, joystick); // Pad1
635 io->set_iomap_single_r (0x04d2, joystick); // Pad 2
636 io->set_iomap_single_rw(0x04d5, adpcm); // mute
637 io->set_iomap_single_w (0x04d6, joystick); // Pad out
640 io->set_iomap_alias_rw(0x04d8, opn2, 0); // STATUS(R)/Addrreg 0(W)
641 io->set_iomap_alias_w (0x04da, opn2, 1); // Datareg 0(W)
642 io->set_iomap_alias_w (0x04dc, opn2, 2); // Addrreg 1(W)
643 io->set_iomap_alias_w (0x04de, opn2, 3); // Datareg 1(W)
645 io->set_iomap_alias_rw(0x04e0, e_volumes[0], 0);
646 io->set_iomap_alias_rw(0x04e1, e_volumes[0], 1);
647 io->set_iomap_alias_rw(0x04e2, e_volumes[1], 0);
648 io->set_iomap_alias_rw(0x04e3, e_volumes[1], 1);
651 io->set_iomap_single_r (0x04e7, adpcm); // A/D SAMPLING DATA REG
652 io->set_iomap_single_rw(0x04e8, adpcm); // A/D SAMPLING FLAG
653 io->set_iomap_single_rw(0x04e9, adpcm); // OPN2/PCM INTERRUPT (INT13) REGISTER
654 io->set_iomap_single_rw(0x04ea, adpcm); // PCM INTERRUPT MASK
655 io->set_iomap_single_r (0x04eb, adpcm); // PCM INTERRUPT STATUS
656 io->set_iomap_single_w (0x04ec, adpcm); // PCM LED/MUTE
658 io->set_iomap_range_w (0x04f0, 0x04f8, rf5c68); // PCM CONTROL REGS (WO?)
660 //io->set_iomap_single_rw(0x510, newpcm); // PCM BANK (after MX)
661 //io->set_iomap_single_rw(0x511, newpcm); // PCM DMA STATUS(after MX)
662 //io->set_iomap_range_rw (0x512, 0x0513, newpcm); // PCM DMA COUNTER(after MX)
663 //io->set_iomap_range_rw (0x514, 0x0517, newpcm); // PCM DMA ADDRESS(after MX)
664 //io->set_iomap_single_rw(0x518, newpcm); // PCM CLOCK (after MX)
665 //io->set_iomap_single_rw(0x519, newpcm); // PCM MODE (after MX)
666 //io->set_iomap_single_rw(0x51a, newpcm); // PCM SYSTEM CONTROL (after MX)
667 //io->set_iomap_single_rw(0x51b, newpcm); // PCM BUFFER STATUS/CONTROL (after MX)
668 //io->set_iomap_single_rw(0x51c, newpcm); // PCM REC/PLAY (after MX)
669 //io->set_iomap_single_rw(0x51d, newpcm); // PCM PEAK MONITOR / TRIGGER (after MX)
670 //io->set_iomap_range_rw (0x51e, 0x051f, newpcm); // PCM LEVEL / SOFTWARE DATA (after MX)
672 io->set_iomap_single_rw(0x05c0, memory); // NMI MASK
673 io->set_iomap_single_r (0x05c2, memory); // NMI STATUS
674 io->set_iomap_single_r (0x05c8, sprite); // TVRAM EMULATION
675 io->set_iomap_single_w (0x05ca, crtc); // VSYNC INTERRUPT
677 io->set_iomap_single_rw(0x05e0, memory); // Hidden MEMORY WAIT REGISTER from AB.COM (Towns 1/2)
678 io->set_iomap_single_rw(0x05e2, memory); // Hidden MEMORY WAIT REGISTER from AB.COM (After Towns 1H/1F/2H/2F )
679 io->set_iomap_single_rw(0x05e6, memory); // Hidden VRAM WAIT REGISTER from TSUGARU (Maybe after Towns 10H/10F/20H/20F )
680 io->set_iomap_single_r (0x05e8, memory); // RAM capacity register.(later Towns1H/2H/1F/2F).
681 io->set_iomap_single_rw(0x05ec, memory); // RAM Wait register , ofcially after Towns2, but exists after Towns1H.
682 io->set_iomap_single_r (0x05ed, memory); // Maximum clock register (After HR/HG).
683 io->set_iomap_single_rw(0x05ee, vram); // VRAM CACHE CONTROLLER (After HR, i486)
685 io->set_iomap_single_rw(0x0600, keyboard);
686 io->set_iomap_single_rw(0x0602, keyboard);
687 io->set_iomap_single_rw(0x0604, keyboard);
688 //io->set_iomap_single_r (0x606, keyboard); // BUFFER FULL (for TUNER)
690 //io->set_iomap_single_rw(0x0800, printer);
691 //io->set_iomap_single_rw(0x0802, printer);
692 //io->set_iomap_single_rw(0x0804, printer);
694 io->set_iomap_alias_rw (0x0a00, sio, 0);
695 io->set_iomap_alias_rw (0x0a02, sio, 1);
696 // io->set_iomap_single_r (0x0a04, serial);
697 // io->set_iomap_single_r (0x0a06, serial);
698 // io->set_iomap_single_w (0x0a08, serial);
699 // io->set_iomap_single_rw(0x0a0a, modem);
700 // io->set_iomap_single_rw(0x0a0c, uart_fifo); // USART HAVE FIFO (after MA)
701 // io->set_iomap_single_rw(0x0a0d, uart_fifo); // USART FIFO STATUS (after MA)
702 // io->set_iomap_single_rw(0x0a0e, uart_fifo); // USART FIFO STATUS (after MA)
704 io->set_iomap_single_rw(0x0c30, scsi);
705 io->set_iomap_single_rw(0x0c32, scsi);
706 io->set_iomap_single_r (0x0c34, scsi);
708 // ToDo: Implement debugging I/Os to 2000h - 2FFFh.
710 for(uint32_t addr = 0x3000; addr < 0x4000; addr += 2) {
711 io->set_iomap_single_rw (addr, dictionary); // CMOS
713 io->set_iomap_range_rw (0xfd90, 0xfd91, crtc); // PALETTE INDEX
714 io->set_iomap_range_rw (0xfd92, 0xfd93, crtc); // PALETTE DATA BLUE
715 io->set_iomap_range_rw (0xfd94, 0xfd95, crtc); // PALETTE DATA RED
716 io->set_iomap_single_rw(0xfd96, crtc); // PALETTE DATA GREEN
717 io->set_iomap_range_rw (0xfd98, 0xfd9f, crtc); // DIGITAL PALETTE REGISTERS(EMULATED)
719 io->set_iomap_single_rw(0xfda0, crtc); // CRTC: VSYNC, HSYNC / OUTPUT CONTROL
720 io->set_iomap_single_r (0xfda2, crtc); // CRTC OUT (after UG)
721 io->set_iomap_single_rw(0xfda4, memory); // CRTC: READ COMPATIBLE (after UG)
723 io->set_iomap_range_rw (0xff80, 0xff83, planevram); // MMIO
724 io->set_iomap_single_r (0xff84, planevram); // MMIO
725 io->set_iomap_single_r (0xff86, planevram); // MMIO
726 io->set_iomap_single_rw(0xff88, memory); // MMIO
728 io->set_iomap_range_rw (0xff94, 0xff99, memory); // MMIO
729 io->set_iomap_range_r (0xff9c, 0xff9d, memory); // MMIO
730 io->set_iomap_single_rw(0xff9e, memory); // MMIO
731 io->set_iomap_single_rw(0xffa0, planevram); // MMIO
733 // Vram allocation may be before initialize().
734 // initialize all devices
735 #if defined(__GIT_REPO_VERSION)
736 set_git_repo_version(__GIT_REPO_VERSION);
738 // ToDo : Use config framework
739 int exram_size = config.current_ram_size;
741 switch((machine_id & 0xff00) >> 8) {
743 case 0x01: // Towns 1/2 : Not Supported.
744 exram_size = 2; // UP TO 5MB.
746 case 0x03: // TOWNS2 UX
747 case 0x06: // TOWNS2 UG
748 exram_size = 4; // UP TO 9MB.
750 case 0x02: // TOWNS 2F/2H
751 case 0x04: // TOWNS 10F/10H/20F/20H
752 exram_size = 6; // MAYBE UP TO 7MB.
754 case 0x05: // TOWNS II CX
755 case 0x08: // TOWNS II HG
756 exram_size = 8; // UP TO 15MB.
758 case 0x07: // Towns II HR
759 case 0x09: // Towns II UR
760 exram_size = 24; // UP TO 31MB.
763 exram_size = 31; // UP TO 127MB.
767 if(exram_size < MIN_RAM_SIZE) {
768 exram_size = MIN_RAM_SIZE;
771 memory->set_extra_ram_size(exram_size);
773 #if defined(WITH_I386SX)
774 cpu->device_model = INTEL_80386;
775 #elif defined(WITH_I486SX)
776 cpu->device_model = INTEL_I486SX;
777 #elif defined(WITH_I486DX)
778 cpu->device_model = INTEL_I486DX;
779 #elif defined(WITH_PENTIUM)
780 cpu->device_model = INTEL_PENTIUM;
783 cpu->device_model = INTEL_80386;
786 initialize_devices();
787 // cpu->set_address_mask(0xffffffff);
792 // delete all devices
796 void VM::set_machine_type(uint16_t machine_id, uint16_t cpu_id)
798 if(memory != nullptr) {
799 memory->set_cpu_id(cpu_id);
800 memory->set_machine_id(machine_id);
802 if(crtc != nullptr) {
803 crtc->set_cpu_id(cpu_id);
804 crtc->set_machine_id(machine_id);
806 if(timer != nullptr) {
807 timer->set_cpu_id(cpu_id);
808 timer->set_machine_id(machine_id);
810 if(cdrom != nullptr) {
811 cdrom->set_cpu_id(cpu_id);
812 cdrom->set_machine_id(machine_id);
814 if(scsi != nullptr) {
815 scsi->set_cpu_id(cpu_id);
816 scsi->set_machine_id(machine_id);
818 if(serialrom != nullptr) {
819 serialrom->set_cpu_id(cpu_id);
820 serialrom->set_machine_id(machine_id);
822 if(floppy != nullptr) {
823 floppy->set_cpu_id(cpu_id);
824 floppy->set_machine_id(machine_id);
826 #if defined(HAS_20PIX_FONTS)
827 if(fontrom_20pix != nullptr) {
828 fontrom_20pix->set_cpu_id(cpu_id);
829 fontrom_20pix->set_machine_id(machine_id);
832 if(vram != nullptr) {
833 vram->set_cpu_id(cpu_id);
834 vram->set_machine_id(machine_id);
840 // ----------------------------------------------------------------------------
841 // drive virtual machine
842 // ----------------------------------------------------------------------------
848 VM_TEMPLATE::reset();
849 // cpu->set_address_mask(0xffffffff);
852 void VM::special_reset(int num)
857 VM_TEMPLATE::reset();
858 // cpu->set_address_mask(0xffffffff);
859 __LIKELY_IF(keyboard != nullptr) {
860 keyboard->special_reset(num);
866 __LIKELY_IF(event != nullptr) {
870 void VM::process_boot_sequence(uint32_t val)
874 if(keyboard != nullptr) {
875 keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
882 // ----------------------------------------------------------------------------
884 // ----------------------------------------------------------------------------
887 DEVICE *VM::get_cpu(int index)
896 // ----------------------------------------------------------------------------
898 // ----------------------------------------------------------------------------
900 void VM::draw_screen()
902 __LIKELY_IF(crtc != nullptr) {
908 // ----------------------------------------------------------------------------
910 // ----------------------------------------------------------------------------
912 void VM::initialize_sound(int rate, int samples)
915 // init sound manager
916 event->initialize_sound(rate, samples);
919 beep->initialize_sound(rate, 8000);
923 // MASTER CLOCK MAYBE 600KHz * 12 = 7200KHz .
924 // From FM-Towns Technical Databook (Rev.2), Page 201
925 // opn2->initialize_sound(rate, (int)(600.0e3 * 12.0) , samples, 0.0, 0.0);
927 // But...20230330 K.O:
928 // From src/ym2612/ym2612.h of Tsugaru below quote.
929 // Thanks to YS-11 San.
931 // What exactly is the master clock given to YM2612 of TOWNS?
932 // FM Towns Technical Databook tells internal clock is 600KHz on page 201.
933 // However, after calibrating the tone to match real Towns, it must be 690KHz. Is it correct?
934 // Master clock must be the internal clock times pre scaler. But, we don't know the default pre scalar value.
935 // Let's say it is 3. Then, 690KHz times 3=2070KHz. Sounds reasonable.
936 // But, now FM77AV's YM2203C uses master clock frequency of 1228.8KHz.
937 // And after calibration we know the ratio between the two.
938 // From there, 1228.8*1698/1038=1999.46. 2MHz. Makes more sense.
939 opn2->initialize_sound(rate, (int)(((1228.8e3 * 1698.0) / 1038.0) * 4.0) , samples, 0.0, 0.0);
942 rf5c68->initialize_sound(rate, samples);
944 // add_sound_in_source() must add after per initialize_sound().
945 adc_in_ch = event->add_sound_in_source(rate, samples, 2);
946 adc->set_sample_rate(19200);
947 adc->set_sound_bank(adc_in_ch);
949 mixer->set_context_out_line(adc_in_ch);
950 mixer->set_context_sample_out(adc_in_ch, rate, samples); // Must be 2ch.
951 // ToDo: Check recording sample rate & channels.
952 mic_in_ch = event->add_sound_in_source(rate, samples, 2);
953 mixer->set_context_mic_in(mic_in_ch, rate, samples);
955 line_in_ch = event->add_sound_in_source(rate, samples, 2);
956 mixer->set_context_line_in(line_in_ch, rate, samples);
961 uint16_t* VM::create_sound(int* extra_frames)
963 __LIKELY_IF(event != nullptr) {
964 return event->create_sound(extra_frames);
966 return VM_TEMPLATE::create_sound(extra_frames);
969 int VM::get_sound_buffer_ptr()
971 __LIKELY_IF(event != nullptr) {
972 return event->get_sound_buffer_ptr();
974 return VM_TEMPLATE::get_sound_buffer_ptr();
977 void VM::clear_sound_in()
979 __UNLIKELY_IF(event == nullptr) return;
981 event->clear_sound_in_source(adc_in_ch);
982 event->clear_sound_in_source(mic_in_ch);
983 event->clear_sound_in_source(line_in_ch);
987 int VM::get_sound_in_data(int ch, int32_t* dst, int expect_samples, int expect_rate, int expect_channels)
989 if(dst == nullptr) return 0;
990 if(expect_samples <= 0) return 0;
1003 if(n_ch < 0) return 0;
1005 if(event != nullptr) {
1006 samples = event->get_sound_in_data(n_ch, dst, expect_samples, expect_rate, expect_channels);
1011 // Write to event's buffer
1012 int VM::sound_in(int ch, int32_t* src, int samples)
1014 if(ch < 0) return 0;
1015 if(ch >= 2) return 0;
1018 case 0x100: // ADC in from MIXER, not connected.
1027 if(n_ch < 0) return 0;
1030 if(event != nullptr) {
1032 ss = event->write_sound_in_buffer(n_ch, src, samples);
1038 #if defined(USE_HARD_DISK)
1039 void VM::open_hard_disk(int drv, const _TCHAR* file_path)
1041 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
1042 if(scsi_hdd[drv] != nullptr) {
1043 scsi_hdd[drv]->open(0, file_path, 512);
1048 void VM::close_hard_disk(int drv)
1050 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
1051 if(scsi_hdd[drv] != nullptr) {
1052 scsi_hdd[drv]->close(0);
1057 bool VM::is_hard_disk_inserted(int drv)
1059 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
1060 if(scsi_hdd[drv] != nullptr) {
1061 return scsi_hdd[drv]->mounted(0);
1067 uint32_t VM::is_hard_disk_accessed()
1069 uint32_t status = 0;
1071 for(int drv = 0; drv < USE_HARD_DISK; drv++) {
1072 if(scsi_hdd[drv] != nullptr) {
1073 if(scsi_hdd[drv]->accessed(0)) {
1078 process_boot_sequence(status);
1081 #endif // USE_HARD_DISK
1083 void VM::open_compact_disc(int drv, const _TCHAR* file_path)
1085 if(cdrom != nullptr) {
1086 cdrom->open(file_path);
1090 void VM::close_compact_disc(int drv)
1092 if(cdrom != nullptr) {
1097 bool VM::is_compact_disc_inserted(int drv)
1099 if(cdrom == nullptr) return false;
1100 return cdrom->mounted();
1103 uint32_t VM::is_compact_disc_accessed()
1105 uint32_t status = cdrom->accessed();
1106 process_boot_sequence(status);
1110 #ifdef USE_SOUND_VOLUME
1111 void VM::set_sound_device_volume(int ch, int decibel_l, int decibel_r)
1113 #ifndef HAS_LINEIN_SOUND
1114 // if(ch >= 7) ch++;
1116 #ifndef HAS_MIC_SOUND
1117 // if(ch >= 8) ch++;
1119 #ifndef HAS_MODEM_SOUND
1120 // if(ch >= 9) ch++;
1122 #ifndef HAS_2ND_ADPCM
1123 // if(ch >= 10) ch++;
1125 if(ch == 0) { // BEEP
1126 if(beep != nullptr) {
1127 beep->set_volume(0, decibel_l, decibel_r);
1130 else if(ch == 1) { // CD-ROM
1131 if(e_volumes[1] != nullptr) {
1132 e_volumes[1]->set_volumes(0, decibel_l, 1, decibel_r);
1133 } else if(cdrom != nullptr) {
1134 cdrom->set_volume(0, decibel_l, decibel_r);
1137 else if(ch == 2) { // OPN2
1138 if(opn2 != nullptr) {
1139 opn2->set_volume(0, decibel_l, decibel_r);
1142 else if(ch == 3) { // ADPCM
1143 if(rf5c68 != nullptr) {
1144 rf5c68->set_volume(0, decibel_l, decibel_r);
1147 else if(ch == 4) { // SEEK, HEAD UP / DOWN
1148 if(seek_sound != nullptr) {
1149 seek_sound->set_volume(0, decibel_l, decibel_r);
1151 if(head_up_sound != nullptr) {
1152 head_up_sound->set_volume(0, decibel_l, decibel_r);
1154 if(head_down_sound != nullptr) {
1155 head_down_sound->set_volume(0, decibel_l, decibel_r);
1161 // ----------------------------------------------------------------------------
1163 // ----------------------------------------------------------------------------
1165 void VM::key_down(int code, bool repeat)
1167 __LIKELY_IF(keyboard != nullptr) {
1168 keyboard->key_down(code);
1172 void VM::key_up(int code)
1174 __LIKELY_IF(keyboard != nullptr) {
1175 keyboard->key_up(code);
1179 // ----------------------------------------------------------------------------
1181 // ----------------------------------------------------------------------------
1182 void VM::open_cart(int drv, const _TCHAR* file_path)
1186 if(iccard1 != nullptr) {
1187 iccard1->open_cart(file_path);
1191 if(iccard2 != nullptr) {
1192 iccard2->open_cart(file_path);
1198 void VM::close_cart(int drv)
1202 if(iccard1 != nullptr) {
1203 iccard1->close_cart();
1207 if(iccard2 != nullptr) {
1208 iccard2->close_cart();
1214 bool VM::is_cart_inserted(int drv)
1218 if(iccard1 != nullptr) {
1219 return iccard1->is_cart_inserted();
1223 if(iccard2 != nullptr) {
1224 return iccard2->is_cart_inserted();
1231 void VM::open_floppy_disk(int drv, const _TCHAR* file_path, int bank)
1234 if(fdc != nullptr) {
1235 fdc->open_disk(drv, file_path, bank);
1236 floppy->change_disk(drv);
1240 void VM::close_floppy_disk(int drv)
1242 if(fdc != nullptr) {
1243 fdc->close_disk(drv);
1244 // floppy->change_disk(drv);
1247 uint32_t VM::is_floppy_disk_accessed()
1249 uint32_t val = fdc->read_signal(0);
1250 process_boot_sequence(val);
1254 bool VM::is_floppy_disk_inserted(int drv)
1256 if(fdc != nullptr) {
1257 return fdc->is_disk_inserted(drv);
1259 return VM_TEMPLATE::is_floppy_disk_inserted(drv);
1262 void VM::is_floppy_disk_protected(int drv, bool value)
1264 __LIKELY_IF(fdc != nullptr) {
1265 fdc->is_disk_protected(drv, value);
1269 bool VM::is_floppy_disk_protected(int drv)
1271 __LIKELY_IF(fdc != nullptr) {
1272 return fdc->is_disk_protected(drv);
1274 return VM_TEMPLATE::is_floppy_disk_protected(drv);
1277 bool VM::is_frame_skippable()
1279 __LIKELY_IF(event != nullptr) {
1280 return event->is_frame_skippable();
1282 return VM_TEMPLATE::is_frame_skippable();
1286 double VM::get_current_usec()
1288 __LIKELY_IF(event != nullptr) {
1289 return event->get_current_usec();
1291 return VM_TEMPLATE::get_current_usec();
1294 uint64_t VM::get_current_clock_uint64()
1296 __LIKELY_IF(event != nullptr) {
1297 return event->get_current_clock_uint64();
1299 return VM_TEMPLATE::get_current_clock_uint64();
1302 #define STATE_VERSION 4
1304 bool VM::process_state(FILEIO* state_fio, bool loading)
1306 if(!(VM_TEMPLATE::process_state_core(state_fio, loading, STATE_VERSION))) {
1310 // Machine specified.
1311 state_fio->StateValue(boot_seq);