Previous 199869 Revisions Next

r17438 Friday 24th August, 2012 at 03:13:51 UTC by R. Belmont
(MESS) Virtual Boy: correct timer handling, gives proper sound/music. [R. Belmont]
[src/mess/drivers]vboy.c

trunk/src/mess/drivers/vboy.c
r17437r17438
8181public:
8282   vboy_state(const machine_config &mconfig, device_type type, const char *tag)
8383      : driver_device(mconfig, type, tag),
84          m_maintimer(*this, "timer_main"),
8485        m_maincpu(*this, "maincpu")
8586        { }
8687
88    required_device<timer_device> m_maintimer;
8789   required_device<cpu_device> m_maincpu;
8890
8991   DECLARE_READ32_MEMBER(io_r);
r17437r17438
128130   UINT8 m_drawfb;
129131   UINT8 m_row_num;
130132   attotime m_input_latch_time;
131   void m_timer_tick(UINT8 setting);
133   void m_timer_tick(void);
132134   void m_scanline_tick(int scanline, UINT8 screen_type);
133135   void m_set_irq(UINT16 irq_vector);
134136
r17437r17438
594596      case 0x14:   // KHB (Keypad High Byte)
595597         //logerror("Ilegal write: offset %02x should be only read\n", offset);
596598         break;
597      case 0x18:   // TLB (Timer Low Byte)
599        case 0x18:   // TLB (Timer Low Byte)
598600         m_vboy_regs.tlb = data;
599601         m_vboy_timer.latch = m_vboy_regs.tlb | (m_vboy_timer.latch & 0xff00);
600602         break;
r17437r17438
611613                ---- --x- timer is zero flag
612614                ---- ---x enables timer
613615            */
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            }
620620
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;
624645         break;
625646      case 0x24:   // WCR (Wait State Control Reg)
626647         m_vboy_regs.wcr = data | 0xfc;   // according to docs: bits 2 to 7 are unused and set to 1.
r17437r17438
11371158   state->m_vip_regs.DPCTRL = 2; // ssquash relies on this at boot otherwise no frame_start irq is fired
11381159   state->m_displayfb = 0;
11391160   state->m_drawfb = 0;
1161
1162    state->m_vboy_timer.count = 0;
1163    state->m_maintimer->adjust(attotime::never);
11401164}
11411165
11421166
1143void vboy_state::m_timer_tick(UINT8 setting)
1167void vboy_state::m_timer_tick()
11441168{
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    }
11621194}
11631195
1164static TIMER_DEVICE_CALLBACK( timer_100us_tick )
1196static TIMER_DEVICE_CALLBACK( timer_main_tick )
11651197{
11661198   vboy_state *state = timer.machine().driver_data<vboy_state>();
11671199
1168   state->m_timer_tick(0x00);
1200    state->m_timer_tick();
11691201}
11701202
1171static TIMER_DEVICE_CALLBACK( timer_20us_tick )
1203static TIMER_DEVICE_CALLBACK( timer_pad_tick )
11721204{
11731205   vboy_state *state = timer.machine().driver_data<vboy_state>();
11741206
1175   state->m_timer_tick(0x10);
1207    device_set_input_line(state->m_maincpu, 0, HOLD_LINE);
11761208}
11771209
11781210static PALETTE_INIT( vboy )
r17437r17438
12931325
12941326   MCFG_MACHINE_RESET(vboy)
12951327
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)
12981330
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
12991334   /* video hardware */
13001335   MCFG_DEFAULT_LAYOUT(layout_vboy)
13011336   MCFG_VIDEO_START(vboy)

Previous 199869 Revisions Next


© 1997-2024 The MAME Team