trunk/src/mess/drivers/vboy.c
r17437 | r17438 | |
81 | 81 | public: |
82 | 82 | vboy_state(const machine_config &mconfig, device_type type, const char *tag) |
83 | 83 | : driver_device(mconfig, type, tag), |
| 84 | m_maintimer(*this, "timer_main"), |
84 | 85 | m_maincpu(*this, "maincpu") |
85 | 86 | { } |
86 | 87 | |
| 88 | required_device<timer_device> m_maintimer; |
87 | 89 | required_device<cpu_device> m_maincpu; |
88 | 90 | |
89 | 91 | DECLARE_READ32_MEMBER(io_r); |
r17437 | r17438 | |
128 | 130 | UINT8 m_drawfb; |
129 | 131 | UINT8 m_row_num; |
130 | 132 | attotime m_input_latch_time; |
131 | | void m_timer_tick(UINT8 setting); |
| 133 | void m_timer_tick(void); |
132 | 134 | void m_scanline_tick(int scanline, UINT8 screen_type); |
133 | 135 | void m_set_irq(UINT16 irq_vector); |
134 | 136 | |
r17437 | r17438 | |
594 | 596 | case 0x14: // KHB (Keypad High Byte) |
595 | 597 | //logerror("Ilegal write: offset %02x should be only read\n", offset); |
596 | 598 | break; |
597 | | case 0x18: // TLB (Timer Low Byte) |
| 599 | case 0x18: // TLB (Timer Low Byte) |
598 | 600 | m_vboy_regs.tlb = data; |
599 | 601 | m_vboy_timer.latch = m_vboy_regs.tlb | (m_vboy_timer.latch & 0xff00); |
600 | 602 | break; |
r17437 | r17438 | |
611 | 613 | ---- --x- timer is zero flag |
612 | 614 | ---- ---x enables timer |
613 | 615 | */ |
614 | | if(data & 1) |
615 | | { |
616 | | m_vboy_regs.tlb = m_vboy_timer.latch & 0xff; |
617 | | m_vboy_regs.thb = m_vboy_timer.latch >> 8; |
618 | | m_vboy_timer.count = m_vboy_timer.latch; |
619 | | } |
| 616 | if (!(data & 0x08)) |
| 617 | { |
| 618 | device_set_input_line(m_maincpu, 1, CLEAR_LINE); |
| 619 | } |
620 | 620 | |
621 | | m_vboy_regs.tcr = (data & 0xfd) | (0xe4) | (m_vboy_regs.tcr & 2); // according to docs: bits 5, 6 & 7 are unused and set to 1, bit 1 is read only. |
622 | | if(data & 4) |
623 | | m_vboy_regs.tcr &= 0xfd; |
| 621 | if (data & 1) |
| 622 | { |
| 623 | m_vboy_regs.tlb = m_vboy_timer.latch & 0xff; |
| 624 | m_vboy_regs.thb = m_vboy_timer.latch >> 8; |
| 625 | m_vboy_timer.count = m_vboy_timer.latch; |
| 626 | |
| 627 | // only start timer if tcr & 1 is 1 and wasn't before? |
| 628 | if (!(m_vboy_regs.tcr & 1)) |
| 629 | { |
| 630 | if (data & 0x10) |
| 631 | { |
| 632 | m_maintimer->adjust(attotime::from_hz(50000)); |
| 633 | } |
| 634 | else |
| 635 | { |
| 636 | m_maintimer->adjust(attotime::from_hz(10000)); |
| 637 | } |
| 638 | |
| 639 | } |
| 640 | } |
| 641 | |
| 642 | m_vboy_regs.tcr = (data & 0xfd) | (0xe4) | (m_vboy_regs.tcr & 2); // according to docs: bits 5, 6 & 7 are unused and set to 1, bit 1 is read only. |
| 643 | if(data & 4) |
| 644 | m_vboy_regs.tcr &= 0xfd; |
624 | 645 | break; |
625 | 646 | case 0x24: // WCR (Wait State Control Reg) |
626 | 647 | m_vboy_regs.wcr = data | 0xfc; // according to docs: bits 2 to 7 are unused and set to 1. |
r17437 | r17438 | |
1137 | 1158 | state->m_vip_regs.DPCTRL = 2; // ssquash relies on this at boot otherwise no frame_start irq is fired |
1138 | 1159 | state->m_displayfb = 0; |
1139 | 1160 | state->m_drawfb = 0; |
| 1161 | |
| 1162 | state->m_vboy_timer.count = 0; |
| 1163 | state->m_maintimer->adjust(attotime::never); |
1140 | 1164 | } |
1141 | 1165 | |
1142 | 1166 | |
1143 | | void vboy_state::m_timer_tick(UINT8 setting) |
| 1167 | void vboy_state::m_timer_tick() |
1144 | 1168 | { |
1145 | | if(m_vboy_regs.tcr & 1 && ((m_vboy_regs.tcr & 0x10) == setting)) |
1146 | | { |
1147 | | if(m_vboy_timer.count != 0) |
1148 | | { |
1149 | | m_vboy_timer.count--; |
1150 | | m_vboy_regs.tlb = m_vboy_timer.count & 0xff; |
1151 | | m_vboy_regs.thb = m_vboy_timer.count >> 8; |
1152 | | } |
1153 | | else |
1154 | | { |
1155 | | m_vboy_timer.count = m_vboy_timer.latch; |
1156 | | //printf("Zero timer trigger\n"); |
1157 | | m_vboy_regs.tcr |= 0x02; |
1158 | | if(m_vboy_regs.tcr & 8) |
1159 | | device_set_input_line(m_maincpu, 1, HOLD_LINE); |
1160 | | } |
1161 | | } |
| 1169 | if(m_vboy_timer.count > 0) |
| 1170 | { |
| 1171 | m_vboy_timer.count--; |
| 1172 | m_vboy_regs.tlb = m_vboy_timer.count & 0xff; |
| 1173 | m_vboy_regs.thb = m_vboy_timer.count >> 8; |
| 1174 | } |
| 1175 | |
| 1176 | if (m_vboy_timer.count == 0) |
| 1177 | { |
| 1178 | m_vboy_timer.count = m_vboy_timer.latch; |
| 1179 | m_vboy_regs.tcr |= 0x02; |
| 1180 | if(m_vboy_regs.tcr & 8) |
| 1181 | { |
| 1182 | device_set_input_line(m_maincpu, 1, ASSERT_LINE); |
| 1183 | } |
| 1184 | } |
| 1185 | |
| 1186 | if (m_vboy_regs.tcr & 0x10) |
| 1187 | { |
| 1188 | m_maintimer->adjust(attotime::from_hz(50000)); |
| 1189 | } |
| 1190 | else |
| 1191 | { |
| 1192 | m_maintimer->adjust(attotime::from_hz(10000)); |
| 1193 | } |
1162 | 1194 | } |
1163 | 1195 | |
1164 | | static TIMER_DEVICE_CALLBACK( timer_100us_tick ) |
| 1196 | static TIMER_DEVICE_CALLBACK( timer_main_tick ) |
1165 | 1197 | { |
1166 | 1198 | vboy_state *state = timer.machine().driver_data<vboy_state>(); |
1167 | 1199 | |
1168 | | state->m_timer_tick(0x00); |
| 1200 | state->m_timer_tick(); |
1169 | 1201 | } |
1170 | 1202 | |
1171 | | static TIMER_DEVICE_CALLBACK( timer_20us_tick ) |
| 1203 | static TIMER_DEVICE_CALLBACK( timer_pad_tick ) |
1172 | 1204 | { |
1173 | 1205 | vboy_state *state = timer.machine().driver_data<vboy_state>(); |
1174 | 1206 | |
1175 | | state->m_timer_tick(0x10); |
| 1207 | device_set_input_line(state->m_maincpu, 0, HOLD_LINE); |
1176 | 1208 | } |
1177 | 1209 | |
1178 | 1210 | static PALETTE_INIT( vboy ) |
r17437 | r17438 | |
1293 | 1325 | |
1294 | 1326 | MCFG_MACHINE_RESET(vboy) |
1295 | 1327 | |
1296 | | MCFG_TIMER_ADD_PERIODIC("timer_100us", timer_100us_tick, attotime::from_usec(1000)) |
1297 | | MCFG_TIMER_ADD_PERIODIC("timer_20us", timer_20us_tick, attotime::from_msec(20)) |
| 1328 | // programmable timer |
| 1329 | MCFG_TIMER_ADD("timer_main", timer_main_tick) |
1298 | 1330 | |
| 1331 | // pad ready, which should be once per VBL |
| 1332 | MCFG_TIMER_ADD_PERIODIC("timer_pad", timer_pad_tick, attotime::from_hz(50.038029f)) |
| 1333 | |
1299 | 1334 | /* video hardware */ |
1300 | 1335 | MCFG_DEFAULT_LAYOUT(layout_vboy) |
1301 | 1336 | MCFG_VIDEO_START(vboy) |