1 // license:BSD-3-Clause
2 // copyright-holders:Miodrag Milanovic
3 /***************************************************************************
4 
5 Pecom driver by Miodrag Milanovic
6 
7 2008-11-08 Preliminary driver.
8 
9 - All commands to be in UPPERCASE.
10 - Change background colour: SCR n
11 - Enter monitor: PROB     (B to exit)
12 - If Capslock is engaged, then Shift doesn't work.
13 - Control hangs the machine while it is pressed. It doesn't work in the
14   expected way.
15 - Don't touch the Shift key while loading a tape because it will corrupt
16   the data.
17 - The screen will flash in a crazy epileptic fashion while loading a tape.
18   Beware!
19 
20 TODO:
21 - Cassette: can load its own recordings, but not those from software list
22   (software-list tapes are slower & wobbly)
23 - Both machines currently have 32k ram.
24 - Autorepeat seems a bit fast
25 
26 ****************************************************************************/
27 
28 #include "emu.h"
29 #include "includes/pecom.h"
30 #include "speaker.h"
31 
32 
33 /* Address maps */
mem_map(address_map & map)34 void pecom_state::mem_map(address_map &map)
35 {
36 	map(0x0000, 0x7fff).ram().share("mainram");
37 	map(0x0000, 0x3fff).bankr("bank1");
38 	map(0x8000, 0xefff).rom().region("maincpu",0);
39 	map(0xf000, 0xf7ff).bankrw("bank3"); // CDP1869 / ROM
40 	map(0xf800, 0xffff).bankrw("bank4"); // CDP1869 / ROM
41 }
42 
io_map(address_map & map)43 void pecom_state::io_map(address_map &map)
44 {
45 	map(0x01, 0x01).w(FUNC(pecom_state::bank_w));
46 	map(0x03, 0x03).r(FUNC(pecom_state::keyboard_r));
47 	map(0x03, 0x07).w(FUNC(pecom_state::cdp1869_w));
48 }
49 
cdp1869_page_ram(address_map & map)50 void pecom_state::cdp1869_page_ram(address_map &map)
51 {
52 	map(0x000, 0x3ff).mirror(0x400).ram();
53 }
54 
55 /* Input ports */
56 /* Pecom 64 keyboard layout is as follows
57 
58     1!     2"     3#     4$     5%     6&     7'     8(     9)     0     BREAK
59 
60    DEL  Q      W      E      R      T      Y      U      I      O      P   ESC
61 
62     CAPS    A      S      D      F      G      H      J      K      L   RETURN
63 
64   CTRL  ,<     Z      X      C      V      B      N      M      :*     /?   LF
65 
66      SHIFT  .>    Down   Left       SPACEBAR       Right    Up      ;+    =-
67 
68 Being keys distributed on four lines, it makes a bit difficult to accurately remap them
69 on modern keyboards. Hence, we move by default Up/Down/Left/Right to Cursor Keys and
70 use Left/Right Ctrl/Alt keys for the remaining keys. Due to the unnatural emulated keyboard
71 mappings, this is another situation where natural keyboard comes very handy!          */
72 
INPUT_CHANGED_MEMBER(pecom_state::ef_w)73 INPUT_CHANGED_MEMBER(pecom_state::ef_w)
74 {
75 	m_maincpu->set_input_line((int)param, newval);
76 }
77 
78 static INPUT_PORTS_START( pecom )
79 	PORT_START("LINE0")
PORT_CODE(KEYCODE_ENTER)80 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Return") PORT_CODE(KEYCODE_ENTER) PORT_CODE(KEYCODE_COLON) PORT_CHAR(13)
81 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Line Feed") PORT_CODE(KEYCODE_SLASH)
82 
83 	PORT_START("LINE1")
84 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Esc") PORT_CODE(KEYCODE_OPENBRACE) PORT_CHAR(UCHAR_MAMEKEY(ESC))
85 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_UNUSED ) // Actually this is again / ? - same key connected as on SLASH
86 
87 	PORT_START("LINE2")
88 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_0) PORT_CHAR('0')
89 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_1) PORT_CHAR('1') PORT_CHAR('!')
90 
91 	PORT_START("LINE3")
92 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_2) PORT_CHAR('2') PORT_CHAR('"')
93 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_3) PORT_CHAR('3') PORT_CHAR('#')
94 
95 	PORT_START("LINE4")
96 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_4) PORT_CHAR('4') PORT_CHAR('$')
97 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_5) PORT_CHAR('5') PORT_CHAR('%')
98 
99 	PORT_START("LINE5")
100 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_6) PORT_CHAR('6') PORT_CHAR('&')
101 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_7) PORT_CHAR('7') PORT_CHAR('\'')
102 
103 	PORT_START("LINE6")
104 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_8) PORT_CHAR('8') PORT_CHAR('(')
105 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_9) PORT_CHAR('9') PORT_CHAR(')')
106 
107 	PORT_START("LINE7")
108 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_COMMA) PORT_CHAR(':') PORT_CHAR('*')
109 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_RALT) PORT_CHAR(';') PORT_CHAR('+')
110 
111 	PORT_START("LINE8")
112 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_TILDE) PORT_CHAR(',') PORT_CHAR('<')
113 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_RCONTROL) PORT_CHAR('=') PORT_CHAR('-')
114 
115 	PORT_START("LINE9")
116 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_LALT) PORT_CHAR('.') PORT_CHAR('>')
117 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_STOP) PORT_CHAR('/') PORT_CHAR('?')
118 
119 	PORT_START("LINE10")
120 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_SPACE) PORT_CHAR(' ')
121 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_A) PORT_CHAR('A') PORT_CHAR('a')
122 
123 	PORT_START("LINE11")
124 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_B) PORT_CHAR('B') PORT_CHAR('b')
125 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_C) PORT_CHAR('C') PORT_CHAR('c')
126 
127 	PORT_START("LINE12")
128 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_D) PORT_CHAR('D') PORT_CHAR('d')
129 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_E) PORT_CHAR('E') PORT_CHAR('e')
130 
131 	PORT_START("LINE13")
132 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_F) PORT_CHAR('F') PORT_CHAR('f')
133 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_G) PORT_CHAR('G') PORT_CHAR('g')
134 
135 	PORT_START("LINE14")
136 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_H) PORT_CHAR('H') PORT_CHAR('h')
137 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_I) PORT_CHAR('I') PORT_CHAR('i')
138 
139 	PORT_START("LINE15")
140 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_J) PORT_CHAR('J') PORT_CHAR('j')
141 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_K) PORT_CHAR('K') PORT_CHAR('k')
142 
143 	PORT_START("LINE16")
144 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_L) PORT_CHAR('L') PORT_CHAR('l')
145 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_M) PORT_CHAR('M') PORT_CHAR('m')
146 
147 	PORT_START("LINE17")
148 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_N) PORT_CHAR('N') PORT_CHAR('n')
149 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_O) PORT_CHAR('O') PORT_CHAR('o')
150 
151 	PORT_START("LINE18")
152 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_P) PORT_CHAR('P') PORT_CHAR('p')
153 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_Q) PORT_CHAR('Q') PORT_CHAR('q')
154 
155 	PORT_START("LINE19")
156 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_R) PORT_CHAR('R') PORT_CHAR('r')
157 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_S) PORT_CHAR('S') PORT_CHAR('s')
158 
159 	PORT_START("LINE20")
160 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_T) PORT_CHAR('T') PORT_CHAR('t')
161 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_U) PORT_CHAR('U') PORT_CHAR('u')
162 
163 	PORT_START("LINE21")
164 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_V) PORT_CHAR('V') PORT_CHAR('v')
165 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_W) PORT_CHAR('W') PORT_CHAR('w')
166 
167 	PORT_START("LINE22")
168 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_X) PORT_CHAR('X') PORT_CHAR('x')
169 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_Y) PORT_CHAR('Y') PORT_CHAR('y')
170 
171 	PORT_START("LINE23")
172 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_CODE(KEYCODE_Z) PORT_CHAR('Z') PORT_CHAR('z')
173 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Sh") PORT_CODE(KEYCODE_DOWN) // sh up
174 
175 	PORT_START("LINE24")
176 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Dj") PORT_CODE(KEYCODE_LEFT) // dj left
177 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Cj") PORT_CODE(KEYCODE_RIGHT) // cj right
178 
179 	PORT_START("LINE25")
180 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Ch") PORT_CODE(KEYCODE_UP)  //ch up
181 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Del") PORT_CODE(KEYCODE_TAB) PORT_CHAR(UCHAR_MAMEKEY(DEL))
182 
183 	PORT_START("CNT")
184 	PORT_BIT( 0x01, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Ctrl") PORT_CODE(KEYCODE_LCONTROL) PORT_CHANGED_MEMBER(DEVICE_SELF, pecom_state, ef_w, COSMAC_INPUT_LINE_EF1)
185 	PORT_BIT( 0x02, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Shift") PORT_CODE(KEYCODE_LSHIFT) PORT_CHAR(UCHAR_SHIFT_1)
186 	PORT_BIT( 0x04, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Caps") PORT_CODE(KEYCODE_CAPSLOCK) PORT_CHAR(UCHAR_MAMEKEY(CAPSLOCK)) PORT_TOGGLE PORT_CHANGED_MEMBER(DEVICE_SELF, pecom_state, ef_w, COSMAC_INPUT_LINE_EF3)
187 	PORT_BIT( 0x08, IP_ACTIVE_HIGH, IPT_KEYBOARD ) PORT_NAME("Break") PORT_CODE(KEYCODE_MINUS) PORT_CHANGED_MEMBER(DEVICE_SELF, pecom_state, ef_w, COSMAC_INPUT_LINE_EF4)
188 INPUT_PORTS_END
189 
190 /* Machine driver */
191 void pecom_state::pecom64(machine_config &config)
192 {
193 	/* basic machine hardware */
194 	CDP1802(config, m_maincpu, cdp1869_device::DOT_CLK_PAL);
195 	m_maincpu->set_addrmap(AS_PROGRAM, &pecom_state::mem_map);
196 	m_maincpu->set_addrmap(AS_IO, &pecom_state::io_map);
197 	m_maincpu->wait_cb().set_constant(1);
198 	m_maincpu->clear_cb().set(FUNC(pecom_state::clear_r));
199 	m_maincpu->ef2_cb().set(FUNC(pecom_state::ef2_r));
200 	m_maincpu->q_cb().set(FUNC(pecom_state::q_w));
201 	m_maincpu->sc_cb().set(FUNC(pecom_state::sc_w));
202 
203 	SPEAKER(config, "mono").front_center();
204 
205 	CDP1869(config, m_cdp1869, cdp1869_device::DOT_CLK_PAL, &pecom_state::cdp1869_page_ram);
206 	m_cdp1869->add_pal_screen(config, "screen", cdp1869_device::DOT_CLK_PAL);
207 	m_cdp1869->set_color_clock(cdp1869_device::COLOR_CLK_PAL);
208 	m_cdp1869->set_pcb_read_callback(FUNC(pecom_state::pcb_r));
209 	m_cdp1869->set_char_ram_read_callback(FUNC(pecom_state::char_ram_r));
210 	m_cdp1869->set_char_ram_write_callback(FUNC(pecom_state::char_ram_w));
211 	m_cdp1869->pal_ntsc_callback().set_constant(1);
212 	m_cdp1869->prd_callback().set(FUNC(pecom_state::prd_w));
213 	m_cdp1869->add_route(ALL_OUTPUTS, "mono", 0.25);
214 
215 	// devices
216 	CASSETTE(config, m_cassette);
217 	m_cassette->set_default_state(CASSETTE_STOPPED | CASSETTE_MOTOR_ENABLED | CASSETTE_SPEAKER_ENABLED);
218 	m_cassette->add_route(ALL_OUTPUTS, "mono", 0.05);
219 	m_cassette->set_interface("pecom_cass");
220 
221 	SOFTWARE_LIST(config, "cass_list").set_original("pecom_cass");
222 }
223 
224 /* ROM definition */
225 ROM_START( pecom32 )
226 	ROM_REGION( 0x8000, "maincpu", ROMREGION_ERASEFF )
227 	ROM_LOAD( "090786.bin", 0x0000, 0x4000, CRC(b3b1ea23) SHA1(de69f22568161ced801973345fa39d6d207b9e8c) )
228 ROM_END
229 
230 ROM_START( pecom64 )
231 	ROM_REGION( 0x8000, "maincpu", ROMREGION_ERASEFF )
232 	ROM_SYSTEM_BIOS(0, "ver4", "version 4")
233 	ROMX_LOAD( "rom_1_g_24.02.88_l.bin", 0x0000, 0x4000, CRC(9a433b47) SHA1(dadb8c399e0a25a2693e10e42a2d7fc2ea9ad427), ROM_BIOS(0) )
234 	ROMX_LOAD( "rom_2_g_24.02.88_d.bin", 0x4000, 0x4000, CRC(2116cadc) SHA1(03f11055cd221d438a40a41874af8fba0fa116d9), ROM_BIOS(0) )
235 	ROM_SYSTEM_BIOS(1, "ver1", "version 1")
236 	ROMX_LOAD( "170887-rom1.bin", 0x0000, 0x4000, CRC(43710fb4) SHA1(f84f75061c9ac3e34af93141ecabd3c955881aa2), ROM_BIOS(1) )
237 	ROMX_LOAD( "170887-rom2.bin", 0x4000, 0x4000, CRC(d0d34f08) SHA1(7baab17d1e68771b8dcef97d0fffc655beabef28), ROM_BIOS(1) )
238 ROM_END
239 
240 /* Driver */
241 
242 /*    YEAR  NAME     PARENT   COMPAT  MACHINE  INPUT  CLASS        INIT        COMPANY   FULLNAME     FLAGS */
243 COMP( 1986, pecom32, 0,       0,      pecom64, pecom, pecom_state, empty_init, "Ei Nis (Elektronska Industrija Nis)", "Pecom 32", MACHINE_SUPPORTS_SAVE )
244 COMP( 1987, pecom64, pecom32, 0,      pecom64, pecom, pecom_state, empty_init, "Ei Nis (Elektronska Industrija Nis)", "Pecom 64", MACHINE_SUPPORTS_SAVE )
245