OSDN Git Service

[VM][FMTOWNS][JOYPAD][MOUSE][WIP] Around MOUSE/Joystick.
[csp-qt/common_source_project-fm7.git] / source / src / vm / fmtowns / fmtowns.cpp
1 /*
2         FUJITSU FM-Towns Emulator 'eFMTowns'
3
4         Author : Kyuma Ohta <whatisthis.sowhat _at_ gmail.com>
5         Date   : 2016.12.28 -
6
7         [ virtual machine ]
8         History: 
9                 2016-12-28 Copy from eFMR-50.
10 */
11
12 #include "fmtowns.h"
13 #include "../../emu.h"
14 #include "../device.h"
15 #include "../event.h"
16
17 //#include "../hd46505.h"
18 #include "../i8251.h"
19 #include "../i8253.h"
20 #include "../i8259.h"
21
22 #include "../i386_np21.h"
23 //#include "../i386.h"
24
25 #include "../io.h"
26 #include "../mb8877.h"
27 #include "../msm58321.h"
28 #include "../noise.h"
29 #include "../pcm1bit.h"
30 #include "../harddisk.h"
31 #include "../scsi_hdd.h"
32 #include "./towns_scsi_host.h"
33 #include "../upd71071.h"
34
35 #include "./cdrom.h"
36 #include "./crtc.h"
37 #include "./dictionary.h"
38 #include "./dmac.h"
39 #include "./towns_memory.h"
40 #include "./sprite.h"
41 #include "./sysrom.h"
42 #include "./vram.h"
43
44 // Electric Volume
45 //#include "mb87078.h"
46 //YM-2612 "OPN2"
47 //#include "../ym2612.h"
48 //RF5C68 PCM
49 #include "rf5c68.h"
50 //AD7820 ADC
51 #include "ad7820kr.h"
52 #include "ym2612.h"
53 // 80387?
54
55 #ifdef USE_DEBUGGER
56 #include "../debugger.h"
57 #endif
58
59 #include "./adpcm.h"
60 //#include "./cdc.h"
61 #include "./floppy.h"
62 #include "./fontroms.h"
63 #include "./joystick.h"
64 #include "./joypad.h"
65 #include "./keyboard.h"
66 #include "./mouse.h"
67 #include "./msdosrom.h"
68 #include "./scsi.h"
69 #include "./serialrom.h"
70 #include "./timer.h"
71 #include "./iccard.h"
72
73 #include "./planevram.h"
74
75 // ----------------------------------------------------------------------------
76 // initialize
77 // ----------------------------------------------------------------------------
78 using FMTOWNS::ADPCM;
79 //using FMTOWNS::CDC;
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;
86 using FMTOWNS::MOUSE;
87 using FMTOWNS::MSDOSROM;
88 using FMTOWNS::SCSI;
89 using FMTOWNS::SERIAL_ROM;
90 using FMTOWNS::SYSROM;
91 using FMTOWNS::TIMER;
92 using FMTOWNS::TOWNS_ICCARD;
93
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;
102
103
104 VM::VM(EMU_TEMPLATE* parent_emu) : VM_TEMPLATE(parent_emu)
105 {
106 /*
107         Machine ID & CPU ID
108
109         FMR-50FD/HD/LT  0xF8
110         FMR-50FX/HX     0xE0
111         FMR-50SFX/SHX   0xE8
112         FMR-50LT        0xF8
113         FMR-50NBX       0x28
114         FMR-50NB        0x60
115         FMR-50NE/T      0x08
116         FMR-CARD        0x70
117
118         80286           0x00
119         80386           0x01
120         80386SX         0x03
121         80486           0x02
122 */
123         
124         // create devices
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
128 #if defined(_USE_QT)
129         dummy->set_device_name(_T("1st Dummy"));
130         event->set_device_name(_T("EVENT"));
131 #endif  
132
133         cpu = new I386(this, emu);
134 #if defined(_USE_QT)
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)"));
141   #endif
142 #endif  
143
144         io = new IO(this, emu);
145         
146         crtc = new TOWNS_CRTC(this, emu);
147         cdrom = new TOWNS_CDROM(this, emu);
148
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);
158 #endif
159         serialrom = new SERIAL_ROM(this, emu);
160
161         adpcm = new ADPCM(this, emu);
162 //      mixer = new MIXER(this, emu); // Pseudo mixer.
163
164         planevram = new PLANEVRAM(this, emu);
165         
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);
170         
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);
179
180         seek_sound = new NOISE(this, emu);
181         head_up_sound = new NOISE(this, emu);
182         head_down_sound = new NOISE(this, emu);
183         
184 //      scsi_host = new TOWNS_SCSI_HOST(this, emu);
185         scsi_host = new SCSI_HOST(this, emu);
186         
187         for(int i = 0; i < 7; i++) {
188                 scsi_hdd[i] = NULL;
189         }       
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]);
200         }
201 #endif
202         dma = new TOWNS_DMAC(this, emu);
203         extra_dma = new TOWNS_DMAC(this, emu);
204
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);
210
211         iccard1 = new TOWNS_ICCARD(this, emu);
212 #if 0
213         iccard2 = new TOWNS_ICCARD(this, emu);
214 #else
215         iccard2 = NULL;
216 #endif
217         joypad[0] = new JOYPAD(this, emu);
218         joypad[1] = new JOYPAD(this, emu);
219         mouse = new MOUSE(this, emu);
220
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)
238         machine_id = 0x0700;
239         cpu_id = 0x0002;      // i486SX
240         cpu_clock = 20000 * 1000; // 20MHz
241 #elif defined(_FMTOWNS_HG_VARIANTS)
242         machine_id = 0x0800;
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)
271         // 20210227 K.O
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.
277 #else
278         // ToDo: Pentium Model (After HB).
279
280 #endif
281
282         event->set_frames_per_sec(FRAMES_PER_SEC);
283         event->set_lines_per_frame(LINES_PER_FRAME);
284         
285         set_machine_type(machine_id, cpu_id);
286         // set contexts
287         event->set_context_cpu(cpu, cpu_clock);
288
289         adc_in_ch = -1;
290         line_in_ch = -1;
291         modem_in_ch = -1;
292         mic_in_ch = -1;
293
294         // Use pseudo mixer instead of event.Due to using ADC.
295 #if 0
296         line_mix_ch = -1;
297         modem_mix_ch = -1;
298         mic_mix_ch = -1;
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);
308 #else
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);
314 #endif
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);
321         
322 #ifdef USE_DEBUGGER
323         pit0->set_context_debugger(new DEBUGGER(this, emu));
324         pit1->set_context_debugger(new DEBUGGER(this, emu));
325 #endif  
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);
343         
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);
352         
353         dma->set_context_ube1(scsi_host, SIG_SCSI_16BIT_BUS, 0x02);
354         
355         dma->set_context_child_dma(extra_dma);
356         
357         floppy->set_context_fdc(fdc);
358         
359         sprite->set_context_vram(vram); 
360         sprite->set_context_font(fontrom);
361         sprite->set_context_crtc(crtc);
362 #ifdef USE_DEBUGGER
363         sprite->set_context_debugger(new DEBUGGER(this, emu));
364 #endif
365         
366         vram->set_context_sprite(sprite);
367         vram->set_context_crtc(crtc);
368         
369         planevram->set_context_vram(vram);
370         planevram->set_context_sprite(sprite);
371         planevram->set_context_crtc(crtc);
372         
373         crtc->set_context_sprite(sprite);
374         crtc->set_context_vram(vram);
375         crtc->set_context_font(fontrom);
376
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);
385         
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);
401
402         adpcm->set_context_opn2(opn2);
403         adpcm->set_context_rf5c68(rf5c68);
404         adpcm->set_context_adc(adc);
405
406         rf5c68->set_context_interrupt_boundary(adpcm, SIG_ADPCM_WRITE_INTERRUPT, 0xffffffff);
407 #ifdef USE_DEBUGGER
408         rf5c68->set_context_debugger(new DEBUGGER(this, emu));
409 #endif
410         opn2->set_context_irq(adpcm, SIG_ADPCM_OPX_INTR, 0xffffffff);
411         
412         adc->set_sample_rate(19200);
413         adc->set_sound_bank(-1);
414         adc->set_context_interrupt(adpcm, SIG_ADPCM_ADC_INTR, 0xffffffff); 
415         
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);
422
423         joystick->set_context_mouse(mouse, SIG_MOUSE_STROBE, 0xff);
424         
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);
431         
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);
436         
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);
440
441         // cpu bus
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);
448 #ifdef USE_DEBUGGER
449         cpu->set_context_debugger(new DEBUGGER(this, emu));
450 #endif
451         // Interrupts
452         // IRQ0  : TIMER
453         // IRQ1  : KEYBOARD
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)
458         // IRQ6  : FDC
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);
463         
464         // IRQ8  : SCSI (-> scsi.cpp)
465         // IRQ9  : CDC/CDROM
466         // IRQ10 : EXTRA I/O (Maybe not implement)
467         // IRQ11 : VSYNC
468         // IRQ12 : PRINTER (ToDo)
469         // IRQ13 : ADPCM AND OPN2 (Route to adpcm.cpp)
470         // IRQ14 : EXTRA I/O (Maybe not implement)
471         // IRQ15 : RESERVED.
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);
475
476         // DMA0  : FDC/DRQ
477         // DMA1  : SCSI (-> scsi.cpp)
478         // DMA2  : PRINTER (ToDo)
479         // DMA3  : CDC/CDROM
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);
488
489         // NMI0 : KEYBOARD (RAS)
490         // NMI1 : Extra SLOT (Maybe not implement)
491         keyboard->set_context_nmi_line(memory, SIG_CPU_NMI, 0xffffffff);
492
493         cdrom->set_context_dmac(dma);
494         // For Debugging, will remove 20200822 K.O
495         cdrom->set_context_cpu(cpu);
496
497         // i/o bus
498         io->set_iowait_range_rw(0x0000, 0xffff, 6); // ToDo: May variable wait.
499                 
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);
504         
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);
508         
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
512
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);
521         
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).
527         
528         io->set_iomap_single_rw(0x0070, timer); // RTC DATA
529         io->set_iomap_single_w (0x0080, timer); // RTC COMMAND
530         
531         io->set_iomap_range_rw (0x00a0, 0x00af, dma);
532         io->set_iomap_range_rw (0x00b0, 0x00bf, extra_dma);
533         
534         io->set_iomap_single_rw(0x00c0, memory);   // CACHE CONTROLLER
535         io->set_iomap_single_rw(0x00c2, memory);   // CACHE CONTROLLER
536         
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
545         
546         io->set_iomap_range_rw (0x0400, 0x0404, memory); // System Status
547 //      io->set_iomap_range_rw (0x0406, 0x043f, memory); // Reserved
548         
549         io->set_iomap_range_rw(0x0440, 0x0443, crtc); // CRTC
550         io->set_iomap_range_rw(0x0448, 0x044f, crtc); // VIDEO OUT (CRTC)
551         
552         io->set_iomap_range_rw(0x0450, 0x0452, sprite); // SPRITE
553         
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)
556         
557         io->set_iomap_single_rw(0x0480, memory); //  MEMORY REGISTER
558         io->set_iomap_single_rw(0x0484, dictionary); // Dictionary
559         
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
563         
564         io->set_iomap_range_rw(0x04c0, 0x04c6, cdrom); // CDROM
565         io->set_iomap_range_r (0x04cc, 0x04cd, cdrom); // CDROM
566         // PAD, Sound
567
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
571
572         io->set_iomap_single_rw(0x04d5, adpcm); // mute 
573         // OPN2(YM2612)
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)
578         // Electrical volume
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);
583
584         // ADPCM
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 
587
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
592         
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
599         
600         io->set_iomap_single_rw(0x0600, keyboard);
601         io->set_iomap_single_rw(0x0602, keyboard);
602         io->set_iomap_single_rw(0x0604, keyboard);
603
604         //io->set_iomap_single_rw(0x0800, printer);
605         //io->set_iomap_single_rw(0x0802, printer);
606         //io->set_iomap_single_rw(0x0804, printer);
607         
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);
614         
615         io->set_iomap_single_rw(0x0c30, scsi);
616         io->set_iomap_single_rw(0x0c32, scsi);
617         io->set_iomap_single_r (0x0c34, scsi);
618
619         io->set_iomap_range_rw (0x3000, 0x3fff, dictionary); // CMOS
620         
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
624         
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
633         
634 //      io->set_iomap_range_w (0xff94, 0xff95, fontrom);
635 //      io->set_iomap_range_r (0xff96, 0xff97, fontrom);
636
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);
641 #endif
642         // ToDo : Use config framework
643         int exram_size = config.current_ram_size;
644         if(exram_size < 1) {
645                 if(machine_id < 0x0200) { // Model1 - 2H
646                         exram_size = 6;
647                 } else if(machine_id == 0x0500) { // CX
648                         exram_size = 15;
649                 } else if(machine_id < 0x0700) {  // 10F,20H
650                         exram_size = 8;
651                 } else if(machine_id == 0x0800) { // HG
652                         exram_size = 15;
653                 } else { 
654                         exram_size = 31;
655                 }
656         }
657         if(exram_size < MIN_RAM_SIZE) {
658                 exram_size = MIN_RAM_SIZE;
659         }
660         
661         memory->set_extra_ram_size(exram_size);
662
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;
671 #else
672         // I386
673         cpu->device_model = INTEL_80386;
674 #endif  
675
676         for(DEVICE* device = first_device; device; device = device->next_device) {
677                 device->initialize();
678         }
679 //      cpu->set_address_mask(0xffffffff);
680 }
681
682 VM::~VM()
683 {
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);
688                 device->release();
689                 delete device;
690                 device = next_device;
691         }
692 }
693
694 DEVICE* VM::get_device(int id)
695 {
696         for(DEVICE* device = first_device; device; device = device->next_device) {
697                 if(device->this_device_id == id) {
698                         return device;
699                 }
700         }
701         return NULL;
702 }
703
704 void VM::set_machine_type(uint16_t machine_id, uint16_t cpu_id)
705 {
706         if(memory != NULL) {
707                 memory->set_cpu_id(cpu_id);
708                 memory->set_machine_id(machine_id);
709         }
710         if(crtc != NULL) {
711                 crtc->set_cpu_id(cpu_id);
712                 crtc->set_machine_id(machine_id);
713         }
714         if(timer != NULL) {
715                 timer->set_cpu_id(cpu_id);
716                 timer->set_machine_id(machine_id);
717         }
718         if(cdrom != NULL) {
719                 cdrom->set_cpu_id(cpu_id);
720                 cdrom->set_machine_id(machine_id);
721         }
722         if(scsi != NULL) {
723                 scsi->set_cpu_id(cpu_id);
724                 scsi->set_machine_id(machine_id);
725         }
726         if(serialrom != NULL) {
727                 serialrom->set_cpu_id(cpu_id);
728                 serialrom->set_machine_id(machine_id);
729         }
730         if(floppy != NULL) {
731                 floppy->set_cpu_id(cpu_id);
732                 floppy->set_machine_id(machine_id);
733         }
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);
738         }
739 #endif
740         if(vram != NULL) {
741                 vram->set_cpu_id(cpu_id);
742                 vram->set_machine_id(machine_id);
743         }
744         
745 }               
746
747
748 // ----------------------------------------------------------------------------
749 // drive virtual machine
750 // ----------------------------------------------------------------------------
751
752 void VM::reset()
753 {
754         // reset all devices
755         boot_seq = true;
756         for(DEVICE* device = first_device; device; device = device->next_device) {
757                 device->reset();
758         }
759 //      cpu->set_address_mask(0xffffffff);
760 }
761
762 void VM::special_reset(int num)
763 {
764         // reset all devices
765         boot_seq = true;
766         
767         for(DEVICE* device = first_device; device; device = device->next_device) {
768                 device->reset();
769         }
770         keyboard->special_reset(num);
771
772 //      cpu->set_address_mask(0xffffffff);
773 }
774
775 void VM::run()
776 {
777         event->drive();
778 }
779
780 // ----------------------------------------------------------------------------
781 // debugger
782 // ----------------------------------------------------------------------------
783
784 #ifdef USE_DEBUGGER
785 DEVICE *VM::get_cpu(int index)
786 {
787         if(index == 0) {
788                 return cpu;
789         }
790         return NULL;
791 }
792 #endif
793
794 // ----------------------------------------------------------------------------
795 // draw screen
796 // ----------------------------------------------------------------------------
797
798 void VM::draw_screen()
799 {
800         crtc->draw_screen();
801 }
802
803 uint32_t VM::is_floppy_disk_accessed()
804 {
805         uint32_t val = fdc->read_signal(0);
806         if(boot_seq) {
807                 if(val != 0) {
808                         keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
809                         boot_seq = false;
810                 }
811         }
812         return val;
813 }
814
815 // ----------------------------------------------------------------------------
816 // soud manager
817 // ----------------------------------------------------------------------------
818
819 void VM::initialize_sound(int rate, int samples)
820 {
821         emu->lock_vm();
822         // init sound manager
823         event->initialize_sound(rate, samples);
824         
825         // init sound gen
826         beep->initialize_sound(rate, 8000);
827
828         // init OPN2
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); 
834
835         // init PCM
836         rf5c68->initialize_sound(rate, samples);
837         
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);
842 #if 0   
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);
848
849         line_in_ch = event->add_sound_in_source(rate, samples, 2);
850         mixer->set_context_line_in(line_in_ch, rate, samples);
851 #endif
852         emu->unlock_vm();
853 }
854
855 uint16_t* VM::create_sound(int* extra_frames)
856 {
857         return event->create_sound(extra_frames);
858 }
859
860 int VM::get_sound_buffer_ptr()
861 {
862         return event->get_sound_buffer_ptr();
863 }
864
865 void VM::clear_sound_in()
866 {
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);
870         return;
871 }
872
873 int VM::get_sound_in_data(int ch, int32_t* dst, int expect_samples, int expect_rate, int expect_channels)
874 {
875         if(dst == NULL) return 0;
876         if(expect_samples <= 0) return 0;
877         int n_ch = -1;
878         switch(ch) {
879         case 0x00:
880                 n_ch = line_in_ch;
881                 break;
882         case 0x01:
883                 n_ch = mic_in_ch;
884                 break;
885         case 0x100:
886                 n_ch = adc_in_ch;
887                 break;
888         }
889         if(n_ch < 0) return 0;
890         int samples = event->get_sound_in_data(n_ch, dst, expect_samples, expect_rate, expect_channels);
891         return samples;
892 }
893
894 // Write to event's buffer
895 int VM::sound_in(int ch, int32_t* src, int samples)
896 {
897         if(ch < 0) return 0;
898         if(ch >= 2) return 0;
899         int n_ch = -1;
900         switch(ch) {
901         case 0x100:  // ADC in from MIXER, not connected.
902                 break;
903         case 0x00: // LINE
904                 n_ch = line_in_ch;
905                 break;
906         case 0x01: // MIC
907                 n_ch = mic_in_ch;
908                 break;
909         }
910         if(n_ch < 0) return 0;
911
912         int ss = 0;
913         {
914                 emu->lock_vm();
915                 ss =  event->write_sound_in_buffer(n_ch, src, samples);
916                 emu->unlock_vm();
917
918         }
919         return ss;
920 }
921
922 #if defined(USE_HARD_DISK)
923 void VM::open_hard_disk(int drv, const _TCHAR* file_path)
924 {
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);
928                 }
929         }
930 }
931
932 void VM::close_hard_disk(int drv)
933 {
934         if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
935                 if(scsi_hdd[drv] != NULL) {
936                         scsi_hdd[drv]->close(0);
937                 }
938         }
939 }
940
941 bool VM::is_hard_disk_inserted(int drv)
942 {
943         if((drv < USE_HARD_DISK) && (drv < 8) && (drv >= 0)) {
944                 if(scsi_hdd[drv] != NULL) {
945                         return scsi_hdd[drv]->mounted(0);
946                 }
947         }
948         return false;
949 }
950
951 uint32_t VM::is_hard_disk_accessed()
952 {
953         uint32_t status = 0;
954         
955         for(int drv = 0; drv < USE_HARD_DISK; drv++) {
956                 if(scsi_hdd[drv] != NULL) {
957                         if(scsi_hdd[drv]->accessed(0)) {
958                                 status |= 1 << drv;
959                         }
960                 }
961         }
962         if(boot_seq) {
963                 if(status != 0) {
964                         keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
965                         boot_seq = false;
966                 }
967         }
968         return status;
969 }
970 #endif // USE_HARD_DISK
971
972 void VM::open_compact_disc(int drv, const _TCHAR* file_path)
973 {
974         cdrom->open(file_path);
975 }
976
977 void VM::close_compact_disc(int drv)
978 {
979         cdrom->close();
980 }
981
982 bool VM::is_compact_disc_inserted(int drv)
983 {
984         return cdrom->mounted();
985 }
986
987 uint32_t VM::is_compact_disc_accessed()
988 {
989         uint32_t status = cdrom->accessed();
990         if(boot_seq) {
991                 if(status != 0) {
992                         keyboard->write_signal(SIG_KEYBOARD_BOOTSEQ_END, 0xffffffff, 0xffffffff);
993                         boot_seq = false;
994                 }
995         }
996         return status;
997 }
998
999 #ifdef USE_SOUND_VOLUME
1000 void VM::set_sound_device_volume(int ch, int decibel_l, int decibel_r)
1001 {
1002 #ifndef HAS_LINEIN_SOUND
1003 //      if(ch >= 7) ch++;
1004 #endif
1005 #ifndef HAS_MIC_SOUND
1006 //      if(ch >= 8) ch++;
1007 #endif
1008 #ifndef HAS_MODEM_SOUND
1009 //      if(ch >= 9) ch++;
1010 #endif
1011 #ifndef HAS_2ND_ADPCM
1012 //      if(ch >= 10) ch++;
1013 #endif
1014 #if 0   
1015         if(ch == 0) { // BEEP
1016                 mixer->set_volume(beep_mix_ch, decibel_l, decibel_r);
1017         }
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);
1022         }
1023         else if(ch == 2) { // OPN2
1024                 mixer->set_volume(opn2_mix_ch, decibel_l, decibel_r);
1025         }
1026         else if(ch == 3) { // ADPCM
1027                 mixer->set_volume(pcm_mix_ch, decibel_l, decibel_r);
1028         }
1029         else if(ch == 4) { // LINE IN
1030                 //mixer->set_volume(line_mix_ch, decibel_l, decibel_r);
1031         } 
1032         else if(ch == 5) { // MIC
1033                 //mic->set_volume(0, (decibel_l + decibel_r) / 2);
1034         } 
1035         else if(ch == 6) { // MODEM
1036                 //modem->set_volume(0, (decibel_l + decibel_r) / 2);
1037         }
1038 #ifdef HAS_2ND_ADPCM
1039         else if(ch == 7) { // ADPCM
1040                 adpcm2->set_volume(0, decibel_l, decibel_r);
1041         }
1042 #endif
1043         else if(ch == 8) { // FDD
1044                 fdc->set_volume(0, decibel_l, decibel_r);
1045         }
1046         else if(ch == 9) { // HDD(ToDo)
1047                 fdc->set_volume(0, decibel_l, decibel_r);
1048         }       
1049 #else
1050         if(ch == 0) { // BEEP
1051                 beep->set_volume(0, decibel_l, decibel_r);
1052         }
1053         else if(ch == 1) { // CD-ROM
1054                 cdrom->set_volume(0, decibel_l, decibel_r);
1055         }       
1056         else if(ch == 2) { // OPN2
1057                 opn2->set_volume(0, decibel_l, decibel_r);
1058         }
1059         else if(ch == 3) { // ADPCM
1060                 rf5c68->set_volume(0, decibel_l, decibel_r);
1061         }
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);
1066         }
1067         
1068 #endif
1069 }
1070 #endif
1071
1072 // ----------------------------------------------------------------------------
1073 // notify key
1074 // ----------------------------------------------------------------------------
1075
1076 void VM::key_down(int code, bool repeat)
1077 {
1078         keyboard->key_down(code);
1079 }
1080
1081 void VM::key_up(int code)
1082 {
1083         keyboard->key_up(code);
1084 }
1085
1086 // ----------------------------------------------------------------------------
1087 // user interface
1088 // ----------------------------------------------------------------------------
1089 void VM::open_cart(int drv, const _TCHAR* file_path)
1090 {
1091         switch(drv) {
1092         case 0:
1093                 if(iccard1 != NULL) {
1094                         iccard1->open_cart(file_path);
1095                 }
1096                 break;
1097         case 1:
1098                 if(iccard2 != NULL) {
1099                         iccard2->open_cart(file_path);
1100                 }
1101                 break;
1102         }
1103 }
1104
1105 void VM::close_cart(int drv)
1106 {
1107         switch(drv) {
1108         case 0:
1109                 if(iccard1 != NULL) {
1110                         iccard1->close_cart();
1111                 }
1112                 break;
1113         case 1:
1114                 if(iccard2 != NULL) {
1115                         iccard2->close_cart();
1116                 }
1117                 break;
1118         }               
1119 }
1120
1121 bool VM::is_cart_inserted(int drv)
1122 {
1123         switch(drv) {
1124         case 0:
1125                 if(iccard1 != NULL) {
1126                         return iccard1->is_cart_inserted();
1127                 }
1128                 break;
1129         case 1:
1130                 if(iccard2 != NULL) {
1131                         return iccard2->is_cart_inserted();
1132                 }
1133                 break;
1134         }
1135         return false;
1136 }
1137
1138 void VM::open_floppy_disk(int drv, const _TCHAR* file_path, int bank)
1139 {
1140         
1141         fdc->open_disk(drv, file_path, bank);
1142         floppy->change_disk(drv);
1143 }
1144
1145 void VM::close_floppy_disk(int drv)
1146 {
1147         fdc->close_disk(drv);
1148 //      floppy->change_disk(drv);
1149 }
1150
1151 bool VM::is_floppy_disk_inserted(int drv)
1152 {
1153         return fdc->is_disk_inserted(drv);
1154 }
1155
1156 void VM::is_floppy_disk_protected(int drv, bool value)
1157 {
1158         fdc->is_disk_protected(drv, value);
1159 }
1160
1161 bool VM::is_floppy_disk_protected(int drv)
1162 {
1163         return fdc->is_disk_protected(drv);
1164 }
1165
1166 bool VM::is_frame_skippable()
1167 {
1168         return event->is_frame_skippable();
1169 }
1170
1171 void VM::update_config()
1172 {
1173         for(DEVICE* device = first_device; device; device = device->next_device) {
1174                 device->update_config();
1175         }
1176 }
1177
1178 double VM::get_current_usec()
1179 {
1180         if(event == NULL) return 0.0;
1181         return event->get_current_usec();
1182 }
1183
1184 uint64_t VM::get_current_clock_uint64()
1185 {
1186                 if(event == NULL) return (uint64_t)0;
1187                 return event->get_current_clock_uint64();
1188 }
1189
1190 #define STATE_VERSION   3
1191
1192 bool VM::process_state(FILEIO* state_fio, bool loading)
1193 {
1194         if(!state_fio->StateCheckUint32(STATE_VERSION)) {
1195                 return false;
1196         }
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)) {
1204                         return false;
1205                 }
1206                 if(!state_fio->StateCheckBuffer(name, len, 1)) {
1207                         return false;
1208                 }
1209                 if(!device->process_state(state_fio, loading)) {
1210                         if(loading) {
1211                                 printf("Data loading Error: DEVID=%d\n", device->this_device_id);
1212                         }
1213                         return false;
1214                 }
1215         }
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);
1224
1225         state_fio->StateValue(boot_seq);
1226         
1227         return true;
1228 }
1229