1 // license:BSD-3-Clause
2 // copyright-holders:Angelo Salese
3 /***************************************************************************
4 
5     Casio FP-1100
6 
7     Info found at various sites:
8 
9     Casio FP1000 and FP1100 are "pre-PC" personal computers, with Cassette,
10     Floppy Disk, Printer and 2 cart/expansion slots. They had 32K ROM, 64K
11     main RAM, 80x25 text display, 320x200, 640x200, 640x400 graphics display.
12     Floppy disk is 2x 5 1/4.
13 
14     The FP1000 had 16K videoram and monochrome only. The monitor had a switch
15     to invert the display (swap foreground and background colours).
16 
17     The FP1100 had 48K videoram and 8 colours.
18 
19     Processors: Z80 @ 4MHz, uPD7801G @ 2MHz
20 
21     Came with Basic built in, and you could run CP/M 2.2 from the floppy disk.
22 
23     The keyboard is a separate unit. It contains a beeper.
24 
25     TODO:
26     - unimplemented instruction PER triggered (can be ignored)
27     - Display can be interlaced or non-interlaced. Interlaced not emulated.
28     - Cassette Load is quite complex, using 6 pins of the sub-cpu. Not done.
29     - subcpu is supposed to be in WAIT except in horizontal blanking period.
30       WAIT is not emulated in our cpu.
31     - Keyboard not working.
32     - FDC not done.
33 
34 
35 ****************************************************************************/
36 
37 #include "emu.h"
38 #include "cpu/z80/z80.h"
39 #include "cpu/upd7810/upd7810.h"
40 #include "machine/gen_latch.h"
41 #include "machine/timer.h"
42 #include "video/mc6845.h"
43 #include "sound/beep.h"
44 #include "bus/centronics/ctronics.h"
45 #include "imagedev/cassette.h"
46 #include "emupal.h"
47 #include "screen.h"
48 #include "speaker.h"
49 
50 #define VERBOSE 0
51 #include "logmacro.h"
52 
53 #define MAIN_CLOCK 15.9744_MHz_XTAL
54 
55 class fp1100_state : public driver_device
56 {
57 public:
fp1100_state(const machine_config & mconfig,device_type type,const char * tag)58 	fp1100_state(const machine_config &mconfig, device_type type, const char *tag)
59 		: driver_device(mconfig, type, tag)
60 		, m_palette(*this, "palette")
61 		, m_maincpu(*this, "maincpu")
62 		, m_subcpu(*this, "sub")
63 		, m_crtc(*this, "crtc")
64 		, m_ipl(*this, "ipl")
65 		, m_wram(*this, "wram")
66 		, m_videoram(*this, "videoram")
67 		, m_keyboard(*this, "KEY.%u", 0)
68 		, m_beep(*this, "beeper")
69 		, m_centronics(*this, "centronics")
70 		, m_cass(*this, "cassette")
71 	{ }
72 
73 	void fp1100(machine_config &config);
74 
75 protected:
76 	virtual void machine_reset() override;
77 
78 private:
79 	void main_bank_w(u8 data);
80 	void irq_mask_w(u8 data);
81 	void slot_bank_w(u8 data);
82 	u8 slot_id_r();
83 	u8 memory_r(offs_t offset);
84 	void colour_control_w(u8 data);
85 	void kbd_row_w(u8 data);
86 	void porta_w(u8 data);
87 	u8 portb_r();
88 	u8 portc_r();
89 	void portc_w(u8 data);
90 	DECLARE_WRITE_LINE_MEMBER(centronics_busy_w);
91 	INTERRUPT_GEN_MEMBER(vblank_irq);
92 	MC6845_UPDATE_ROW(crtc_update_row);
93 	TIMER_DEVICE_CALLBACK_MEMBER(kansas_w);
94 
95 	required_device<palette_device> m_palette;
96 
97 	void io_map(address_map &map);
98 	void main_map(address_map &map);
99 	void sub_map(address_map &map);
100 	void handle_int_to_main();
101 
102 	u8 m_irq_mask;
103 	u8 m_slot_num;
104 	u8 m_kbd_row;
105 	u8 m_col_border;
106 	u8 m_col_cursor;
107 	u8 m_col_display;
108 	u8 m_centronics_busy;
109 	u8 m_cass_data[4];
110 	bool m_bank_sel;
111 	bool m_main_irq_status;
112 	bool m_sub_irq_status;
113 	bool m_cassbit;
114 	bool m_cassold;
115 
116 	struct {
117 		u8 id;
118 	}m_slot[8];
119 
120 	struct {
121 		u8 porta;
122 		u8 portb;
123 		u8 portc;
124 	}m_upd7801;
125 	required_device<cpu_device> m_maincpu;
126 	required_device<cpu_device> m_subcpu;
127 	required_device<mc6845_device> m_crtc;
128 	required_region_ptr<u8> m_ipl;
129 	required_shared_ptr<u8> m_wram;
130 	required_shared_ptr<u8> m_videoram;
131 	required_ioport_array<16> m_keyboard;
132 	required_device<beep_device> m_beep;
133 	required_device<centronics_device> m_centronics;
134 	required_device<cassette_image_device> m_cass;
135 };
136 
MC6845_UPDATE_ROW(fp1100_state::crtc_update_row)137 MC6845_UPDATE_ROW( fp1100_state::crtc_update_row )
138 {
139 	const rgb_t *palette = m_palette->palette()->entry_list_raw();
140 	u32 *p = &bitmap.pix(y);
141 
142 	if (BIT(m_upd7801.porta, 4))
143 	{ // green screen
144 		for (u16 x = 0; x < x_count; x++)
145 		{
146 			u16 mem = (((ma+x)<<3) + ra) & 0x3fff;
147 			u8 g = m_videoram[mem];
148 			for (u8 i = 0; i < 8; i++)
149 			{
150 				u8 col = BIT(g, i);
151 				if (x == cursor_x) col ^= 1;
152 				*p++ = palette[col<<1];
153 			}
154 		}
155 	}
156 	else
157 	{ // RGB screen
158 		for (u16 x = 0; x < x_count; x++)
159 		{
160 			u16 mem = (((ma+x)<<3) + ra) & 0x3fff;
161 			u8 b = m_videoram[mem];
162 			u8 r = m_videoram[mem+0x4000];
163 			u8 g = m_videoram[mem+0x8000];
164 			for (u8 i = 0; i < 8; i++)
165 			{
166 				u8 col = BIT(r, i) + (BIT(g, i) << 1) + (BIT(b, i) << 2);
167 				if (x == cursor_x) col = m_col_cursor;
168 				*p++ = palette[col];
169 			}
170 		}
171 	}
172 }
173 
174 /*
175 d0 - Package select
176 d1 - Bank select (at boot time)
177 other bits not used
178 */
main_bank_w(u8 data)179 void fp1100_state::main_bank_w(u8 data)
180 {
181 	m_bank_sel = BIT(data, 1);
182 	m_slot_num = (m_slot_num & 3) | ((data & 1) << 2); //??
183 }
184 
185 // tell sub that latch has a byte
irq_mask_w(u8 data)186 void fp1100_state::irq_mask_w(u8 data)
187 {
188 	m_irq_mask = data;
189 	handle_int_to_main();
190 
191 	if (BIT(data, 7) && !m_sub_irq_status)
192 	{
193 		m_subcpu->set_input_line(UPD7810_INTF2, ASSERT_LINE);
194 		LOG("%s: Sub IRQ asserted\n",machine().describe_context());
195 		m_sub_irq_status = true;
196 	}
197 	else
198 	if (!BIT(data, 7) && m_sub_irq_status)
199 	{
200 		m_subcpu->set_input_line(UPD7810_INTF2, CLEAR_LINE);
201 		LOG("%s: Sub IRQ cleared\n",machine().describe_context());
202 		m_sub_irq_status = false;
203 	}
204 
205 	LOG("%s: IRQmask=%X\n",machine().describe_context(),data);
206 }
207 
slot_bank_w(u8 data)208 void fp1100_state::slot_bank_w(u8 data)
209 {
210 	m_slot_num = (data & 3) | (m_slot_num & 4);
211 }
212 
slot_id_r()213 u8 fp1100_state::slot_id_r()
214 {
215 	//return 0xff;
216 	return m_slot[m_slot_num & 7].id;
217 }
218 
memory_r(offs_t offset)219 u8 fp1100_state::memory_r(offs_t offset)
220 {
221 	if (offset < 0x9000 && !m_bank_sel)
222 		return m_ipl[offset];
223 	else
224 		return m_wram[offset];
225 }
226 
main_map(address_map & map)227 void fp1100_state::main_map(address_map &map)
228 {
229 	map.unmap_value_high();
230 	map(0x0000, 0xffff).r(FUNC(fp1100_state::memory_r));
231 	map(0x0000, 0xffff).writeonly().share("wram"); // always write to ram
232 }
233 
io_map(address_map & map)234 void fp1100_state::io_map(address_map &map)
235 {
236 	map.unmap_value_high();
237 	//map(0x0000, 0xfeff) slot memory area
238 	map(0xff00, 0xff00).mirror(0x7f).rw(FUNC(fp1100_state::slot_id_r), FUNC(fp1100_state::slot_bank_w));
239 	map(0xff80, 0xff80).mirror(0x7f).r("sub2main", FUNC(generic_latch_8_device::read));
240 	map(0xff80, 0xff80).mirror(0x1f).w(FUNC(fp1100_state::irq_mask_w));
241 	map(0xffa0, 0xffa0).mirror(0x1f).w(FUNC(fp1100_state::main_bank_w));
242 	map(0xffc0, 0xffc0).mirror(0x3f).w("main2sub", FUNC(generic_latch_8_device::write));
243 }
244 
245 /*
246 d0,1,2 - border colour (B,R,G)
247 d3     - not used
248 d4,5,6 - colour of cursor; or display area (B,R,G) (see d7)
249 d7     - 1=display area; 0=cursor
250 */
colour_control_w(u8 data)251 void fp1100_state::colour_control_w(u8 data)
252 {
253 	data = bitswap<8>(data, 7, 4, 6, 5, 3, 0, 2, 1);  // change BRG to RGB
254 
255 	m_col_border = data & 7;
256 
257 	if (BIT(data, 7))
258 		m_col_display = (data >> 4) & 7;
259 	else
260 		m_col_cursor = data >> 4;
261 }
262 
263 /*
264 d0,1,2,3 - keyboard scan row
265          - if 13, turn on shift-lock LED
266          - if 14, turn on caps-lock LED
267          - if 15, turn off both LEDs
268 d4       - Beeper
269 d5       - "3state buffer of key data line (1=open, 0=closed)"
270 d6,7     - not used
271 */
kbd_row_w(u8 data)272 void fp1100_state::kbd_row_w(u8 data)
273 {
274 	m_kbd_row = data;
275 	m_beep->set_state(BIT(data, 4));
276 }
277 
sub_map(address_map & map)278 void fp1100_state::sub_map(address_map &map)
279 {
280 	map(0x0000, 0x1fff).rom().region("sub_ipl", 0x0000);
281 	map(0x2000, 0xdfff).ram().share("videoram"); //vram B/R/G
282 	map(0xe000, 0xe000).mirror(0x3fe).rw(m_crtc, FUNC(mc6845_device::status_r), FUNC(mc6845_device::address_w));
283 	map(0xe001, 0xe001).mirror(0x3fe).rw(m_crtc, FUNC(mc6845_device::register_r), FUNC(mc6845_device::register_w));
284 	map(0xe400, 0xe400).mirror(0x3ff).portr("DSW").w(FUNC(fp1100_state::kbd_row_w));
285 	map(0xe800, 0xe800).mirror(0x3ff).r("main2sub", FUNC(generic_latch_8_device::read));
286 	map(0xe800, 0xe800).mirror(0x3ff).w("sub2main", FUNC(generic_latch_8_device::write));
287 	map(0xec00, 0xec00).mirror(0x3ff).lw8(NAME([this] (u8 data) { m_subcpu->set_input_line(UPD7810_INTF0, CLEAR_LINE); }));
288 	map(0xf000, 0xf000).mirror(0x3ff).w(FUNC(fp1100_state::colour_control_w));
289 	map(0xf400, 0xff7f).rom().region("sub_ipl", 0x2400);
290 }
291 
292 /*
293 d0,1,2 - enable RGB guns (G,R,B)
294 d3     - CRTC clock (80 or 40 cols)
295 d4     - RGB (0) or Green (1)
296 d5     - clear videoram
297 d6     - CMT baud rate (1=300; 0=1200)
298 d7     - CMT load clock
299 The SO pin is Serial Output to CMT (1=2400Hz; 0=1200Hz)
300 */
porta_w(u8 data)301 void fp1100_state::porta_w(u8 data)
302 {
303 	m_upd7801.porta = data;
304 
305 	if (BIT(data, 5))
306 		memset(m_videoram, 0, 0xc000);
307 }
308 
portb_r()309 u8 fp1100_state::portb_r()
310 {
311 	u8 data = m_keyboard[m_kbd_row & 15]->read() ^ 0xff;
312 	LOG("%s: PortB:%X:%X\n",machine().describe_context(),m_kbd_row,data);
313 	//m_subcpu->set_input_line(UPD7810_INTF0, BIT(data, 7) ? ASSERT_LINE : CLEAR_LINE);
314 	if (BIT(m_kbd_row, 5))
315 		return data;
316 	else
317 		return 0;
318 }
319 
320 /*
321 d0 - Centronics busy
322 d1 - Centronics error
323 d2 - CMT load input clock
324 d7 - CMT load serial data
325 */
portc_r()326 u8 fp1100_state::portc_r()
327 {
328 	return (m_upd7801.portc & 0x78) | m_centronics_busy;
329 }
330 
331 /*
332 d3 - cause INT on main cpu
333 d4 - Centronics port is used for input or output
334 d5 - CMT relay
335 d6 - Centronics strobe
336 */
portc_w(u8 data)337 void fp1100_state::portc_w(u8 data)
338 {
339 	u8 bits = data ^ m_upd7801.portc;
340 	m_upd7801.portc = data;
341 
342 	if (BIT(bits, 3))
343 	{
344 		LOG("%s: PortC:%X\n",machine().describe_context(),data);
345 		handle_int_to_main();
346 	}
347 	if (BIT(bits, 5))
348 		m_cass->change_state(BIT(data, 5) ? CASSETTE_MOTOR_ENABLED : CASSETTE_MOTOR_DISABLED, CASSETTE_MASK_MOTOR);
349 	if (BIT(bits, 6))
350 		m_centronics->write_strobe(BIT(data, 6));
351 }
352 
353 // HOLD_LINE used because the interrupt is set and cleared by successive instructions, too fast for the maincpu to notice
handle_int_to_main()354 void fp1100_state::handle_int_to_main()
355 {
356 	// IRQ is on if bit 4 of mask AND bit 3 portC
357 	if (BIT(m_upd7801.portc, 3) && BIT(m_irq_mask, 4))
358 	{
359 		if (!m_main_irq_status)
360 		{
361 			m_maincpu->set_input_line(0, HOLD_LINE);
362 			LOG("%s: Main IRQ asserted\n",machine().describe_context());
363 //          m_main_irq_status = true;
364 		}
365 	}
366 	else
367 	{
368 		if (m_main_irq_status)
369 		{
370 //          m_maincpu->set_input_line(0, CLEAR_LINE);
371 //          LOG("%s: Main IRQ cleared\n",machine().describe_context());
372 			m_main_irq_status = false;
373 		}
374 	}
375 }
376 
377 /* Input ports */
378 static INPUT_PORTS_START( fp1100 )
379 	PORT_START("KEY.0")
380 	PORT_BIT(0xff, IP_ACTIVE_LOW, IPT_UNUSED)
381 
382 	PORT_START("KEY.1")
383 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("Break")
384 	PORT_BIT(0x60, IP_ACTIVE_LOW, IPT_UNUSED)
385 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_PGUP) PORT_NAME("Kanji")  // guess, it's in Japanese
386 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_CAPSLOCK) PORT_NAME("Caps")
387 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_LALT) PORT_NAME("Graph")
388 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_LCONTROL) PORT_CODE(KEYCODE_RCONTROL) PORT_NAME("Ctrl")
389 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_LSHIFT) PORT_CODE(KEYCODE_RSHIFT) PORT_NAME("Shift") PORT_CHAR(UCHAR_SHIFT_1)
390 
391 	PORT_START("KEY.2")
392 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F10) PORT_NAME("PF0")
393 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_ENTER_PAD) PORT_NAME("Enter") PORT_CHAR(13)
394 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_ASTERISK) PORT_NAME("*")
395 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_Z) PORT_NAME("Z") PORT_CHAR('Z') PORT_CHAR('z')
396 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_Q) PORT_NAME("Q") PORT_CHAR('Q') PORT_CHAR('q')
397 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP-")    PORT_CODE(KEYCODE_MINUS_PAD)
398 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_ESC) PORT_NAME("Esc") PORT_CHAR(27)
399 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_A) PORT_NAME("A") PORT_CHAR('A') PORT_CHAR('a')
400 
401 	PORT_START("KEY.3")
402 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F1) PORT_NAME("PF1")
403 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_INSERT) PORT_NAME(",")
404 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_SLASH_PAD) PORT_NAME("/")
405 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_X) PORT_NAME("X") PORT_CHAR('X') PORT_CHAR('x')
406 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_W) PORT_NAME("W") PORT_CHAR('W') PORT_CHAR('w')
407 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP+")    PORT_CODE(KEYCODE_PLUS_PAD)
408 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_1) PORT_NAME("1") PORT_CHAR('1') PORT_CHAR('!')
409 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_S) PORT_NAME("S") PORT_CHAR('S') PORT_CHAR('s')
410 
411 	PORT_START("KEY.4")
412 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F2) PORT_NAME("PF2")
413 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_DEL) PORT_NAME(".")
414 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_DEL_PAD) PORT_NAME("Del")
415 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_C) PORT_NAME("C") PORT_CHAR('C') PORT_CHAR('c')
416 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_E) PORT_NAME("E") PORT_CHAR('E') PORT_CHAR('e')
417 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP3")    PORT_CODE(KEYCODE_3_PAD)
418 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_2) PORT_NAME("2") PORT_CHAR('2') PORT_CHAR('@')
419 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_D) PORT_NAME("D") PORT_CHAR('D') PORT_CHAR('d')
420 
421 	PORT_START("KEY.5")
422 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F3) PORT_NAME("PF3")
423 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("000")
424 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_RIGHT) PORT_NAME("Right")
425 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_V) PORT_NAME("V") PORT_CHAR('V') PORT_CHAR('v')
426 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_R) PORT_NAME("R") PORT_CHAR('R') PORT_CHAR('r')
427 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP6")    PORT_CODE(KEYCODE_6_PAD)
428 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_3) PORT_NAME("3") PORT_CHAR('3') PORT_CHAR('#')
429 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F) PORT_NAME("F") PORT_CHAR('F') PORT_CHAR('f')
430 
431 	PORT_START("KEY.6")
432 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F4) PORT_NAME("PF4")
433 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_SPACE) PORT_NAME("Space") PORT_CHAR(32)
434 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_PGDN) PORT_NAME("Ins")
435 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_B) PORT_NAME("B") PORT_CHAR('B') PORT_CHAR('b')
436 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_T) PORT_NAME("T") PORT_CHAR('T') PORT_CHAR('t')
437 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP9")    PORT_CODE(KEYCODE_9_PAD)
438 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_4) PORT_NAME("4") PORT_CHAR('4') PORT_CHAR('$')
439 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_G) PORT_NAME("G") PORT_CHAR('G') PORT_CHAR('g')
440 
441 	PORT_START("KEY.7")
442 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F5) PORT_NAME("PF5")
443 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP0")    PORT_CODE(KEYCODE_0_PAD)
444 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_DOWN) PORT_NAME("Down")
445 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_N) PORT_NAME("N") PORT_CHAR('N') PORT_CHAR('n')
446 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_Y) PORT_NAME("Y") PORT_CHAR('Y') PORT_CHAR('y')
447 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP8")    PORT_CODE(KEYCODE_8_PAD)
448 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_5) PORT_NAME("5") PORT_CHAR('5') PORT_CHAR('^')
449 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_H) PORT_NAME("H") PORT_CHAR('H') PORT_CHAR('h')
450 
451 	PORT_START("KEY.8")
452 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F6) PORT_NAME("PF6")
453 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP2")    PORT_CODE(KEYCODE_2_PAD)
454 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_UP) PORT_NAME("Up")
455 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_M) PORT_NAME("M") PORT_CHAR('M') PORT_CHAR('m')
456 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_U) PORT_NAME("U") PORT_CHAR('U') PORT_CHAR('u')
457 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP5")    PORT_CODE(KEYCODE_5_PAD)
458 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_6) PORT_NAME("6") PORT_CHAR('6') PORT_CHAR('&')
459 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_J) PORT_NAME("J") PORT_CHAR('J') PORT_CHAR('j')
460 
461 	PORT_START("KEY.9")
462 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F7) PORT_NAME("PF7")
463 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP1")    PORT_CODE(KEYCODE_1_PAD)
464 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_HOME) PORT_NAME("Home")
465 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_COMMA) PORT_NAME(",") PORT_CHAR(',') PORT_CHAR('<')
466 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_I) PORT_NAME("I") PORT_CHAR('I') PORT_CHAR('i')
467 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP4")    PORT_CODE(KEYCODE_4_PAD)
468 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_7) PORT_NAME("7") PORT_CHAR('7') PORT_CHAR('%')
469 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_K) PORT_NAME("K") PORT_CHAR('K') PORT_CHAR('k')
470 
471 	PORT_START("KEY.10")
472 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F8) PORT_NAME("PF8")
473 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_CLOSEBRACE) PORT_NAME("]") PORT_CHAR(']') PORT_CHAR('}')
474 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_LEFT) PORT_NAME("Left")
475 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_STOP) PORT_NAME(".") PORT_CHAR('.') PORT_CHAR('>')
476 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_O) PORT_NAME("O") PORT_CHAR('O') PORT_CHAR('o')
477 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("KP7")    PORT_CODE(KEYCODE_7_PAD)
478 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_8) PORT_NAME("8") PORT_CHAR('8') PORT_CHAR('*')
479 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_L) PORT_NAME("L") PORT_CHAR('L') PORT_CHAR('l')
480 
481 	PORT_START("KEY.11")
482 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_F9) PORT_NAME("PF9")
483 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_OPENBRACE) PORT_NAME("[") PORT_CHAR('[') PORT_CHAR('{')
484 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_BACKSPACE) PORT_NAME("Rubout") PORT_CHAR(8)
485 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_SLASH) PORT_NAME("/") PORT_CHAR('/') PORT_CHAR('?')
486 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_P) PORT_NAME("P") PORT_CHAR('P') PORT_CHAR('p')
487 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_ENTER) PORT_NAME("Return") PORT_CHAR(13)
488 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_9) PORT_NAME("9") PORT_CHAR('9') PORT_CHAR('(')
489 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_COLON) PORT_NAME(";") PORT_CHAR(';') PORT_CHAR('+')
490 
491 	PORT_START("KEY.12")
492 	PORT_BIT(0x80, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("Stop/Cont")
493 	PORT_BIT(0x40, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_MINUS) PORT_NAME("-") PORT_CHAR('-') PORT_CHAR('_')
494 	PORT_BIT(0x20, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_BACKSLASH) PORT_NAME("Yen") PORT_CHAR('\\') PORT_CHAR('|')
495 	PORT_BIT(0x10, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("(c)")
496 	PORT_BIT(0x08, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_NAME("|")
497 	PORT_BIT(0x04, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_EQUALS) PORT_NAME("^") PORT_CHAR('^') PORT_CHAR('+')
498 	PORT_BIT(0x02, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_0) PORT_NAME("0") PORT_CHAR('0') PORT_CHAR(')')
499 	PORT_BIT(0x01, IP_ACTIVE_LOW, IPT_KEYBOARD) PORT_CODE(KEYCODE_QUOTE) PORT_NAME(":") PORT_CHAR(':') PORT_CHAR('*')
500 
501 	PORT_START("KEY.13")
502 	PORT_BIT(0xff, IP_ACTIVE_LOW, IPT_UNUSED) // Capslock LED on
503 
504 	PORT_START("KEY.14")
505 	PORT_BIT(0xff, IP_ACTIVE_LOW, IPT_UNUSED) // Kanji LED on
506 
507 	PORT_START("KEY.15")
508 	PORT_BIT(0xff, IP_ACTIVE_LOW, IPT_UNUSED) // LEDs off
509 
510 	PORT_START("DSW")
511 	PORT_DIPNAME( 0x01, 0x01, "Text width" ) PORT_DIPLOCATION("SW1:1")
512 	PORT_DIPSETTING(    0x00, "40 chars/line" )
513 	PORT_DIPSETTING(    0x01, "80 chars/line" )
514 	PORT_DIPNAME( 0x02, 0x02, "Screen Mode" ) PORT_DIPLOCATION("SW1:2")
515 	PORT_DIPSETTING(    0x00, "Screen 0" )
516 	PORT_DIPSETTING(    0x02, "Screen 1" )
517 	PORT_DIPNAME( 0x04, 0x04, "FP Mode" ) PORT_DIPLOCATION("SW1:3")
518 	PORT_DIPSETTING(    0x00, "FP-1000" ) // Green screen
519 	PORT_DIPSETTING(    0x04, "FP-1100" ) // RGB
520 	PORT_DIPNAME( 0x08, 0x08, "CMT Baud Rate" ) PORT_DIPLOCATION("SW1:4")
521 	PORT_DIPSETTING(    0x00, "1200 Baud" )
522 	PORT_DIPSETTING(    0x08, "300 Baud" )
523 	PORT_DIPNAME( 0x10, 0x10, "Printer Type" ) PORT_DIPLOCATION("SW1:5")
524 	PORT_DIPSETTING(    0x00, "<undefined>" )
525 	PORT_DIPSETTING(    0x10, "FP-1012PR" )
526 	PORT_DIPNAME( 0x20, 0x20, "Keyboard Type" ) PORT_DIPLOCATION("SW1:6")
527 	PORT_DIPSETTING(    0x00, "<undefined>" )
528 	PORT_DIPSETTING(    0x20, "Normal" )
529 	PORT_BIT(0xc0, IP_ACTIVE_LOW, IPT_UNUSED )
530 
531 	PORT_START("SLOTS")
532 	PORT_CONFNAME( 0x0003, 0x0002, "Slot #0" )
533 	PORT_CONFSETTING(    0x0000, "(null)" )
534 	PORT_CONFSETTING(    0x0001, "ROM" )
535 	PORT_CONFSETTING(    0x0002, "RAM" )
536 	PORT_CONFSETTING(    0x0003, "FDC" )
537 	PORT_CONFNAME( 0x000c, 0x0008, "Slot #1" )
538 	PORT_CONFSETTING(    0x0000, "(null)" )
539 	PORT_CONFSETTING(    0x0004, "ROM" )
540 	PORT_CONFSETTING(    0x0008, "RAM" )
541 	PORT_CONFSETTING(    0x000c, "FDC" )
542 	PORT_CONFNAME( 0x0030, 0x0020, "Slot #2" )
543 	PORT_CONFSETTING(    0x0000, "(null)" )
544 	PORT_CONFSETTING(    0x0010, "ROM" )
545 	PORT_CONFSETTING(    0x0020, "RAM" )
546 	PORT_CONFSETTING(    0x0030, "FDC" )
547 	PORT_CONFNAME( 0x00c0, 0x0080, "Slot #3" )
548 	PORT_CONFSETTING(    0x0000, "(null)" )
549 	PORT_CONFSETTING(    0x0040, "ROM" )
550 	PORT_CONFSETTING(    0x0080, "RAM" )
551 	PORT_CONFSETTING(    0x00c0, "FDC" )
552 	PORT_CONFNAME( 0x0300, 0x0200, "Slot #4" )
553 	PORT_CONFSETTING(    0x0000, "(null)" )
554 	PORT_CONFSETTING(    0x0100, "ROM" )
555 	PORT_CONFSETTING(    0x0200, "RAM" )
556 	PORT_CONFSETTING(    0x0300, "FDC" )
557 	PORT_CONFNAME( 0x0c00, 0x0800, "Slot #5" )
558 	PORT_CONFSETTING(    0x0000, "(null)" )
559 	PORT_CONFSETTING(    0x0400, "ROM" )
560 	PORT_CONFSETTING(    0x0800, "RAM" )
561 	PORT_CONFSETTING(    0x0c00, "FDC" )
562 	PORT_CONFNAME( 0x3000, 0x2000, "Slot #6" )
563 	PORT_CONFSETTING(    0x0000, "(null)" )
564 	PORT_CONFSETTING(    0x1000, "ROM" )
565 	PORT_CONFSETTING(    0x2000, "RAM" )
566 	PORT_CONFSETTING(    0x3000, "FDC" )
567 	PORT_CONFNAME( 0xc000, 0x8000, "Slot #7" )
568 	PORT_CONFSETTING(    0x0000, "(null)" )
569 	PORT_CONFSETTING(    0x4000, "ROM" )
570 	PORT_CONFSETTING(    0x8000, "RAM" )
571 	PORT_CONFSETTING(    0xc000, "FDC" )
572 INPUT_PORTS_END
573 
574 
575 static const gfx_layout chars_8x8 =
576 {
577 	8,8,
578 	256,
579 	1,
580 	{ 0 },
581 	{ 7, 6, 5, 4, 3, 2, 1, 0 },
582 	{ 0*8, 1*8, 2*8, 3*8, 4*8, 5*8, 6*8, 7*8 },
583 	8*8
584 };
585 
586 static GFXDECODE_START( gfx_fp1100 )
587 	GFXDECODE_ENTRY( "sub_ipl", 0x2400, chars_8x8, 0, 1 )
588 GFXDECODE_END
589 
WRITE_LINE_MEMBER(fp1100_state::centronics_busy_w)590 WRITE_LINE_MEMBER( fp1100_state::centronics_busy_w )
591 {
592 	m_centronics_busy = state;
593 }
594 
TIMER_DEVICE_CALLBACK_MEMBER(fp1100_state::kansas_w)595 TIMER_DEVICE_CALLBACK_MEMBER( fp1100_state::kansas_w )
596 {
597 	m_cass_data[3]++;
598 
599 	if (m_cassbit != m_cassold)
600 	{
601 		m_cass_data[3] = 0;
602 		m_cassold = m_cassbit;
603 	}
604 
605 	if (m_cassbit)
606 		m_cass->output(BIT(m_cass_data[3], 0) ? -1.0 : +1.0); // 2400Hz
607 	else
608 		m_cass->output(BIT(m_cass_data[3], 1) ? -1.0 : +1.0); // 1200Hz
609 }
610 
INTERRUPT_GEN_MEMBER(fp1100_state::vblank_irq)611 INTERRUPT_GEN_MEMBER( fp1100_state::vblank_irq )
612 {
613 //  if (BIT(m_irq_mask, 4))
614 //      m_maincpu->set_input_line_and_vector(0, HOLD_LINE, 0xf8); // Z80
615 }
616 
machine_reset()617 void fp1100_state::machine_reset()
618 {
619 	m_main_irq_status = false;
620 	m_sub_irq_status = false;
621 	int i;
622 	u8 slot_type;
623 	const u8 id_type[4] = { 0xff, 0x00, 0x01, 0x04};
624 	for(i=0;i<8;i++)
625 	{
626 		slot_type = (ioport("SLOTS")->read() >> i*2) & 3;
627 		m_slot[i].id = id_type[slot_type];
628 	}
629 
630 	m_beep->set_state(0);
631 
632 	m_bank_sel = false; // point at rom
633 
634 	m_irq_mask = 0;
635 	m_slot_num = 0;
636 	m_kbd_row = 0;
637 	m_col_border = 0;
638 	m_col_cursor = 0;
639 	m_col_display = 0;
640 	m_upd7801.porta = 0;
641 	m_upd7801.portb = 0;
642 	m_upd7801.portc = 0;
643 	m_maincpu->set_input_line_vector(0, 0xF0);
644 }
645 
fp1100(machine_config & config)646 void fp1100_state::fp1100(machine_config &config)
647 {
648 	/* basic machine hardware */
649 	Z80(config, m_maincpu, MAIN_CLOCK/4);
650 	m_maincpu->set_addrmap(AS_PROGRAM, &fp1100_state::main_map);
651 	m_maincpu->set_addrmap(AS_IO, &fp1100_state::io_map);
652 	m_maincpu->set_vblank_int("screen", FUNC(fp1100_state::vblank_irq));
653 
654 	upd7801_device &sub(UPD7801(config, m_subcpu, MAIN_CLOCK/4));
655 	sub.set_addrmap(AS_PROGRAM, &fp1100_state::sub_map);
656 	sub.pa_out_cb().set(FUNC(fp1100_state::porta_w));
657 	sub.pb_in_cb().set(FUNC(fp1100_state::portb_r));
658 	sub.pb_out_cb().set("cent_data_out", FUNC(output_latch_device::write));
659 	sub.pc_in_cb().set(FUNC(fp1100_state::portc_r));
660 	sub.pc_out_cb().set(FUNC(fp1100_state::portc_w));
661 	sub.txd_func().set([this] (bool state) { m_cassbit = state; });
662 
663 	GENERIC_LATCH_8(config, "main2sub");
664 	GENERIC_LATCH_8(config, "sub2main");
665 
666 	/* video hardware */
667 	screen_device &screen(SCREEN(config, "screen", SCREEN_TYPE_RASTER));
668 	screen.set_refresh_hz(60);
669 	screen.set_vblank_time(ATTOSECONDS_IN_USEC(2500)); /* not accurate */
670 	screen.set_size(640, 480);
671 	screen.set_visarea_full();
672 	screen.set_screen_update("crtc", FUNC(mc6845_device::screen_update));
673 	PALETTE(config, m_palette).set_entries(8);
674 	GFXDECODE(config, "gfxdecode", m_palette, gfx_fp1100);
675 
676 	/* sound hardware */
677 	SPEAKER(config, "mono").front_center();
678 	BEEP(config, "beeper", 950) // guess
679 			.add_route(ALL_OUTPUTS, "mono", 0.50); // inside the keyboard
680 
681 	/* CRTC */
682 	MC6845(config, m_crtc, MAIN_CLOCK/8);   /* unknown variant; hand tuned to get ~60 fps */
683 	m_crtc->set_screen("screen");
684 	m_crtc->set_show_border_area(false);
685 	m_crtc->set_char_width(8);
686 	m_crtc->set_update_row_callback(FUNC(fp1100_state::crtc_update_row));
687 
688 	/* Printer */
689 	CENTRONICS(config, m_centronics, centronics_devices, "printer");
690 	m_centronics->busy_handler().set(FUNC(fp1100_state::centronics_busy_w));
691 
692 	output_latch_device &latch(OUTPUT_LATCH(config, "cent_data_out"));
693 	m_centronics->set_output_latch(latch);
694 
695 	/* Cassette */
696 	CASSETTE(config, m_cass);
697 	m_cass->set_default_state(CASSETTE_PLAY | CASSETTE_MOTOR_DISABLED | CASSETTE_SPEAKER_ENABLED);
698 	m_cass->add_route(ALL_OUTPUTS, "mono", 0.05);
699 	TIMER(config, "kansas_w").configure_periodic(FUNC(fp1100_state::kansas_w), attotime::from_hz(4800)); // cass write
700 }
701 
702 /* ROM definition */
703 ROM_START( fp1100 )
704 	ROM_REGION( 0x9000, "ipl", ROMREGION_ERASEFF )
705 	ROM_LOAD( "basic.rom", 0x0000, 0x9000, BAD_DUMP CRC(7c7dd17c) SHA1(985757b9c62abd17b0bd77db751d7782f2710ec3))
706 
707 	ROM_REGION( 0x3000, "sub_ipl", ROMREGION_ERASEFF )
708 	ROM_LOAD( "sub1.rom", 0x0000, 0x1000, CRC(8feda489) SHA1(917d5b398b9e7b9a6bfa5e2f88c5b99923c3c2a3))
709 	ROM_LOAD( "sub2.rom", 0x1000, 0x1000, CRC(359f007e) SHA1(0188d5a7b859075cb156ee55318611bd004128d7))
710 	ROM_LOAD( "sub3.rom", 0x2000, 0xf80, BAD_DUMP CRC(fb2b577a) SHA1(a9ae6b03e06ea2f5db30dfd51ebf5aede01d9672))
711 ROM_END
712 
713 /* Driver */
714 
715 /*    YEAR  NAME    PARENT  COMPAT  MACHINE  INPUT   CLASS         INIT        COMPANY  FULLNAME   FLAGS */
716 COMP( 1983, fp1100, 0,      0,      fp1100,  fp1100, fp1100_state, empty_init, "Casio", "FP-1100", MACHINE_NOT_WORKING)
717