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"
25 #include "../mb8877.h"
26 #include "../msm58321.h"
28 #include "../pcm1bit.h"
29 #include "../harddisk.h"
30 #include "../scsi_hdd.h"
31 #include "./towns_scsi_host.h"
32 #include "../upd71071.h"
34 #include "towns_cdrom.h"
35 #include "towns_crtc.h"
36 #include "towns_dictionary.h"
37 #include "towns_dmac.h"
38 #include "towns_memory.h"
39 #include "towns_sprite.h"
40 #include "towns_sysrom.h"
41 #include "towns_vram.h"
43 //#include "mb87078.h"
45 //#include "../ym2612.h"
54 #include "../debugger.h"
60 #include "./fontroms.h"
61 #include "./joystick.h"
62 #include "./keyboard.h"
63 #include "./msdosrom.h"
65 #include "./serialrom.h"
68 // ----------------------------------------------------------------------------
70 // ----------------------------------------------------------------------------
73 using FMTOWNS::DICTIONARY;
74 using FMTOWNS::FLOPPY;
75 using FMTOWNS::FONT_ROMS;
76 using FMTOWNS::JOYSTICK;
77 using FMTOWNS::KEYBOARD;
78 using FMTOWNS::MSDOSROM;
80 using FMTOWNS::SERIAL_ROM;
81 using FMTOWNS::SYSROM;
84 using FMTOWNS::TOWNS_CDROM;
85 using FMTOWNS::TOWNS_CRTC;
86 using FMTOWNS::TOWNS_DMAC;
87 using FMTOWNS::TOWNS_MEMORY;
88 using FMTOWNS::TOWNS_SCSI_HOST;
89 using FMTOWNS::TOWNS_SPRITE;
90 using FMTOWNS::TOWNS_VRAM;
93 VM::VM(EMU* parent_emu) : VM_TEMPLATE(parent_emu)
114 first_device = last_device = NULL;
115 dummy = new DEVICE(this, emu); // must be 1st device
116 event = new EVENT(this, emu); // must be 2nd device
118 dummy->set_device_name(_T("1st Dummy"));
119 event->set_device_name(_T("EVENT"));
122 cpu = new I386(this, emu);
124 #if defined(HAS_I386)
125 cpu->set_device_name(_T("CPU(i386)"));
126 #elif defined(HAS_I486)
127 cpu->set_device_name(_T("CPU(i486)"));
128 #elif defined(HAS_PENTIUM)
129 cpu->set_device_name(_T("CPU(Pentium)"));
133 io = new IO(this, emu);
135 crtc = new TOWNS_CRTC(this, emu);
136 cdrom = new TOWNS_CDROM(this, emu);
138 memory = new TOWNS_MEMORY(this, emu);
139 vram = new TOWNS_VRAM(this, emu);
140 sprite = new TOWNS_SPRITE(this, emu);
141 sysrom = new SYSROM(this, emu);
142 msdosrom = new MSDOSROM(this, emu);
143 fontrom = new FONT_ROMS(this, emu);
144 dictionary = new DICTIONARY(this, emu);
145 #if defined(HAS_20PIX_FONTS)
146 fontrom_20pix = new FONT_ROM_20PIX(this, emu);
148 serialrom = new SERIAL_ROM(this, emu);
149 adpcm = new ADPCM(this, emu);
150 // mixer = new MIXER(this, emu); // Pseudo mixer.
152 adc = new AD7820KR(this, emu);
153 rf5c68 = new RF5C68(this, emu);
154 // e_volume[0] = new MB87878(this, emu);
155 // e_volume[1] = new MB87878(this, emu);
157 sio = new I8251(this, emu);
158 pit0 = new I8253(this, emu);
159 pit1 = new I8253(this, emu);
160 pic = new I8259(this, emu);
161 fdc = new MB8877(this, emu);
162 rtc = new MSM58321(this, emu);
163 beep = new PCM1BIT(this, emu);
164 opn2 = new YM2612(this, emu);
166 seek_sound = new NOISE(this, emu);
167 head_up_sound = new NOISE(this, emu);
168 head_down_sound = new NOISE(this, emu);
170 // scsi_host = new TOWNS_SCSI_HOST(this, emu);
171 scsi_host = new SCSI_HOST(this, emu);
173 for(int i = 0; i < 7; i++) {
176 #if defined(USE_HARD_DISK)
177 for(int i = 0; i < USE_HARD_DISK; i++) {
178 scsi_hdd[i] = new SCSI_HDD(this, emu);
179 scsi_hdd[i]->set_device_name(_T("SCSI Hard Disk Drive #%d"), i + 1);
180 scsi_hdd[i]->scsi_id = i ;
181 scsi_hdd[i]->set_disk_handler(0, new HARDDISK(emu));
182 scsi_hdd[i]->set_context_interface(scsi_host);
183 my_sprintf_s(scsi_hdd[i]->vendor_id, 9, "FUJITSU");
184 my_sprintf_s(scsi_hdd[i]->product_id, 17, "SCSI-HDD");
185 scsi_host->set_context_target(scsi_hdd[i]);
188 dma = new TOWNS_DMAC(this, emu);
189 extra_dma = new TOWNS_DMAC(this, emu);
191 floppy = new FLOPPY(this, emu);
192 keyboard = new KEYBOARD(this, emu);
193 joystick = new JOYSTICK(this, emu);
194 scsi = new SCSI(this, emu);
195 timer = new TIMER(this, emu);
197 uint16_t machine_id = 0x0100; // FM-Towns1
198 uint16_t cpu_id = 0x0001; // i386DX
199 uint32_t cpu_clock = 16000 * 1000; // 16MHz
200 #if defined(_FMTOWNS1_2ND_GEN)
201 machine_id = 0x0200; // 1F/2F/1H/2H
202 #elif defined(_FMTOWNS_UX_VARIANTS)
203 machine_id = 0x0300; // UX10/20/40
204 cpu_id = 0x0003; // i386SX
205 #elif defined(_FMTOWNS1_3RD_GEN)
206 machine_id = 0x0400; // 10F/20F.40H/80H
207 #elif defined(_FMTOWNS2_CX_VARIANTS)
208 machine_id = 0x0500; // CX10/20/30/40
209 #elif defined(_FMTOWNS_UG_VARIANTS)
210 machine_id = 0x0600; // UG10/20/40/80
211 cpu_id = 0x0003; // i386SX
212 cpu_clock = 20000 * 1000; // 20MHz
213 #elif defined(_FMTOWNS_HR_VARIANTS)
215 cpu_id = 0x0002; // i486SX
216 cpu_clock = 20000 * 1000; // 20MHz
217 #elif defined(_FMTOWNS_HG_VARIANTS)
219 cpu_clock = 20000 * 1000; // 20MHz
220 #elif defined(_FMTOWNS_SG_VARIANTS)
221 machine_id = 0x0800; // OK?
222 #elif defined(_FMTOWNS_SR_VARIANTS)
223 machine_id = 0x0700; // OK?
224 cpu_id = 0x0002; // i486SX
225 cpu_clock = 20000 * 1000; // 20MHz
226 #elif defined(_FMTOWNS_MA_VARIANTS)
227 machine_id = 0x0b00; // OK?
228 cpu_id = 0x0002; // i486SX
229 cpu_clock = 33000 * 1000; // 33MHz
230 #elif defined(_FMTOWNS_ME_VARIANTS)
231 machine_id = 0x0d00; // OK?
232 cpu_id = 0x0002; // i486SX
233 cpu_clock = 25000 * 1000; // 25MHz
234 #elif defined(_FMTOWNS_MF_VARIANTS)
235 machine_id = 0x0f00; // OK?
236 cpu_id = 0x0002; // i486SX
237 cpu_clock = 33000 * 1000; // 33MHz
238 #elif defined(_FMTOWNS_MX_VARIANTS)
239 machine_id = 0x0c00; // OK?
240 cpu_id = 0x0002; // i486DX (With FPU?)
241 cpu_clock = 66000 * 1000; // 66MHz
242 #elif defined(_FMTOWNS_MX_VARIANTS)
243 machine_id = 0x0c00; // OK?
244 cpu_id = 0x0002; // i486DX (With FPU?)
245 cpu_clock = 66000 * 1000; // 66MHz
247 // ToDo: Pentium Model (After HB).
250 event->set_frames_per_sec(FRAMES_PER_SEC);
251 event->set_lines_per_frame(LINES_PER_FRAME);
253 set_machine_type(machine_id, cpu_id);
255 event->set_context_cpu(cpu, cpu_clock);
262 // Use pseudo mixer instead of event.Due to using ADC.
267 //line_mix_ch = mixer->set_context_sound(line_in);
268 //modem_mix_ch = mixer->set_context_sound(modem_in);
269 //mic_mix_ch = mixer->set_context_sound(mic_in);
270 beep_mix_ch = mixer->set_context_sound(beep);
271 pcm_mix_ch = mixer->set_context_sound(rf5c68);
272 opn2_mix_ch = mixer->set_context_sound(opn2);
273 cdc_mix_ch = mixer->set_context_sound(cdrom);
274 mixer->set_interpolate_filter_freq(pcm_mix_ch, 4000); // channel, freq; disable if freq <= 0.
275 event->set_context_sound(mixer);
277 // Temporally not use mixer.
278 event->set_context_sound(beep);
279 event->set_context_sound(opn2);
280 event->set_context_sound(rf5c68);
281 event->set_context_sound(cdrom);
283 fdc->set_context_noise_seek(seek_sound);
284 fdc->set_context_noise_head_down(head_down_sound);
285 fdc->set_context_noise_head_up(head_up_sound);
286 event->set_context_sound(seek_sound);
287 event->set_context_sound(head_down_sound);
288 event->set_context_sound(head_up_sound);
290 pit0->set_context_ch0(timer, SIG_TIMER_CH0, 1);
291 pit0->set_context_ch1(timer, SIG_TIMER_CH1, 1);
292 pit0->set_context_ch2(beep, SIG_PCM1BIT_SIGNAL, 1);
293 pit0->set_constant_clock(0, 307200);
294 pit0->set_constant_clock(1, 307200);
295 pit0->set_constant_clock(2, 307200);
296 pit1->set_constant_clock(1, 1228800);
297 // pic->set_context_cpu(cpu);
298 pic->set_context_cpu(memory);
299 fdc->set_context_irq(floppy, SIG_FLOPPY_IRQ, 1);
300 rtc->set_context_data(timer, SIG_TIMER_RTC, 0x0f, 0);
301 rtc->set_context_busy(timer, SIG_TIMER_RTC_BUSY, 0x80);
302 scsi_host->set_context_irq(scsi, SIG_SCSI_IRQ, 1);
303 scsi_host->set_context_drq(scsi, SIG_SCSI_DRQ, 1);
304 dma->set_context_memory(memory);
305 dma->set_context_ch0(fdc);
306 dma->set_context_ch1(scsi_host);
307 //dma->set_context_ch2(printer);
308 dma->set_context_ch3(cdrom);
309 dma->set_context_tc(cdrom, SIG_TOWNS_CDROM_DMAINT, 1 << 3);
310 dma->set_context_ube1(scsi_host, SIG_SCSI_16BIT_BUS, 0x02);
312 dma->set_context_child_dma(extra_dma);
314 floppy->set_context_fdc(fdc);
316 sprite->set_context_vram(vram);
317 sprite->set_context_font(fontrom);
319 sprite->set_context_debugger(new DEBUGGER(this, emu));
322 vram->set_context_sprite(sprite);
323 vram->set_context_crtc(crtc);
325 crtc->set_context_sprite(sprite);
326 crtc->set_context_vram(vram);
327 crtc->set_context_font(fontrom);
329 //e_volume[0]->set_context_ch0(line_in, MB87878_VOLUME_LEFT);
330 //e_volume[0]->set_context_ch1(line_in, MB87878_VOLUME_RIGHT);
331 //e_volume[0]->set_context_ch2(NULL, MB87878_VOLUME_LEFT);
332 //e_volume[0]->set_context_ch3(NULL, MB87878_VOLUME_RIGHT);
333 // e_volume[1]->set_context_ch0(cdrom, MB87878_VOLUME_LEFT);
334 // e_volume[1]->set_context_ch1(cdrom, MB87878_VOLUME_RIGHT);
335 //e_volume[1]->set_context_ch2(mic, MB87878_VOLUME_LEFT | MB87878_VOLUME_RIGHT);
336 //e_volume[1]->set_context_ch3(modem, MB87878_VOLUME_LEFT | MB87878_VOLUME_RIGHT);
338 memory->set_context_cpu(cpu);
339 memory->set_context_dmac(dma);
340 memory->set_context_vram(vram);
341 memory->set_context_crtc(crtc);
342 memory->set_context_system_rom(sysrom);
343 memory->set_context_msdos(msdosrom);
344 memory->set_context_dictionary(dictionary);
345 memory->set_context_font_rom(fontrom);
346 memory->set_context_beep(beep);
347 memory->set_context_serial_rom(serialrom);
348 memory->set_context_sprite(sprite);
349 memory->set_context_pcm(rf5c68);
351 adpcm->set_context_opn2(opn2);
352 adpcm->set_context_rf5c68(rf5c68);
353 adpcm->set_context_adc(adc);
355 rf5c68->set_context_interrupt_boundary(adpcm, SIG_ADPCM_WRITE_INTERRUPT, 0xffffffff);
357 rf5c68->set_context_debugger(new DEBUGGER(this, emu));
359 opn2->set_context_irq(adpcm, SIG_ADPCM_OPX_INTR, 0xffffffff);
361 adc->set_sample_rate(19200);
362 adc->set_sound_bank(-1);
363 adc->set_context_interrupt(adpcm, SIG_ADPCM_ADC_INTR, 0xffffffff);
365 scsi->set_context_dma(dma);
366 scsi->set_context_host(scsi_host);
367 scsi->set_context_pic(pic);
368 timer->set_context_pcm(beep);
369 timer->set_context_rtc(rtc);
370 timer->set_context_halt_line(cpu, SIG_CPU_HALTREQ, 0xffffffff);
373 cpu->set_context_mem(memory);
374 cpu->set_context_io(io);
375 cpu->set_context_intr(pic);
376 cpu->set_context_dma(dma);
377 cpu->set_context_bios(NULL);
378 cpu->set_context_extreset(memory, SIG_FMTOWNS_NOTIFY_RESET, 0xffffffff);
380 cpu->set_context_debugger(new DEBUGGER(this, emu));
385 // IRQ2 : USART (ToDo)
386 // IRQ3 : EXTRA USART (ToDo)
387 // IRQ4 : EXTRA I/O (Maybe not implement)
388 // IRQ5 : EXTRA I/O (Maybe not implement)
390 // IRQ7 : Deisy chain (to IRQ8 - 15)
391 timer->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR0, 0xffffffff);
392 keyboard->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR1, 0xffffffff);
393 floppy->set_context_intr_line(pic, SIG_I8259_CHIP0 | SIG_I8259_IR6, 0xffffffff);
395 // IRQ8 : SCSI (-> scsi.cpp)
397 // IRQ10 : EXTRA I/O (Maybe not implement)
399 // IRQ12 : PRINTER (ToDo)
400 // IRQ13 : ADPCM AND OPN2 (Route to adpcm.cpp)
401 // IRQ14 : EXTRA I/O (Maybe not implement)
403 cdrom->set_context_mpuint_line(pic, SIG_I8259_CHIP1 | SIG_I8259_IR1, 0xffffffff);
404 crtc->set_context_vsync(pic, SIG_I8259_CHIP1 | SIG_I8259_IR3, 0xffffffff);
405 adpcm->set_context_intr_line(pic, SIG_I8259_CHIP1 | SIG_I8259_IR5, 0xffffffff);
408 // DMA1 : SCSI (-> scsi.cpp)
409 // DMA2 : PRINTER (ToDo)
411 // EXTRA DMA0 : EXTRA SLOT (Maybe not implement)
412 // EXTRA DMA1 : Reserved
413 // EXTRA DMA2 : Reserved
414 // EXTRA DMA3 : Reserved
415 fdc->set_context_drq(dma, SIG_UPD71071_CH0, 1);
416 cdrom->set_context_drq_line(dma, SIG_UPD71071_CH3, 0xff);
418 // NMI0 : KEYBOARD (RAS)
419 // NMI1 : Extra SLOT (Maybe not implement)
420 keyboard->set_context_nmi_line(memory, SIG_CPU_NMI, 0xffffffff);
423 io->set_iomap_alias_rw (0x0000, pic, I8259_ADDR_CHIP0 | 0);
424 io->set_iomap_alias_rw (0x0002, pic, I8259_ADDR_CHIP0 | 1);
425 io->set_iomap_alias_rw (0x0010, pic, I8259_ADDR_CHIP1 | 0);
426 io->set_iomap_alias_rw (0x0012, pic, I8259_ADDR_CHIP1 | 1);
428 io->set_iomap_range_rw (0x0020, 0x0025, memory);
429 io->set_iomap_range_rw (0x0026, 0x0027, timer); // Freerun counter
430 io->set_iomap_single_rw(0x0028, memory);
432 io->set_iomap_range_r (0x0030, 0x0031, memory); // cpu id / machine id
433 io->set_iomap_single_rw(0x0032, memory); // serial rom (routed from memory)
434 io->set_iomap_single_r (0x0034, scsi); // ENABLE/ UNABLE to WORD DMA for SCSI
436 io->set_iomap_alias_rw(0x0040, pit0, 0);
437 io->set_iomap_alias_rw(0x0042, pit0, 1);
438 io->set_iomap_alias_rw(0x0044, pit0, 2);
439 io->set_iomap_alias_rw(0x0046, pit0, 3);
440 io->set_iomap_alias_rw(0x0050, pit1, 0);
441 io->set_iomap_alias_rw(0x0052, pit1, 1);
442 io->set_iomap_alias_rw(0x0054, pit1, 2);
443 io->set_iomap_alias_rw(0x0056, pit1, 3);
445 io->set_iomap_single_rw(0x0060, timer); // Beep and interrupts register
446 io->set_iomap_single_rw(0x0068, timer); // Interval timer register2 (after Towns 10F).
447 io->set_iomap_single_rw(0x006a, timer); // Interval timer register2 (after Towns 10F).
448 io->set_iomap_single_rw(0x006b, timer); // Interval timer register2 (after Towns 10F).
449 io->set_iomap_single_rw(0x006c, timer); // 1uS wait register (after Towns 10F).
451 io->set_iomap_single_rw(0x0070, timer); // RTC DATA
452 io->set_iomap_single_w (0x0080, timer); // RTC COMMAND
454 io->set_iomap_range_rw (0x00a0, 0x00af, dma);
455 io->set_iomap_range_rw (0x00b0, 0x00bf, extra_dma);
457 io->set_iomap_alias_rw (0x0200, fdc, 0); // STATUS/COMMAND
458 io->set_iomap_alias_rw (0x0202, fdc, 1); // TRACK
459 io->set_iomap_alias_rw (0x0204, fdc, 2); // SECTOR
460 io->set_iomap_alias_rw (0x0206, fdc, 3); // DATA
461 io->set_iomap_single_rw(0x0208, floppy); // DRIVE STATUS / DRIVE CONTROL
462 io->set_iomap_single_rw(0x020c, floppy); // DRIVE SELECT
463 io->set_iomap_single_rw(0x020e, floppy); // Towns drive SW
465 io->set_iomap_range_rw (0x0400, 0x0404, memory); // System Status
466 io->set_iomap_range_rw (0x0406, 0x403f, memory); // Reserved
468 io->set_iomap_range_rw(0x0440, 0x0443, crtc); // CRTC
469 io->set_iomap_range_rw(0x0448, 0x044f, crtc); // VIDEO OUT (CRTC)
471 io->set_iomap_range_rw(0x0450, 0x0452, sprite); // SPRITE
473 io->set_iomap_single_rw(0x0458, vram); // VRAM ACCESS CONTROLLER (ADDRESS)
474 io->set_iomap_range_rw (0x045a, 0x045f, vram); // VRAM ACCESS CONTROLLER (DATA)
476 io->set_iomap_single_rw(0x0480, sysrom); // MEMORY REGISTER
477 io->set_iomap_single_rw(0x0484, dictionary); // Dictionary
479 //io->set_iomap_alias_r(0x48a, memory_card, 0); //
480 //io->set_iomap_alias_rw(0x490, memory_card); // After Towns2
481 //io->set_iomap_alias_rw(0x491, memory_card); // After Towns2
483 io->set_iomap_range_rw(0x04c0, 0x04cf, cdrom); // CDROM
486 io->set_iomap_single_r(0x04d0, joystick); // Pad1
487 io->set_iomap_single_r(0x04d2, joystick); // Pad 2
488 io->set_iomap_single_w(0x04d6, joystick); // Pad out
490 io->set_iomap_single_rw(0x04d5, adpcm); // mute
492 io->set_iomap_alias_rw(0x04d8, opn2, 0); // STATUS(R)/Addrreg 0(W)
493 io->set_iomap_alias_w (0x04da, opn2, 1); // Datareg 0(W)
494 io->set_iomap_alias_w (0x04dc, opn2, 2); // Addrreg 1(W)
495 io->set_iomap_alias_w (0x04de, opn2, 3); // Datareg 1(W)
497 // io->set_iomap_alias_rw(0x04e0, e_volume[0], 0);
498 // io->set_iomap_alias_rw(0x04e1, e_volume[0], 1);
499 // io->set_iomap_alias_rw(0x04e2, e_volume[1], 0);
500 // io->set_iomap_alias_rw(0x04e3, e_volume[1], 1);
503 io->set_iomap_range_rw(0x04e7, 0x04ef, adpcm); // A/D SAMPLING DATA REG
504 io->set_iomap_range_rw(0x04f0, 0x04ff, adpcm); // A/D SAMPLING DATA REG
506 io->set_iomap_single_rw(0x05c0, memory); // NMI MASK
507 io->set_iomap_single_r (0x05c2, memory); // NMI STATUS
508 io->set_iomap_single_r (0x05c8, sprite); // TVRAM EMULATION
509 io->set_iomap_single_w (0x05ca, crtc); // VSYNC INTERRUPT
511 io->set_iomap_single_rw(0x05e8, memory); // RAM capacity register.(later Towns1H/2H/1F/2F).
512 io->set_iomap_single_rw(0x05ec, memory); // RAM Wait register , ofcially after Towns2, but exists after Towns1H.
513 io->set_iomap_single_r (0x05ed, memory); // RAM Wait register , ofcially after Towns2, but exists after Towns1H.
515 io->set_iomap_single_rw(0x0600, keyboard);
516 io->set_iomap_single_rw(0x0602, keyboard);
517 io->set_iomap_single_rw(0x0604, keyboard);
519 //io->set_iomap_single_rw(0x0800, printer);
520 //io->set_iomap_single_rw(0x0802, printer);
521 //io->set_iomap_single_rw(0x0804, printer);
523 io->set_iomap_alias_rw (0x0a00, sio, 0);
524 io->set_iomap_alias_rw (0x0a02, sio, 1);
525 // io->set_iomap_single_r (0x0a04, serial);
526 // io->set_iomap_single_r (0x0a06, serial);
527 // io->set_iomap_single_w (0x0a08, serial);
528 // io->set_iomap_single_rw(0x0a0a, modem);
530 io->set_iomap_single_rw(0x0c30, scsi);
531 io->set_iomap_single_rw(0x0c32, scsi);
533 io->set_iomap_range_rw(0x3000, 0x3fff, dictionary); // CMOS
535 io->set_iomap_range_rw(0xfd90, 0xfda2, crtc); // Palette and CRTC
536 io->set_iomap_single_rw(0xfda4, memory); // memory
538 io->set_iomap_range_rw(0xff80, 0xff87, vram); // MMIO
539 io->set_iomap_range_rw(0xff88, 0xff94, memory); // MMIO
540 io->set_iomap_range_w (0xff94, 0xff95, fontrom);
541 io->set_iomap_range_r (0xff96, 0xff97, fontrom);
542 io->set_iomap_range_rw(0xff98, 0xffff, memory); // MMIO
543 io->set_iomap_range_rw(0xff9c, 0xffa0, memory); // MMIO
545 // Vram allocation may be before initialize().
546 // initialize all devices
547 #if defined(__GIT_REPO_VERSION)
548 strncpy(_git_revision, __GIT_REPO_VERSION, sizeof(_git_revision) - 1);
550 // ToDo : Use config framework
553 if(machine_id < 0x0200) { // Model1 - 2H
555 } else if(machine_id == 0x0500) { // CX
557 } else if(machine_id < 0x0700) { // 10F,20H
559 } else if(machine_id == 0x0800) { // HG
565 memory->set_extra_ram_size(exram_size);
567 #if defined(WITH_I386SX)
568 cpu->device_model = INTEL_80386;
569 #elif defined(WITH_I486SX)
570 cpu->device_model = INTEL_I486SX;
571 #elif defined(WITH_I486DX)
572 cpu->device_model = INTEL_I486DX;
573 #elif defined(WITH_PENTIUM)
574 cpu->device_model = INTEL_PENTIUM;
577 cpu->device_model = INTEL_80386;
580 for(DEVICE* device = first_device; device; device = device->next_device) {
581 device->initialize();
583 // cpu->set_address_mask(0xffffffff);
588 // delete all devices
589 for(DEVICE* device = first_device; device;) {
590 DEVICE *next_device = device->next_device;
591 // printf("DEVID=%d\n", device->this_device_id);
594 device = next_device;
598 DEVICE* VM::get_device(int id)
600 for(DEVICE* device = first_device; device; device = device->next_device) {
601 if(device->this_device_id == id) {
608 void VM::set_machine_type(uint16_t machine_id, uint16_t cpu_id)
611 memory->set_cpu_id(cpu_id);
612 memory->set_machine_id(machine_id);
615 crtc->set_cpu_id(cpu_id);
616 crtc->set_machine_id(machine_id);
619 timer->set_cpu_id(cpu_id);
620 timer->set_machine_id(machine_id);
623 scsi->set_cpu_id(cpu_id);
624 scsi->set_machine_id(machine_id);
626 #if defined(HAS_20PIX_FONTS)
627 if(fontrom_20pix != NULL) {
628 fontrom_20pix->set_cpu_id(cpu_id);
629 fontrom_20pix->set_machine_id(machine_id);
636 // ----------------------------------------------------------------------------
637 // drive virtual machine
638 // ----------------------------------------------------------------------------
643 for(DEVICE* device = first_device; device; device = device->next_device) {
646 // cpu->set_address_mask(0xffffffff);
654 // ----------------------------------------------------------------------------
656 // ----------------------------------------------------------------------------
659 DEVICE *VM::get_cpu(int index)
668 // ----------------------------------------------------------------------------
670 // ----------------------------------------------------------------------------
672 void VM::draw_screen()
677 uint32_t VM::is_floppy_disk_accessed()
679 return fdc->read_signal(0);
682 // ----------------------------------------------------------------------------
684 // ----------------------------------------------------------------------------
686 void VM::initialize_sound(int rate, int samples)
689 // init sound manager
690 event->initialize_sound(rate, samples);
693 beep->initialize_sound(rate, 8000);
696 // MASTER CLOCK MAYBE 600KHz * 12 = 7200KHz .
697 // From FM-Towns Technical Databook (Rev.2), Page 201
698 opn2->initialize_sound(rate, (int)(600.0e3 * 12.0) , samples, 0.0, 0.0);
701 rf5c68->initialize_sound(rate, samples);
703 // add_sound_in_source() must add after per initialize_sound().
704 adc_in_ch = event->add_sound_in_source(rate, samples, 2);
705 adc->set_sample_rate(19200);
706 adc->set_sound_bank(adc_in_ch);
708 mixer->set_context_out_line(adc_in_ch);
709 mixer->set_context_sample_out(adc_in_ch, rate, samples); // Must be 2ch.
710 // ToDo: Check recording sample rate & channels.
711 mic_in_ch = event->add_sound_in_source(rate, samples, 2);
712 mixer->set_context_mic_in(mic_in_ch, rate, samples);
714 line_in_ch = event->add_sound_in_source(rate, samples, 2);
715 mixer->set_context_line_in(line_in_ch, rate, samples);
720 uint16_t* VM::create_sound(int* extra_frames)
722 return event->create_sound(extra_frames);
725 int VM::get_sound_buffer_ptr()
727 return event->get_sound_buffer_ptr();
730 void VM::clear_sound_in()
732 event->clear_sound_in_source(adc_in_ch);
733 event->clear_sound_in_source(mic_in_ch);
734 event->clear_sound_in_source(line_in_ch);
738 int VM::get_sound_in_data(int ch, int32_t* dst, int expect_samples, int expect_rate, int expect_channels)
740 if(dst == NULL) return 0;
741 if(expect_samples <= 0) return 0;
754 if(n_ch < 0) return 0;
755 int samples = event->get_sound_in_data(n_ch, dst, expect_samples, expect_rate, expect_channels);
759 // Write to event's buffer
760 int VM::sound_in(int ch, int32_t* src, int samples)
763 if(ch >= 2) return 0;
766 case 0x100: // ADC in from MIXER, not connected.
775 if(n_ch < 0) return 0;
780 ss = event->write_sound_in_buffer(n_ch, src, samples);
787 #if defined(USE_HARD_DISK)
788 void VM::open_hard_disk(int drv, const _TCHAR* file_path)
790 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
791 if(scsi_hdd[drv] != NULL) {
792 scsi_hdd[drv]->open(0, file_path, 512);
797 void VM::close_hard_disk(int drv)
799 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
800 if(scsi_hdd[drv] != NULL) {
801 scsi_hdd[drv]->close(0);
806 bool VM::is_hard_disk_inserted(int drv)
808 if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
809 if(scsi_hdd[drv] != NULL) {
810 return scsi_hdd[drv]->mounted(0);
816 uint32_t VM::is_hard_disk_accessed()
820 for(int drv = 0; drv < USE_HARD_DISK; drv++) {
821 if(scsi_hdd[drv] != NULL) {
822 if(scsi_hdd[drv]->accessed(0)) {
829 #endif // USE_HARD_DISK
831 void VM::open_compact_disc(int drv, const _TCHAR* file_path)
833 cdrom->open(file_path);
836 void VM::close_compact_disc(int drv)
841 bool VM::is_compact_disc_inserted(int drv)
843 return cdrom->mounted();
846 uint32_t VM::is_compact_disc_accessed()
848 return cdrom->accessed();
851 #ifdef USE_SOUND_VOLUME
852 void VM::set_sound_device_volume(int ch, int decibel_l, int decibel_r)
854 #ifndef HAS_LINEIN_SOUND
857 #ifndef HAS_MIC_SOUND
860 #ifndef HAS_MODEM_SOUND
863 #ifndef HAS_2ND_ADPCM
864 // if(ch >= 10) ch++;
867 if(ch == 0) { // BEEP
868 mixer->set_volume(beep_mix_ch, decibel_l, decibel_r);
870 else if(ch == 1) { // CD-ROM
871 //e_volume[1]->set_volume(0, decibel_l);
872 //e_volume[1]->set_volume(1, decibel_r);
873 mixer->set_volume(cdc_mix_ch, decibel_l, decibel_r);
875 else if(ch == 2) { // OPN2
876 mixer->set_volume(opn2_mix_ch, decibel_l, decibel_r);
878 else if(ch == 3) { // ADPCM
879 mixer->set_volume(pcm_mix_ch, decibel_l, decibel_r);
881 else if(ch == 4) { // LINE IN
882 //mixer->set_volume(line_mix_ch, decibel_l, decibel_r);
884 else if(ch == 5) { // MIC
885 //mic->set_volume(0, (decibel_l + decibel_r) / 2);
887 else if(ch == 6) { // MODEM
888 //modem->set_volume(0, (decibel_l + decibel_r) / 2);
891 else if(ch == 7) { // ADPCM
892 adpcm2->set_volume(0, decibel_l, decibel_r);
895 else if(ch == 8) { // FDD
896 fdc->set_volume(0, decibel_l, decibel_r);
898 else if(ch == 9) { // HDD(ToDo)
899 fdc->set_volume(0, decibel_l, decibel_r);
902 if(ch == 0) { // BEEP
903 beep->set_volume(0, decibel_l, decibel_r);
905 else if(ch == 1) { // CD-ROM
906 cdrom->set_volume(0, decibel_l, decibel_r);
908 else if(ch == 2) { // OPN2
909 opn2->set_volume(0, decibel_l, decibel_r);
911 else if(ch == 3) { // ADPCM
912 rf5c68->set_volume(0, decibel_l, decibel_r);
914 else if(ch == 4) { // SEEK, HEAD UP / DOWN
915 seek_sound->set_volume(0, decibel_l, decibel_r);
916 head_up_sound->set_volume(0, decibel_l, decibel_r);
917 head_down_sound->set_volume(0, decibel_l, decibel_r);
924 // ----------------------------------------------------------------------------
926 // ----------------------------------------------------------------------------
928 void VM::key_down(int code, bool repeat)
930 keyboard->key_down(code);
933 void VM::key_up(int code)
935 keyboard->key_up(code);
938 // ----------------------------------------------------------------------------
940 // ----------------------------------------------------------------------------
942 void VM::open_floppy_disk(int drv, const _TCHAR* file_path, int bank)
944 fdc->open_disk(drv, file_path, bank);
945 // floppy->change_disk(drv);
948 void VM::close_floppy_disk(int drv)
950 fdc->close_disk(drv);
953 bool VM::is_floppy_disk_inserted(int drv)
955 return fdc->is_disk_inserted(drv);
958 void VM::is_floppy_disk_protected(int drv, bool value)
960 fdc->is_disk_protected(drv, value);
963 bool VM::is_floppy_disk_protected(int drv)
965 return fdc->is_disk_protected(drv);
968 bool VM::is_frame_skippable()
970 return event->is_frame_skippable();
973 void VM::update_config()
975 for(DEVICE* device = first_device; device; device = device->next_device) {
976 device->update_config();
980 #define STATE_VERSION 3
982 bool VM::process_state(FILEIO* state_fio, bool loading)
984 if(!state_fio->StateCheckUint32(STATE_VERSION)) {
987 for(DEVICE* device = first_device; device; device = device->next_device) {
988 // Note: typeid(foo).name is fixed by recent ABI.Not decr. 6.
989 // const char *name = typeid(*device).name();
990 // But, using get_device_name() instead of typeid(foo).name() 20181008 K.O
991 const char *name = device->get_device_name();
992 int len = strlen(name);
993 if(!state_fio->StateCheckInt32(len)) {
996 if(!state_fio->StateCheckBuffer(name, len, 1)) {
999 if(!device->process_state(state_fio, loading)) {
1001 printf("Data loading Error: DEVID=%d\n", device->this_device_id);
1006 // Machine specified.
1007 state_fio->StateValue(beep_mix_ch);
1008 state_fio->StateValue(cdc_mix_ch);
1009 state_fio->StateValue(opn2_mix_ch);
1010 state_fio->StateValue(pcm_mix_ch);
1011 state_fio->StateValue(line_mix_ch);
1012 state_fio->StateValue(modem_mix_ch);
1013 state_fio->StateValue(mic_mix_ch);