3016 lines
130 KiB
C
3016 lines
130 KiB
C
/* Z80 v0.2
|
|
______ ______ ______
|
|
/\___ \/\ __ \\ __ \
|
|
____ \/__/ /\_\ __ \\ \/\ \ ________________________________________________
|
|
| /\_____\\_____\\_____\ |
|
|
| Zilog \/_____//_____//_____/ CPU Emulator |
|
|
| Copyright (C) 1999-2026 Manuel Sainz de Baranda y Goñi. |
|
|
| |
|
|
| This emulator is free software: you can redistribute it and/or modify it |
|
|
| under the terms of the GNU Lesser General Public License as published by |
|
|
| the Free Software Foundation, either version 3 of the License, or (at your |
|
|
| option) any later version. |
|
|
| |
|
|
| This emulator is distributed in the hope that it will be useful, but |
|
|
| WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY |
|
|
| or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public |
|
|
| License for more details. |
|
|
| |
|
|
| You should have received a copy of the GNU Lesser General Public License |
|
|
| along with this emulator. If not, see <http://www.gnu.org/licenses/>. |
|
|
| |
|
|
| -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- |
|
|
| |
|
|
| A NOTE FROM THE ORIGINAL AUTHOR |
|
|
| |
|
|
| Those familiar with the official documentation of the Zilog Z80 CPU will |
|
|
| find this source code quite intuitive. The purpose has not been to write |
|
|
| the fastest possible emulator, although the speed aspect is not neglected, |
|
|
| but a portable, hackable and well- |
|
|
| .----._.----. structured piece of software; |
|
|
| A11 <-01--|1 o|--40-> A10 something small, solid and elegant |
|
|
| A12 <-02--| |--39-> A09 that can stand the test of time |
|
|
| A13 <-03--| |--38-> A08 with no need for major changes. |
|
|
| A14 <-04--| |--37-> A07 |
|
|
| A15 <-05--| |--36-> A06 Some of the main design decisions |
|
|
| CLK --06->| |--35-> A05 have been the following: |
|
|
| D4 <-07->| |--34-> A04 |
|
|
| D3 <-08->|.---------.|--33-> A03 1. Opcode partial decoding keeps |
|
|
| D5 <-09->|| ZILOG ||--32-> A02 the code small and maintainable. |
|
|
| D6 <-10->|| Z80 ||--31-> A01 |
|
|
| +5V --11--|| CPU ||--30-> A00 2. Function pointer tables for |
|
|
| D2 <-12->|| ||--29-- GND opcode selection allow easy reuse |
|
|
| D7 <-13->|'---------'|--28-> RFSH of almost all instruction code |
|
|
| D0 <-14->| |--27-> M1 in the interrupt mode 0. |
|
|
| D1 <-15->| |<-26-- RESET |
|
|
| INT --16->| |<-25-- BUSREQ 3. Avoiding conditional statements |
|
|
| NMI --17->| |<-24-- WAIT as much as possible reduces the |
|
|
| HALT <-18--| |--23-> BUSACK branch penalty in modern pipelined |
|
|
| MREQ <-19--| |--22-> WR processors. |
|
|
| IORQ <-20--| |--21-> RD |
|
|
| '-----------' |
|
|
| Zilog Z80 CPU, May 1976 version |
|
|
| 40-pin ceramic DIP pinout Manuel |
|
|
| |
|
|
'=============================================================================*/
|
|
|
|
#ifndef Z80_EXTERNAL_HEADER
|
|
# include <Z/constants/pointer.h>
|
|
# include <Z/macros/bitwise.h>
|
|
# include <Z/macros/structure.h>
|
|
#endif
|
|
|
|
#ifdef Z80_STATIC
|
|
# define Z80_API
|
|
#else
|
|
# define Z80_API Z_API_EXPORT
|
|
#endif
|
|
|
|
#ifdef Z80_WITH_LOCAL_HEADER
|
|
# include "Z80.h"
|
|
#else
|
|
# include <Z80.h>
|
|
#endif
|
|
|
|
|
|
/* MARK: - Precomputed Values of AF for `daa` */
|
|
/*---------------------------------------------------------------------------.
|
|
| Enabling `Z80_WITH_PRECOMPUTED_DAA` makes the `daa` instruction faster by |
|
|
| using a lookup table. However, this instruction is rarely used in typical |
|
|
| programs, so the overall speedup is minimal, and incresing the size of the |
|
|
| emulator by 2 KiB may negatively impact cache efficiency. It is therefore |
|
|
| recommended to leave this option disabled unless serious profiling on the |
|
|
| target platform shows a significant benefit. |
|
|
'===========================================================================*/
|
|
|
|
#ifdef Z80_WITH_PRECOMPUTED_DAA
|
|
# define H(value) Z_UINT16(0x##value)
|
|
|
|
static zuint16 const daa_af_table[2048] = {
|
|
/* HNC */
|
|
/* 000 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(0044), H(0100), H(0200), H(0304), H(0400), H(0504), H(0604), H(0700), H(0808), H(090C), H(1010), H(1114), H(1214), H(1310), H(1414), H(1510),
|
|
/* 1 */ H(1000), H(1104), H(1204), H(1300), H(1404), H(1500), H(1600), H(1704), H(180C), H(1908), H(2030), H(2134), H(2234), H(2330), H(2434), H(2530),
|
|
/* 2 */ H(2020), H(2124), H(2224), H(2320), H(2424), H(2520), H(2620), H(2724), H(282C), H(2928), H(3034), H(3130), H(3230), H(3334), H(3430), H(3534),
|
|
/* 3 */ H(3024), H(3120), H(3220), H(3324), H(3420), H(3524), H(3624), H(3720), H(3828), H(392C), H(4010), H(4114), H(4214), H(4310), H(4414), H(4510),
|
|
/* 4 */ H(4000), H(4104), H(4204), H(4300), H(4404), H(4500), H(4600), H(4704), H(480C), H(4908), H(5014), H(5110), H(5210), H(5314), H(5410), H(5514),
|
|
/* 5 */ H(5004), H(5100), H(5200), H(5304), H(5400), H(5504), H(5604), H(5700), H(5808), H(590C), H(6034), H(6130), H(6230), H(6334), H(6430), H(6534),
|
|
/* 6 */ H(6024), H(6120), H(6220), H(6324), H(6420), H(6524), H(6624), H(6720), H(6828), H(692C), H(7030), H(7134), H(7234), H(7330), H(7434), H(7530),
|
|
/* 7 */ H(7020), H(7124), H(7224), H(7320), H(7424), H(7520), H(7620), H(7724), H(782C), H(7928), H(8090), H(8194), H(8294), H(8390), H(8494), H(8590),
|
|
/* 8 */ H(8080), H(8184), H(8284), H(8380), H(8484), H(8580), H(8680), H(8784), H(888C), H(8988), H(9094), H(9190), H(9290), H(9394), H(9490), H(9594),
|
|
/* 9 */ H(9084), H(9180), H(9280), H(9384), H(9480), H(9584), H(9684), H(9780), H(9888), H(998C), H(0055), H(0111), H(0211), H(0315), H(0411), H(0515),
|
|
/* A */ H(0045), H(0101), H(0201), H(0305), H(0401), H(0505), H(0605), H(0701), H(0809), H(090D), H(1011), H(1115), H(1215), H(1311), H(1415), H(1511),
|
|
/* B */ H(1001), H(1105), H(1205), H(1301), H(1405), H(1501), H(1601), H(1705), H(180D), H(1909), H(2031), H(2135), H(2235), H(2331), H(2435), H(2531),
|
|
/* C */ H(2021), H(2125), H(2225), H(2321), H(2425), H(2521), H(2621), H(2725), H(282D), H(2929), H(3035), H(3131), H(3231), H(3335), H(3431), H(3535),
|
|
/* D */ H(3025), H(3121), H(3221), H(3325), H(3421), H(3525), H(3625), H(3721), H(3829), H(392D), H(4011), H(4115), H(4215), H(4311), H(4415), H(4511),
|
|
/* E */ H(4001), H(4105), H(4205), H(4301), H(4405), H(4501), H(4601), H(4705), H(480D), H(4909), H(5015), H(5111), H(5211), H(5315), H(5411), H(5515),
|
|
/* F */ H(5005), H(5101), H(5201), H(5305), H(5401), H(5505), H(5605), H(5701), H(5809), H(590D), H(6035), H(6131), H(6231), H(6335), H(6431), H(6535),
|
|
/* HNC */
|
|
/* 001 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(6025), H(6121), H(6221), H(6325), H(6421), H(6525), H(6625), H(6721), H(6829), H(692D), H(7031), H(7135), H(7235), H(7331), H(7435), H(7531),
|
|
/* 1 */ H(7021), H(7125), H(7225), H(7321), H(7425), H(7521), H(7621), H(7725), H(782D), H(7929), H(8091), H(8195), H(8295), H(8391), H(8495), H(8591),
|
|
/* 2 */ H(8081), H(8185), H(8285), H(8381), H(8485), H(8581), H(8681), H(8785), H(888D), H(8989), H(9095), H(9191), H(9291), H(9395), H(9491), H(9595),
|
|
/* 3 */ H(9085), H(9181), H(9281), H(9385), H(9481), H(9585), H(9685), H(9781), H(9889), H(998D), H(A0B5), H(A1B1), H(A2B1), H(A3B5), H(A4B1), H(A5B5),
|
|
/* 4 */ H(A0A5), H(A1A1), H(A2A1), H(A3A5), H(A4A1), H(A5A5), H(A6A5), H(A7A1), H(A8A9), H(A9AD), H(B0B1), H(B1B5), H(B2B5), H(B3B1), H(B4B5), H(B5B1),
|
|
/* 5 */ H(B0A1), H(B1A5), H(B2A5), H(B3A1), H(B4A5), H(B5A1), H(B6A1), H(B7A5), H(B8AD), H(B9A9), H(C095), H(C191), H(C291), H(C395), H(C491), H(C595),
|
|
/* 6 */ H(C085), H(C181), H(C281), H(C385), H(C481), H(C585), H(C685), H(C781), H(C889), H(C98D), H(D091), H(D195), H(D295), H(D391), H(D495), H(D591),
|
|
/* 7 */ H(D081), H(D185), H(D285), H(D381), H(D485), H(D581), H(D681), H(D785), H(D88D), H(D989), H(E0B1), H(E1B5), H(E2B5), H(E3B1), H(E4B5), H(E5B1),
|
|
/* 8 */ H(E0A1), H(E1A5), H(E2A5), H(E3A1), H(E4A5), H(E5A1), H(E6A1), H(E7A5), H(E8AD), H(E9A9), H(F0B5), H(F1B1), H(F2B1), H(F3B5), H(F4B1), H(F5B5),
|
|
/* 9 */ H(F0A5), H(F1A1), H(F2A1), H(F3A5), H(F4A1), H(F5A5), H(F6A5), H(F7A1), H(F8A9), H(F9AD), H(0055), H(0111), H(0211), H(0315), H(0411), H(0515),
|
|
/* A */ H(0045), H(0101), H(0201), H(0305), H(0401), H(0505), H(0605), H(0701), H(0809), H(090D), H(1011), H(1115), H(1215), H(1311), H(1415), H(1511),
|
|
/* B */ H(1001), H(1105), H(1205), H(1301), H(1405), H(1501), H(1601), H(1705), H(180D), H(1909), H(2031), H(2135), H(2235), H(2331), H(2435), H(2531),
|
|
/* C */ H(2021), H(2125), H(2225), H(2321), H(2425), H(2521), H(2621), H(2725), H(282D), H(2929), H(3035), H(3131), H(3231), H(3335), H(3431), H(3535),
|
|
/* D */ H(3025), H(3121), H(3221), H(3325), H(3421), H(3525), H(3625), H(3721), H(3829), H(392D), H(4011), H(4115), H(4215), H(4311), H(4415), H(4511),
|
|
/* E */ H(4001), H(4105), H(4205), H(4301), H(4405), H(4501), H(4601), H(4705), H(480D), H(4909), H(5015), H(5111), H(5211), H(5315), H(5411), H(5515),
|
|
/* F */ H(5005), H(5101), H(5201), H(5305), H(5401), H(5505), H(5605), H(5701), H(5809), H(590D), H(6035), H(6131), H(6231), H(6335), H(6431), H(6535),
|
|
/* HNC */
|
|
/* 010 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(0046), H(0102), H(0202), H(0306), H(0402), H(0506), H(0606), H(0702), H(080A), H(090E), H(0402), H(0506), H(0606), H(0702), H(080A), H(090E),
|
|
/* 1 */ H(1002), H(1106), H(1206), H(1302), H(1406), H(1502), H(1602), H(1706), H(180E), H(190A), H(1406), H(1502), H(1602), H(1706), H(180E), H(190A),
|
|
/* 2 */ H(2022), H(2126), H(2226), H(2322), H(2426), H(2522), H(2622), H(2726), H(282E), H(292A), H(2426), H(2522), H(2622), H(2726), H(282E), H(292A),
|
|
/* 3 */ H(3026), H(3122), H(3222), H(3326), H(3422), H(3526), H(3626), H(3722), H(382A), H(392E), H(3422), H(3526), H(3626), H(3722), H(382A), H(392E),
|
|
/* 4 */ H(4002), H(4106), H(4206), H(4302), H(4406), H(4502), H(4602), H(4706), H(480E), H(490A), H(4406), H(4502), H(4602), H(4706), H(480E), H(490A),
|
|
/* 5 */ H(5006), H(5102), H(5202), H(5306), H(5402), H(5506), H(5606), H(5702), H(580A), H(590E), H(5402), H(5506), H(5606), H(5702), H(580A), H(590E),
|
|
/* 6 */ H(6026), H(6122), H(6222), H(6326), H(6422), H(6526), H(6626), H(6722), H(682A), H(692E), H(6422), H(6526), H(6626), H(6722), H(682A), H(692E),
|
|
/* 7 */ H(7022), H(7126), H(7226), H(7322), H(7426), H(7522), H(7622), H(7726), H(782E), H(792A), H(7426), H(7522), H(7622), H(7726), H(782E), H(792A),
|
|
/* 8 */ H(8082), H(8186), H(8286), H(8382), H(8486), H(8582), H(8682), H(8786), H(888E), H(898A), H(8486), H(8582), H(8682), H(8786), H(888E), H(898A),
|
|
/* 9 */ H(9086), H(9182), H(9282), H(9386), H(9482), H(9586), H(9686), H(9782), H(988A), H(998E), H(3423), H(3527), H(3627), H(3723), H(382B), H(392F),
|
|
/* A */ H(4003), H(4107), H(4207), H(4303), H(4407), H(4503), H(4603), H(4707), H(480F), H(490B), H(4407), H(4503), H(4603), H(4707), H(480F), H(490B),
|
|
/* B */ H(5007), H(5103), H(5203), H(5307), H(5403), H(5507), H(5607), H(5703), H(580B), H(590F), H(5403), H(5507), H(5607), H(5703), H(580B), H(590F),
|
|
/* C */ H(6027), H(6123), H(6223), H(6327), H(6423), H(6527), H(6627), H(6723), H(682B), H(692F), H(6423), H(6527), H(6627), H(6723), H(682B), H(692F),
|
|
/* D */ H(7023), H(7127), H(7227), H(7323), H(7427), H(7523), H(7623), H(7727), H(782F), H(792B), H(7427), H(7523), H(7623), H(7727), H(782F), H(792B),
|
|
/* E */ H(8083), H(8187), H(8287), H(8383), H(8487), H(8583), H(8683), H(8787), H(888F), H(898B), H(8487), H(8583), H(8683), H(8787), H(888F), H(898B),
|
|
/* F */ H(9087), H(9183), H(9283), H(9387), H(9483), H(9587), H(9687), H(9783), H(988B), H(998F), H(9483), H(9587), H(9687), H(9783), H(988B), H(998F),
|
|
/* HNC */
|
|
/* 011 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(A0A7), H(A1A3), H(A2A3), H(A3A7), H(A4A3), H(A5A7), H(A6A7), H(A7A3), H(A8AB), H(A9AF), H(A4A3), H(A5A7), H(A6A7), H(A7A3), H(A8AB), H(A9AF),
|
|
/* 1 */ H(B0A3), H(B1A7), H(B2A7), H(B3A3), H(B4A7), H(B5A3), H(B6A3), H(B7A7), H(B8AF), H(B9AB), H(B4A7), H(B5A3), H(B6A3), H(B7A7), H(B8AF), H(B9AB),
|
|
/* 2 */ H(C087), H(C183), H(C283), H(C387), H(C483), H(C587), H(C687), H(C783), H(C88B), H(C98F), H(C483), H(C587), H(C687), H(C783), H(C88B), H(C98F),
|
|
/* 3 */ H(D083), H(D187), H(D287), H(D383), H(D487), H(D583), H(D683), H(D787), H(D88F), H(D98B), H(D487), H(D583), H(D683), H(D787), H(D88F), H(D98B),
|
|
/* 4 */ H(E0A3), H(E1A7), H(E2A7), H(E3A3), H(E4A7), H(E5A3), H(E6A3), H(E7A7), H(E8AF), H(E9AB), H(E4A7), H(E5A3), H(E6A3), H(E7A7), H(E8AF), H(E9AB),
|
|
/* 5 */ H(F0A7), H(F1A3), H(F2A3), H(F3A7), H(F4A3), H(F5A7), H(F6A7), H(F7A3), H(F8AB), H(F9AF), H(F4A3), H(F5A7), H(F6A7), H(F7A3), H(F8AB), H(F9AF),
|
|
/* 6 */ H(0047), H(0103), H(0203), H(0307), H(0403), H(0507), H(0607), H(0703), H(080B), H(090F), H(0403), H(0507), H(0607), H(0703), H(080B), H(090F),
|
|
/* 7 */ H(1003), H(1107), H(1207), H(1303), H(1407), H(1503), H(1603), H(1707), H(180F), H(190B), H(1407), H(1503), H(1603), H(1707), H(180F), H(190B),
|
|
/* 8 */ H(2023), H(2127), H(2227), H(2323), H(2427), H(2523), H(2623), H(2727), H(282F), H(292B), H(2427), H(2523), H(2623), H(2727), H(282F), H(292B),
|
|
/* 9 */ H(3027), H(3123), H(3223), H(3327), H(3423), H(3527), H(3627), H(3723), H(382B), H(392F), H(3423), H(3527), H(3627), H(3723), H(382B), H(392F),
|
|
/* A */ H(4003), H(4107), H(4207), H(4303), H(4407), H(4503), H(4603), H(4707), H(480F), H(490B), H(4407), H(4503), H(4603), H(4707), H(480F), H(490B),
|
|
/* B */ H(5007), H(5103), H(5203), H(5307), H(5403), H(5507), H(5607), H(5703), H(580B), H(590F), H(5403), H(5507), H(5607), H(5703), H(580B), H(590F),
|
|
/* C */ H(6027), H(6123), H(6223), H(6327), H(6423), H(6527), H(6627), H(6723), H(682B), H(692F), H(6423), H(6527), H(6627), H(6723), H(682B), H(692F),
|
|
/* D */ H(7023), H(7127), H(7227), H(7323), H(7427), H(7523), H(7623), H(7727), H(782F), H(792B), H(7427), H(7523), H(7623), H(7727), H(782F), H(792B),
|
|
/* E */ H(8083), H(8187), H(8287), H(8383), H(8487), H(8583), H(8683), H(8787), H(888F), H(898B), H(8487), H(8583), H(8683), H(8787), H(888F), H(898B),
|
|
/* F */ H(9087), H(9183), H(9283), H(9387), H(9483), H(9587), H(9687), H(9783), H(988B), H(998F), H(9483), H(9587), H(9687), H(9783), H(988B), H(998F),
|
|
/* HNC */
|
|
/* 100 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(0604), H(0700), H(0808), H(090C), H(0A0C), H(0B08), H(0C0C), H(0D08), H(0E08), H(0F0C), H(1010), H(1114), H(1214), H(1310), H(1414), H(1510),
|
|
/* 1 */ H(1600), H(1704), H(180C), H(1908), H(1A08), H(1B0C), H(1C08), H(1D0C), H(1E0C), H(1F08), H(2030), H(2134), H(2234), H(2330), H(2434), H(2530),
|
|
/* 2 */ H(2620), H(2724), H(282C), H(2928), H(2A28), H(2B2C), H(2C28), H(2D2C), H(2E2C), H(2F28), H(3034), H(3130), H(3230), H(3334), H(3430), H(3534),
|
|
/* 3 */ H(3624), H(3720), H(3828), H(392C), H(3A2C), H(3B28), H(3C2C), H(3D28), H(3E28), H(3F2C), H(4010), H(4114), H(4214), H(4310), H(4414), H(4510),
|
|
/* 4 */ H(4600), H(4704), H(480C), H(4908), H(4A08), H(4B0C), H(4C08), H(4D0C), H(4E0C), H(4F08), H(5014), H(5110), H(5210), H(5314), H(5410), H(5514),
|
|
/* 5 */ H(5604), H(5700), H(5808), H(590C), H(5A0C), H(5B08), H(5C0C), H(5D08), H(5E08), H(5F0C), H(6034), H(6130), H(6230), H(6334), H(6430), H(6534),
|
|
/* 6 */ H(6624), H(6720), H(6828), H(692C), H(6A2C), H(6B28), H(6C2C), H(6D28), H(6E28), H(6F2C), H(7030), H(7134), H(7234), H(7330), H(7434), H(7530),
|
|
/* 7 */ H(7620), H(7724), H(782C), H(7928), H(7A28), H(7B2C), H(7C28), H(7D2C), H(7E2C), H(7F28), H(8090), H(8194), H(8294), H(8390), H(8494), H(8590),
|
|
/* 8 */ H(8680), H(8784), H(888C), H(8988), H(8A88), H(8B8C), H(8C88), H(8D8C), H(8E8C), H(8F88), H(9094), H(9190), H(9290), H(9394), H(9490), H(9594),
|
|
/* 9 */ H(9684), H(9780), H(9888), H(998C), H(9A8C), H(9B88), H(9C8C), H(9D88), H(9E88), H(9F8C), H(0055), H(0111), H(0211), H(0315), H(0411), H(0515),
|
|
/* A */ H(0605), H(0701), H(0809), H(090D), H(0A0D), H(0B09), H(0C0D), H(0D09), H(0E09), H(0F0D), H(1011), H(1115), H(1215), H(1311), H(1415), H(1511),
|
|
/* B */ H(1601), H(1705), H(180D), H(1909), H(1A09), H(1B0D), H(1C09), H(1D0D), H(1E0D), H(1F09), H(2031), H(2135), H(2235), H(2331), H(2435), H(2531),
|
|
/* C */ H(2621), H(2725), H(282D), H(2929), H(2A29), H(2B2D), H(2C29), H(2D2D), H(2E2D), H(2F29), H(3035), H(3131), H(3231), H(3335), H(3431), H(3535),
|
|
/* D */ H(3625), H(3721), H(3829), H(392D), H(3A2D), H(3B29), H(3C2D), H(3D29), H(3E29), H(3F2D), H(4011), H(4115), H(4215), H(4311), H(4415), H(4511),
|
|
/* E */ H(4601), H(4705), H(480D), H(4909), H(4A09), H(4B0D), H(4C09), H(4D0D), H(4E0D), H(4F09), H(5015), H(5111), H(5211), H(5315), H(5411), H(5515),
|
|
/* F */ H(5605), H(5701), H(5809), H(590D), H(5A0D), H(5B09), H(5C0D), H(5D09), H(5E09), H(5F0D), H(6035), H(6131), H(6231), H(6335), H(6431), H(6535),
|
|
/* HNC */
|
|
/* 101 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(6625), H(6721), H(6829), H(692D), H(6A2D), H(6B29), H(6C2D), H(6D29), H(6E29), H(6F2D), H(7031), H(7135), H(7235), H(7331), H(7435), H(7531),
|
|
/* 1 */ H(7621), H(7725), H(782D), H(7929), H(7A29), H(7B2D), H(7C29), H(7D2D), H(7E2D), H(7F29), H(8091), H(8195), H(8295), H(8391), H(8495), H(8591),
|
|
/* 2 */ H(8681), H(8785), H(888D), H(8989), H(8A89), H(8B8D), H(8C89), H(8D8D), H(8E8D), H(8F89), H(9095), H(9191), H(9291), H(9395), H(9491), H(9595),
|
|
/* 3 */ H(9685), H(9781), H(9889), H(998D), H(9A8D), H(9B89), H(9C8D), H(9D89), H(9E89), H(9F8D), H(A0B5), H(A1B1), H(A2B1), H(A3B5), H(A4B1), H(A5B5),
|
|
/* 4 */ H(A6A5), H(A7A1), H(A8A9), H(A9AD), H(AAAD), H(ABA9), H(ACAD), H(ADA9), H(AEA9), H(AFAD), H(B0B1), H(B1B5), H(B2B5), H(B3B1), H(B4B5), H(B5B1),
|
|
/* 5 */ H(B6A1), H(B7A5), H(B8AD), H(B9A9), H(BAA9), H(BBAD), H(BCA9), H(BDAD), H(BEAD), H(BFA9), H(C095), H(C191), H(C291), H(C395), H(C491), H(C595),
|
|
/* 6 */ H(C685), H(C781), H(C889), H(C98D), H(CA8D), H(CB89), H(CC8D), H(CD89), H(CE89), H(CF8D), H(D091), H(D195), H(D295), H(D391), H(D495), H(D591),
|
|
/* 7 */ H(D681), H(D785), H(D88D), H(D989), H(DA89), H(DB8D), H(DC89), H(DD8D), H(DE8D), H(DF89), H(E0B1), H(E1B5), H(E2B5), H(E3B1), H(E4B5), H(E5B1),
|
|
/* 8 */ H(E6A1), H(E7A5), H(E8AD), H(E9A9), H(EAA9), H(EBAD), H(ECA9), H(EDAD), H(EEAD), H(EFA9), H(F0B5), H(F1B1), H(F2B1), H(F3B5), H(F4B1), H(F5B5),
|
|
/* 9 */ H(F6A5), H(F7A1), H(F8A9), H(F9AD), H(FAAD), H(FBA9), H(FCAD), H(FDA9), H(FEA9), H(FFAD), H(0055), H(0111), H(0211), H(0315), H(0411), H(0515),
|
|
/* A */ H(0605), H(0701), H(0809), H(090D), H(0A0D), H(0B09), H(0C0D), H(0D09), H(0E09), H(0F0D), H(1011), H(1115), H(1215), H(1311), H(1415), H(1511),
|
|
/* B */ H(1601), H(1705), H(180D), H(1909), H(1A09), H(1B0D), H(1C09), H(1D0D), H(1E0D), H(1F09), H(2031), H(2135), H(2235), H(2331), H(2435), H(2531),
|
|
/* C */ H(2621), H(2725), H(282D), H(2929), H(2A29), H(2B2D), H(2C29), H(2D2D), H(2E2D), H(2F29), H(3035), H(3131), H(3231), H(3335), H(3431), H(3535),
|
|
/* D */ H(3625), H(3721), H(3829), H(392D), H(3A2D), H(3B29), H(3C2D), H(3D29), H(3E29), H(3F2D), H(4011), H(4115), H(4215), H(4311), H(4415), H(4511),
|
|
/* E */ H(4601), H(4705), H(480D), H(4909), H(4A09), H(4B0D), H(4C09), H(4D0D), H(4E0D), H(4F09), H(5015), H(5111), H(5211), H(5315), H(5411), H(5515),
|
|
/* F */ H(5605), H(5701), H(5809), H(590D), H(5A0D), H(5B09), H(5C0D), H(5D09), H(5E09), H(5F0D), H(6035), H(6131), H(6231), H(6335), H(6431), H(6535),
|
|
/* HNC */
|
|
/* 110 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(FABE), H(FBBA), H(FCBE), H(FDBA), H(FEBA), H(FFBE), H(0046), H(0102), H(0202), H(0306), H(0402), H(0506), H(0606), H(0702), H(080A), H(090E),
|
|
/* 1 */ H(0A1E), H(0B1A), H(0C1E), H(0D1A), H(0E1A), H(0F1E), H(1002), H(1106), H(1206), H(1302), H(1406), H(1502), H(1602), H(1706), H(180E), H(190A),
|
|
/* 2 */ H(1A1A), H(1B1E), H(1C1A), H(1D1E), H(1E1E), H(1F1A), H(2022), H(2126), H(2226), H(2322), H(2426), H(2522), H(2622), H(2726), H(282E), H(292A),
|
|
/* 3 */ H(2A3A), H(2B3E), H(2C3A), H(2D3E), H(2E3E), H(2F3A), H(3026), H(3122), H(3222), H(3326), H(3422), H(3526), H(3626), H(3722), H(382A), H(392E),
|
|
/* 4 */ H(3A3E), H(3B3A), H(3C3E), H(3D3A), H(3E3A), H(3F3E), H(4002), H(4106), H(4206), H(4302), H(4406), H(4502), H(4602), H(4706), H(480E), H(490A),
|
|
/* 5 */ H(4A1A), H(4B1E), H(4C1A), H(4D1E), H(4E1E), H(4F1A), H(5006), H(5102), H(5202), H(5306), H(5402), H(5506), H(5606), H(5702), H(580A), H(590E),
|
|
/* 6 */ H(5A1E), H(5B1A), H(5C1E), H(5D1A), H(5E1A), H(5F1E), H(6026), H(6122), H(6222), H(6326), H(6422), H(6526), H(6626), H(6722), H(682A), H(692E),
|
|
/* 7 */ H(6A3E), H(6B3A), H(6C3E), H(6D3A), H(6E3A), H(6F3E), H(7022), H(7126), H(7226), H(7322), H(7426), H(7522), H(7622), H(7726), H(782E), H(792A),
|
|
/* 8 */ H(7A3A), H(7B3E), H(7C3A), H(7D3E), H(7E3E), H(7F3A), H(8082), H(8186), H(8286), H(8382), H(8486), H(8582), H(8682), H(8786), H(888E), H(898A),
|
|
/* 9 */ H(8A9A), H(8B9E), H(8C9A), H(8D9E), H(8E9E), H(8F9A), H(9086), H(9182), H(9282), H(9386), H(3423), H(3527), H(3627), H(3723), H(382B), H(392F),
|
|
/* A */ H(3A3F), H(3B3B), H(3C3F), H(3D3B), H(3E3B), H(3F3F), H(4003), H(4107), H(4207), H(4303), H(4407), H(4503), H(4603), H(4707), H(480F), H(490B),
|
|
/* B */ H(4A1B), H(4B1F), H(4C1B), H(4D1F), H(4E1F), H(4F1B), H(5007), H(5103), H(5203), H(5307), H(5403), H(5507), H(5607), H(5703), H(580B), H(590F),
|
|
/* C */ H(5A1F), H(5B1B), H(5C1F), H(5D1B), H(5E1B), H(5F1F), H(6027), H(6123), H(6223), H(6327), H(6423), H(6527), H(6627), H(6723), H(682B), H(692F),
|
|
/* D */ H(6A3F), H(6B3B), H(6C3F), H(6D3B), H(6E3B), H(6F3F), H(7023), H(7127), H(7227), H(7323), H(7427), H(7523), H(7623), H(7727), H(782F), H(792B),
|
|
/* E */ H(7A3B), H(7B3F), H(7C3B), H(7D3F), H(7E3F), H(7F3B), H(8083), H(8187), H(8287), H(8383), H(8487), H(8583), H(8683), H(8787), H(888F), H(898B),
|
|
/* F */ H(8A9B), H(8B9F), H(8C9B), H(8D9F), H(8E9F), H(8F9B), H(9087), H(9183), H(9283), H(9387), H(9483), H(9587), H(9687), H(9783), H(988B), H(998F),
|
|
/* HNC */
|
|
/* 111 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ H(9A9F), H(9B9B), H(9C9F), H(9D9B), H(9E9B), H(9F9F), H(A0A7), H(A1A3), H(A2A3), H(A3A7), H(A4A3), H(A5A7), H(A6A7), H(A7A3), H(A8AB), H(A9AF),
|
|
/* 1 */ H(AABF), H(ABBB), H(ACBF), H(ADBB), H(AEBB), H(AFBF), H(B0A3), H(B1A7), H(B2A7), H(B3A3), H(B4A7), H(B5A3), H(B6A3), H(B7A7), H(B8AF), H(B9AB),
|
|
/* 2 */ H(BABB), H(BBBF), H(BCBB), H(BDBF), H(BEBF), H(BFBB), H(C087), H(C183), H(C283), H(C387), H(C483), H(C587), H(C687), H(C783), H(C88B), H(C98F),
|
|
/* 3 */ H(CA9F), H(CB9B), H(CC9F), H(CD9B), H(CE9B), H(CF9F), H(D083), H(D187), H(D287), H(D383), H(D487), H(D583), H(D683), H(D787), H(D88F), H(D98B),
|
|
/* 4 */ H(DA9B), H(DB9F), H(DC9B), H(DD9F), H(DE9F), H(DF9B), H(E0A3), H(E1A7), H(E2A7), H(E3A3), H(E4A7), H(E5A3), H(E6A3), H(E7A7), H(E8AF), H(E9AB),
|
|
/* 5 */ H(EABB), H(EBBF), H(ECBB), H(EDBF), H(EEBF), H(EFBB), H(F0A7), H(F1A3), H(F2A3), H(F3A7), H(F4A3), H(F5A7), H(F6A7), H(F7A3), H(F8AB), H(F9AF),
|
|
/* 6 */ H(FABF), H(FBBB), H(FCBF), H(FDBB), H(FEBB), H(FFBF), H(0047), H(0103), H(0203), H(0307), H(0403), H(0507), H(0607), H(0703), H(080B), H(090F),
|
|
/* 7 */ H(0A1F), H(0B1B), H(0C1F), H(0D1B), H(0E1B), H(0F1F), H(1003), H(1107), H(1207), H(1303), H(1407), H(1503), H(1603), H(1707), H(180F), H(190B),
|
|
/* 8 */ H(1A1B), H(1B1F), H(1C1B), H(1D1F), H(1E1F), H(1F1B), H(2023), H(2127), H(2227), H(2323), H(2427), H(2523), H(2623), H(2727), H(282F), H(292B),
|
|
/* 9 */ H(2A3B), H(2B3F), H(2C3B), H(2D3F), H(2E3F), H(2F3B), H(3027), H(3123), H(3223), H(3327), H(3423), H(3527), H(3627), H(3723), H(382B), H(392F),
|
|
/* A */ H(3A3F), H(3B3B), H(3C3F), H(3D3B), H(3E3B), H(3F3F), H(4003), H(4107), H(4207), H(4303), H(4407), H(4503), H(4603), H(4707), H(480F), H(490B),
|
|
/* B */ H(4A1B), H(4B1F), H(4C1B), H(4D1F), H(4E1F), H(4F1B), H(5007), H(5103), H(5203), H(5307), H(5403), H(5507), H(5607), H(5703), H(580B), H(590F),
|
|
/* C */ H(5A1F), H(5B1B), H(5C1F), H(5D1B), H(5E1B), H(5F1F), H(6027), H(6123), H(6223), H(6327), H(6423), H(6527), H(6627), H(6723), H(682B), H(692F),
|
|
/* D */ H(6A3F), H(6B3B), H(6C3F), H(6D3B), H(6E3B), H(6F3F), H(7023), H(7127), H(7227), H(7323), H(7427), H(7523), H(7623), H(7727), H(782F), H(792B),
|
|
/* E */ H(7A3B), H(7B3F), H(7C3B), H(7D3F), H(7E3F), H(7F3B), H(8083), H(8187), H(8287), H(8383), H(8487), H(8583), H(8683), H(8787), H(888F), H(898B),
|
|
/* F */ H(8A9B), H(8B9F), H(8C9B), H(8D9F), H(8E9F), H(8F9B), H(9087), H(9183), H(9283), H(9387), H(9483), H(9587), H(9687), H(9783), H(988B), H(998F)};
|
|
|
|
# undef H
|
|
#endif
|
|
|
|
|
|
/* MARK: - Types */
|
|
|
|
typedef zuint8 (* Insn)(Z80 *self);
|
|
|
|
#ifdef Z80_WITH_FULL_IM0
|
|
typedef struct {
|
|
Z80* z80;
|
|
void* context;
|
|
Z80Read fetch;
|
|
Z80Read read;
|
|
Z80Write write;
|
|
Z80Read in;
|
|
Z80Write out;
|
|
Z80Notify ld_i_a;
|
|
Z80Notify ld_r_a;
|
|
Z80Notify reti;
|
|
Z80Notify retn;
|
|
zuint16 pc;
|
|
} IM0;
|
|
#endif
|
|
|
|
|
|
/* MARK: - Shortcuts for Instance Variables and Callbacks */
|
|
|
|
#define MEMPTR self->memptr.uint16_value
|
|
#define PC self->pc.uint16_value
|
|
#define SP self->sp.uint16_value
|
|
#define XY self->xy.uint16_value
|
|
#define IX self->ix_iy[0].uint16_value
|
|
#define IY self->ix_iy[1].uint16_value
|
|
#define AF self->af.uint16_value
|
|
#define BC self->bc.uint16_value
|
|
#define DE self->de.uint16_value
|
|
#define HL self->hl.uint16_value
|
|
#define AF_ self->af_.uint16_value
|
|
#define BC_ self->bc_.uint16_value
|
|
#define DE_ self->de_.uint16_value
|
|
#define HL_ self->hl_.uint16_value
|
|
#define MEMPTRH self->memptr.uint8_values.at_1
|
|
#define MEMPTRL self->memptr.uint8_values.at_0
|
|
#define PCH self->pc.uint8_values.at_1
|
|
#define A self->af.uint8_values.at_1
|
|
#define F self->af.uint8_values.at_0
|
|
#define B self->bc.uint8_values.at_1
|
|
#define C self->bc.uint8_values.at_0
|
|
#define E self->de.uint8_values.at_0
|
|
#define L self->hl.uint8_values.at_0
|
|
#define I self->i
|
|
#define R self->r
|
|
#define R7 self->r7
|
|
#define Q self->q
|
|
#define IFF1 self->iff1
|
|
#define IFF2 self->iff2
|
|
#define IM self->im
|
|
#define HALT_LINE self->halt_line
|
|
#define INT_LINE self->int_line
|
|
#define DATA self->data.uint8_array
|
|
#define REQUEST self->request
|
|
#define RESUME self->resume
|
|
#define OPTIONS self->options
|
|
#define CONTEXT self->context
|
|
|
|
#define FETCH_OPCODE(address) self->fetch_opcode(CONTEXT, address)
|
|
#define FETCH(address) self->fetch (CONTEXT, address)
|
|
#define READ(address) self->read (CONTEXT, address)
|
|
#define WRITE(address, value) self->write (CONTEXT, address, value)
|
|
#define IN(port) self->in (CONTEXT, port)
|
|
#define OUT(port, value) self->out (CONTEXT, port, value)
|
|
#define NOTIFY(callback) if (self->callback != Z_NULL) self->callback(CONTEXT)
|
|
|
|
|
|
/* MARK: - 16-bit Callback Operations */
|
|
|
|
static inline zuint16 fetch_16(Z80 *self, zuint16 address)
|
|
{
|
|
zuint8 l = FETCH(address);
|
|
return (zuint16)(l | ((zuint16)FETCH(address + 1) << 8));
|
|
}
|
|
|
|
|
|
static inline zuint16 read_16(Z80 *self, zuint16 address)
|
|
{
|
|
zuint8 l = READ(address);
|
|
return (zuint16)(l | ((zuint16)READ(address + 1) << 8));
|
|
}
|
|
|
|
|
|
static inline void write_16f(Z80 *self, zuint16 address, zuint16 value)
|
|
{
|
|
WRITE(address, (zuint8)value);
|
|
WRITE(address + 1, (zuint8)(value >> 8));
|
|
}
|
|
|
|
|
|
static inline void write_16b(Z80 *self, zuint16 address, zuint16 value)
|
|
{
|
|
WRITE(address + 1, (zuint8)(value >> 8));
|
|
WRITE(address, (zuint8)value);
|
|
}
|
|
|
|
|
|
#ifndef Z80_WITH_FULL_IM0
|
|
static inline zuint16 int_fetch_16(Z80 *self)
|
|
{
|
|
zuint8 l = self->int_fetch(CONTEXT, PC);
|
|
return (zuint16)(l | ((zuint16)self->int_fetch(CONTEXT, PC) << 8));
|
|
}
|
|
#endif
|
|
|
|
|
|
#define FETCH_16(address) fetch_16 (self, address)
|
|
#define READ_16(address) read_16 (self, address)
|
|
#define WRITE_16F(address, value) write_16f(self, address, value)
|
|
#define WRITE_16B(address, value) write_16b(self, address, value)
|
|
|
|
|
|
/* MARK: - Interrupt Mode 0: Callback Trampolines */
|
|
|
|
#ifdef Z80_WITH_FULL_IM0
|
|
static zuint8 im0_fetch(IM0 const *self, zuint16 address)
|
|
{
|
|
Z_UNUSED(address)
|
|
return self->z80->int_fetch(CONTEXT, self->pc);
|
|
}
|
|
|
|
|
|
static zuint8 im0_read(IM0 const *self, zuint16 address)
|
|
{return READ(address);}
|
|
|
|
|
|
static void im0_write(IM0 const *self, zuint16 address, zuint8 value)
|
|
{WRITE(address, value);}
|
|
|
|
|
|
static zuint8 im0_in(IM0 const *self, zuint16 port)
|
|
{return IN(port);}
|
|
|
|
|
|
static void im0_out(IM0 const *self, zuint16 port, zuint8 value)
|
|
{OUT(port, value);}
|
|
|
|
|
|
static void im0_ld_i_a(IM0 const *self) {NOTIFY(ld_i_a);}
|
|
static void im0_ld_r_a(IM0 const *self) {NOTIFY(ld_r_a);}
|
|
|
|
|
|
# ifdef Z80_WITH_IM0_RETX_NOTIFICATIONS
|
|
# define IM0_NOTIFY_RETX(callback) \
|
|
if ( self->callback != Z_NULL && \
|
|
(self->z80->options & \
|
|
Z80_OPTION_IM0_RETX_NOTIFICATIONS) \
|
|
) \
|
|
{ \
|
|
self->z80->data.uint8_array[2] |= 4; \
|
|
self->callback(CONTEXT); \
|
|
}
|
|
|
|
|
|
static void im0_reti(IM0 const *self) {IM0_NOTIFY_RETX(reti)}
|
|
static void im0_retn(IM0 const *self) {IM0_NOTIFY_RETX(retn)}
|
|
# endif
|
|
#endif
|
|
|
|
|
|
/* MARK: - Interrupt Mode 0: PC Decrements for Unprefixed Instructions */
|
|
|
|
#ifdef Z80_WITH_FULL_IM0
|
|
static zuint8 const im0_pc_decrement_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 1 */ 2, 0, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 2 */ 2, 0, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 3 */ 2, 0, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 4 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 5 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 6 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 7 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 8 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* 9 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* A */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* B */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
/* C */ 1, 0, 3, 3, 3, 0, 0, 1, 1, 1, 3, 0, 3, 3, 0, 1,
|
|
/* D */ 1, 0, 3, 0, 3, 0, 0, 1, 1, 0, 3, 0, 3, 0, 0, 1,
|
|
/* E */ 1, 0, 3, 0, 3, 0, 0, 1, 1, 1, 3, 0, 3, 0, 0, 1,
|
|
/* F */ 1, 0, 3, 0, 3, 0, 0, 1, 1, 0, 3, 0, 3, 0, 0, 1};
|
|
#endif
|
|
|
|
|
|
/* MARK: - Flags */
|
|
|
|
/*----------------.
|
|
| 7 6 5 4 3 2 1 0 |
|
|
| S Z Y H X P N C |
|
|
'-|-|-|-|-|-|-|-|-'
|
|
| | | | | | | '-- carry / borrow
|
|
| | | | | | '---- addition / subtraction
|
|
| | | | | '------ parity (P) / two's complement signed overflow (V)
|
|
| | | | '-------- result's bit 3 (undocumented)
|
|
| | | '---------- half carry / half borrow
|
|
| | '------------ result's bit 5 (undocumented)
|
|
| '-------------- zero
|
|
'---------------- sign */
|
|
|
|
#define SF 128
|
|
#define ZF 64
|
|
#define YF 32
|
|
#define HF 16
|
|
#define XF 8
|
|
#define PF 4
|
|
#define NF 2
|
|
#define CF 1
|
|
|
|
#define SZPCF (SF | ZF | PF | CF)
|
|
#define SZPF (SF | ZF | PF )
|
|
#define SZCF (SF | ZF | CF )
|
|
#define SYXF (SF | YF | XF )
|
|
#define ZPF (ZF | PF )
|
|
#define YXCF (YF | XF | CF )
|
|
#define YXF (YF | XF )
|
|
#define HCF (HF | CF )
|
|
|
|
#define F_SZPC (F & SZPCF)
|
|
#define F_SZP (F & SZPF)
|
|
#define F_SZC (F & SZCF)
|
|
#define F_C (F & CF)
|
|
#define A_SYX (A & SYXF)
|
|
#define A_YX (A & YXF)
|
|
|
|
#define ZF_ZERO(value) (!(value) << 6)
|
|
|
|
/*---------------------------------------------------------------------.
|
|
| `PF_PARITY` computes PF according to the parity of the given byte. |
|
|
| Enabling `Z80_WITH_PARITY_COMPUTATION` is strongly discouraged and |
|
|
| is provided only for benchmarking and educational purposes. |
|
|
| |
|
|
| For an explanation of the parity computation formula, check: |
|
|
| * http://graphics.stanford.edu/~seander/bithacks.html#ParityParallel |
|
|
'=====================================================================*/
|
|
|
|
#ifdef Z80_WITH_PARITY_COMPUTATION
|
|
static inline zuint8 pf_parity(zuint8 value)
|
|
{return (zuint8)(((0x9669U >> ((value ^ (value >> 4)) & 0xF)) & 1) << 2);}
|
|
|
|
# define PF_PARITY pf_parity
|
|
#else
|
|
static zuint8 const pf_parity_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* 1 */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* 2 */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* 3 */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* 4 */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* 5 */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* 6 */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* 7 */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* 8 */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* 9 */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* A */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* B */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* C */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4,
|
|
/* D */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* E */ 0, 4, 4, 0, 4, 0, 0, 4, 4, 0, 0, 4, 0, 4, 4, 0,
|
|
/* F */ 4, 0, 0, 4, 0, 4, 4, 0, 0, 4, 4, 0, 4, 0, 0, 4};
|
|
|
|
# define PF_PARITY(value) pf_parity_table[value]
|
|
#endif
|
|
|
|
/*-----------------------------------------------------------------------------.
|
|
| `PF_OVERFLOW` computes PF according to whether signed overflow occurs in the |
|
|
| addition or subtraction of two integers. For additions, `rhs` must be passed |
|
|
| bitwise ~inverted. |
|
|
| |
|
|
| For an explanation of the formula, check: |
|
|
| * https://stackoverflow.com/a/199668 |
|
|
| * http://www.cs.umd.edu/class/spring2003/cmsc311/Notes/Comb/overflow.html |
|
|
'=============================================================================*/
|
|
|
|
#define PF_OVERFLOW(width, result, lhs, rhs) \
|
|
(((zuint##width)((lhs ^ rhs) & (lhs ^ result)) >> (width - 3)) & PF)
|
|
|
|
/*-----------------------------------------------------------------------------.
|
|
| Q serves as an abstraction for a set of latches related to flag computation. |
|
|
| From an emulation perspective, instructions that affect the flags copy the |
|
|
| final value of F to Q, whereas instructions that do not affect the flags |
|
|
| (including `ex af,af'`, `pop af`, internal NOPs and interrupt responses) set |
|
|
| Q to 0. Q is used to compute YF and XF in the `ccf` and `scf` instructions. |
|
|
| |
|
|
| References: |
|
|
| * https://worldofspectrum.org/forums/discussion/20345 |
|
|
| * https://worldofspectrum.org/forums/discussion/41704 |
|
|
'=============================================================================*/
|
|
|
|
#ifdef Z80_WITH_Q
|
|
# define FLAGS Q = F
|
|
# define Q_0 Q = 0;
|
|
#else
|
|
# define FLAGS F
|
|
# define Q_0
|
|
#endif
|
|
|
|
|
|
/* MARK: - Bit Rotation */
|
|
|
|
#define ROL(value) Z_UINT8_ROTATE_LEFT (value, 1)
|
|
#define ROR(value) Z_UINT8_ROTATE_RIGHT(value, 1)
|
|
|
|
|
|
/* MARK: - 8-Bit Register Resolution */
|
|
|
|
/*---------. .---------------------.
|
|
| 76543210 | | J / K | O / P |
|
|
|----------| |---------+-----------|
|
|
| __jjj___ | | 000 = b | 000 = b |
|
|
| _____kkk | | 001 = c | 001 = c |
|
|
| __ooo___ | | 010 = d | 010 = d |
|
|
| _____ppp | | 011 = e | 011 = e |
|
|
'----------' | 100 = h | 100 = XYh |
|
|
| 101 = l | 101 = XYl |
|
|
| 111 = a | 111 = a |
|
|
'--------------------*/
|
|
|
|
static zusize const j_k_table[8] = {
|
|
Z_MEMBER_OFFSET(Z80, bc.uint8_values.at_1),
|
|
Z_MEMBER_OFFSET(Z80, bc.uint8_values.at_0),
|
|
Z_MEMBER_OFFSET(Z80, de.uint8_values.at_1),
|
|
Z_MEMBER_OFFSET(Z80, de.uint8_values.at_0),
|
|
Z_MEMBER_OFFSET(Z80, hl.uint8_values.at_1),
|
|
Z_MEMBER_OFFSET(Z80, hl.uint8_values.at_0),
|
|
0,
|
|
Z_MEMBER_OFFSET(Z80, af.uint8_values.at_1)};
|
|
|
|
static zusize const o_p_table[8] = {
|
|
Z_MEMBER_OFFSET(Z80, bc.uint8_values.at_1),
|
|
Z_MEMBER_OFFSET(Z80, bc.uint8_values.at_0),
|
|
Z_MEMBER_OFFSET(Z80, de.uint8_values.at_1),
|
|
Z_MEMBER_OFFSET(Z80, de.uint8_values.at_0),
|
|
Z_MEMBER_OFFSET(Z80, xy.uint8_values.at_1),
|
|
Z_MEMBER_OFFSET(Z80, xy.uint8_values.at_0),
|
|
0,
|
|
Z_MEMBER_OFFSET(Z80, af.uint8_values.at_1)};
|
|
|
|
#define REGISTER_8(table, offset, shift) \
|
|
*((zuint8 *)self + table[(DATA[offset] shift) & 7])
|
|
|
|
#define J0 REGISTER_8(j_k_table, 0, >> 3 )
|
|
#define J1 REGISTER_8(j_k_table, 1, >> 3 )
|
|
#define K0 REGISTER_8(j_k_table, 0, Z_EMPTY)
|
|
#define K1 REGISTER_8(j_k_table, 1, Z_EMPTY)
|
|
#define K3 REGISTER_8(j_k_table, 3, Z_EMPTY)
|
|
#define O REGISTER_8(o_p_table, 1, >> 3 )
|
|
#define P REGISTER_8(o_p_table, 1, Z_EMPTY)
|
|
|
|
|
|
/* MARK: - 16-Bit Register Resolution */
|
|
|
|
/*---------. .-----------------------------.
|
|
| 76543210 | | S | T | W |
|
|
|----------| |---------+---------+---------|
|
|
| __ss____ | | 00 = bc | 00 = bc | 00 = bc |
|
|
| __tt____ | | 01 = de | 01 = de | 01 = de |
|
|
| __ww____ | | 10 = hl | 10 = hl | 10 = XY |
|
|
'----------' | 11 = sp | 11 = af | 11 = sp |
|
|
'----------------------------*/
|
|
|
|
static zusize const s_table[4] = {
|
|
Z_MEMBER_OFFSET(Z80, bc.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, de.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, hl.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, sp.uint16_value)};
|
|
|
|
static zusize const t_table[4] = {
|
|
Z_MEMBER_OFFSET(Z80, bc.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, de.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, hl.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, af.uint16_value)};
|
|
|
|
static zusize const w_table[4] = {
|
|
Z_MEMBER_OFFSET(Z80, bc.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, de.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, xy.uint16_value),
|
|
Z_MEMBER_OFFSET(Z80, sp.uint16_value)};
|
|
|
|
#define REGISTER_16(table, offset) \
|
|
*(zuint16 *)(void *)((zchar *)self + table[(DATA[offset] >> 4) & 3])
|
|
|
|
#define SS0 REGISTER_16(s_table, 0)
|
|
#define SS1 REGISTER_16(s_table, 1)
|
|
#define TT REGISTER_16(t_table, 0)
|
|
#define WW REGISTER_16(w_table, 1)
|
|
|
|
|
|
/* MARK: - Condition Evaluation */
|
|
|
|
/*---------. .----------.
|
|
| 76543210 | | Z |
|
|
|----------| |----------|
|
|
| __zzz___ | | 000 = nz |
|
|
| ___zz___ | | 001 = z |
|
|
'----------' | 010 = nc |
|
|
| 011 = c |
|
|
| 100 = po |
|
|
| 101 = pe |
|
|
| 110 = p |
|
|
| 111 = m |
|
|
'---------*/
|
|
|
|
static zuint8 const z_table[8] = {ZF, ZF, CF, CF, PF, PF, SF, SF};
|
|
|
|
|
|
static inline zsint zzz(Z80 const *self, zuint8 mask)
|
|
{
|
|
zsint z = (DATA[0] >> 3) & mask;
|
|
|
|
return !(F & z_table[z]) ^ (z & 1);
|
|
}
|
|
|
|
|
|
/* MARK: - 8-Bit Arithmetic and Logical Operations */
|
|
|
|
/*---------. .---------------------.
|
|
| 76543210 | | U SZYHXPNC |
|
|
|----------| |---------------------|
|
|
| __uuu___ | | 000 = add szycxv0c |
|
|
| _____10v | | 001 = adc szycxv0c |
|
|
'----------' | 010 = sub szybxv1b |
|
|
| 011 = sbc szybxv1b |
|
|
| 100 = and szy1xp00 |
|
|
| 101 = xor szy0xp00 |
|
|
| 110 = or szy0xp00 |
|
|
| 111 = cp sz*b*v1b |
|
|
|---------------------|
|
|
| V SZYHXPNC |
|
|
|---------------------|
|
|
| 100 = inc szycxv0. |
|
|
| 101 = dec szybxv1. |
|
|
'--------------------*/
|
|
|
|
static void uuu(Z80 *self, zuint8 offset, zuint8 rhs)
|
|
{
|
|
zuint8 t, f;
|
|
|
|
switch ((DATA[offset] >> 3) & 7)
|
|
{
|
|
case 0: /* add */
|
|
t = A + rhs;
|
|
|
|
f = ((zuint)A + rhs > 255) | /* CF = carry */
|
|
PF_OVERFLOW(8, t, A, ~rhs) | /* PF = overflow */
|
|
((A ^ rhs ^ t) & HF); /* HF = half-carry */
|
|
/* NF = 0 */
|
|
A = t;
|
|
break;
|
|
|
|
case 1: /* adc */
|
|
t = A + rhs + (f = F_C);
|
|
|
|
f = ((zuint)A + rhs + f > 255) | /* CF = carry */
|
|
PF_OVERFLOW(8, t, A, ~rhs) | /* PF = overflow */
|
|
((A ^ rhs ^ t) & HF); /* HF = half-carry */
|
|
/* NF = 0 */
|
|
A = t;
|
|
break;
|
|
|
|
case 2: /* sub */
|
|
t = A - rhs;
|
|
|
|
f = (A < rhs) | /* CF = borrow */
|
|
NF | /* NF = 1 */
|
|
PF_OVERFLOW(8, t, A, rhs) | /* PF = overflow */
|
|
((A ^ rhs ^ t) & HF); /* HF = half-borrow */
|
|
|
|
A = t;
|
|
break;
|
|
|
|
case 3: /* sbc */
|
|
t = A - rhs - (f = F_C);
|
|
|
|
f = ((zsint)A - rhs - f < 0) | /* CF = borrow */
|
|
NF | /* NF = 1 */
|
|
PF_OVERFLOW(8, t, A, rhs) | /* PF = overflow */
|
|
((A ^ rhs ^ t) & HF); /* HF = half-borrow */
|
|
|
|
A = t;
|
|
break;
|
|
|
|
case 4: /* and */
|
|
A &= rhs;
|
|
f = HF | PF_PARITY(A); /* HF = 1; PF = parity */
|
|
break; /* NF, CF = 0 */
|
|
|
|
case 5: /* xor */
|
|
A ^= rhs;
|
|
f = PF_PARITY(A); /* PF = parity */
|
|
break; /* HF, NF, CF = 0 */
|
|
|
|
case 6: /* or */
|
|
A |= rhs;
|
|
f = PF_PARITY(A); /* PF = parity */
|
|
break; /* HF, NF, CF = 0 */
|
|
|
|
case 7: /* cp */
|
|
t = A - rhs;
|
|
|
|
FLAGS = (zuint8)(
|
|
(t & SF) | /* SF = sign */
|
|
ZF_ZERO(t) | /* ZF = zero */
|
|
((A ^ rhs ^ t) & HF) | /* HF = half-borrow */
|
|
PF_OVERFLOW(8, t, A, rhs) | /* PF = overflow */
|
|
(A < rhs) | /* CF = borrow */
|
|
(rhs & YXF) | /* YF = rhs.5; XF = rhs.3 */
|
|
NF); /* NF = 1 */
|
|
|
|
return;
|
|
}
|
|
|
|
FLAGS = (zuint8)(
|
|
f | /* HF, PF, NF and CF already computed */
|
|
A_SYX | /* SF = sign; YF = Y; XF = X */
|
|
ZF_ZERO(A)); /* ZF = zero */
|
|
}
|
|
|
|
|
|
static zuint8 vvv(Z80 *self, zuint8 offset, zuint8 value)
|
|
{
|
|
zuint8 dec = DATA[offset] & 1;
|
|
zuint8 nf = (zuint8)(dec << 1);
|
|
zuint8 t = value + 1 - nf;
|
|
|
|
FLAGS = (zuint8)(
|
|
(t & SYXF) | /* SF = sign; YF = Y; XF = X */
|
|
ZF_ZERO(t) | /* ZF = zero */
|
|
((value ^ t) & HF) | /* HF = half-carry/borrow */
|
|
((value == 127 + dec) << 2) | /* PF = overflow */
|
|
nf | /* NF = 0 (inc), 1 (dec) */
|
|
F_C); /* CF unchanged */
|
|
|
|
return t;
|
|
}
|
|
|
|
|
|
/* MARK: - Rotation and Shift Operations */
|
|
|
|
/*---------. .-----------.
|
|
| 76543210 | | G |
|
|
|----------| |-----------|
|
|
| __ggg___ | | 000 = rlc |
|
|
'----------' | 001 = rrc |
|
|
| 010 = rl |
|
|
| 011 = rr |
|
|
| 100 = sla |
|
|
| 101 = sra |
|
|
| 110 = sll |
|
|
| 111 = srl |
|
|
'----------*/
|
|
|
|
static zuint8 ggg(Z80 *self, zuint8 offset, zuint8 value)
|
|
{
|
|
zuint8 cf;
|
|
|
|
switch ((DATA[offset] >> 3) & 7)
|
|
{
|
|
/* rlc .----------------.
|
|
.----. | .---------. |
|
|
| CF |<-----| 7 <-- 0 |<--'
|
|
'----' '--------*/
|
|
case 0:
|
|
cf = (value = ROL(value)) & 1;
|
|
break;
|
|
|
|
/* rrc .----------------.
|
|
| .---------. | .----.
|
|
'-->| 7 --> 0 |----->| CF |
|
|
'---------' '---*/
|
|
case 1:
|
|
cf = value & 1;
|
|
value = ROR(value);
|
|
break;
|
|
|
|
/* rl .-------------------------.
|
|
| .----. .---------. |
|
|
'--| CF |<--| 7 <-- 0 |<--'
|
|
'----' '--------*/
|
|
case 2:
|
|
cf = value >> 7;
|
|
value = (zuint8)((value << 1) | F_C);
|
|
break;
|
|
|
|
/* rr .-------------------------.
|
|
| .---------. .----. |
|
|
'-->| 7 --> 0 |-->| CF |--'
|
|
'---------' '---*/
|
|
case 3:
|
|
cf = value & 1;
|
|
value = (zuint8)((value >> 1) | ((zuint8)F_C << 7));
|
|
break;
|
|
|
|
/* sla .----. .---------.
|
|
| CF |<--| 7 <-- 0 |<-- 0
|
|
'----' '--------*/
|
|
case 4:
|
|
cf = value >> 7;
|
|
value <<= 1;
|
|
break;
|
|
|
|
/* sra .---------. .----.
|
|
.-->| 7 --> 0 |-->| CF |
|
|
| '-|-------' '----'
|
|
'----*/
|
|
case 5:
|
|
cf = value & 1;
|
|
value = (zuint8)((value & 128) | (value >> 1));
|
|
break;
|
|
|
|
/* sll .----. .---------.
|
|
| CF |<--| 7 <-- 0 |<-- 1
|
|
'----' '--------*/
|
|
case 6:
|
|
cf = value >> 7;
|
|
value = (zuint8)((value << 1) | 1);
|
|
break;
|
|
|
|
/* srl .---------. .----.
|
|
0 -->| 7 --> 0 |-->| CF |
|
|
'---------' '---*/
|
|
case 7:
|
|
cf = value & 1;
|
|
value >>= 1;
|
|
break;
|
|
|
|
/* Uncoment to avoid compiler warnings */
|
|
/*default: cf = 0;*/
|
|
}
|
|
|
|
FLAGS = (zuint8)( /* HF, NF = 0 */
|
|
(value & SYXF) | /* SF = sign; YF = Y; XF = X */
|
|
ZF_ZERO(value) | /* ZF = zero */
|
|
PF_PARITY(value) | /* PF = parity */
|
|
cf); /* CF already computed */
|
|
|
|
return value;
|
|
}
|
|
|
|
|
|
/* MARK: - Bit Set and Reset Operations */
|
|
|
|
/*---------. .---------.
|
|
| 76543210 | | M |
|
|
|----------| |---------|
|
|
| _m______ | | 0 = res |
|
|
'----------' | 1 = set |
|
|
'--------*/
|
|
|
|
static inline zuint8 m(Z80 *self, zuint8 offset, zuint8 value)
|
|
{
|
|
zuint8 t;
|
|
|
|
Q_0
|
|
|
|
return (zuint8)(((t = DATA[offset]) & 64)
|
|
? value | (1U << ((t >> 3) & 7))
|
|
: value & ~(1U << ((t >> 3) & 7)));
|
|
}
|
|
|
|
|
|
/* MARK: - Function Shortcuts and Reusable Code */
|
|
|
|
#define INSN(name) static zuint8 name(Z80 *self)
|
|
#define N(offset) ((DATA[offset] >> 3) & 7)
|
|
#define Z(mask) zzz(self, mask)
|
|
#define U0(value) uuu(self, 0, value)
|
|
#define U1(value) uuu(self, 1, value)
|
|
#define V0(value) vvv(self, 0, value)
|
|
#define V1(value) vvv(self, 1, value)
|
|
#define G1(value) ggg(self, 1, value)
|
|
#define G3(value) ggg(self, 3, value)
|
|
#define M1(value) m (self, 1, value)
|
|
#define M3(value) m (self, 3, value)
|
|
#define PUSH(value) WRITE_16B(SP -= 2, value)
|
|
#define R_ALL ((R & 127) | (R7 & 128))
|
|
#define RET MEMPTR = PC = READ_16(SP); SP += 2
|
|
#define FETCH_XY_EA(address) MEMPTR = (zuint16)(XY + (zsint8)FETCH(address))
|
|
#define IS_XY_PREFIX(opcode) ((opcode) & 0xDF) == 0xDD
|
|
#define EX(a, b) t = a; a = b; b = t
|
|
|
|
|
|
#define LD_A_IR(rhs) \
|
|
A = rhs; \
|
|
\
|
|
FLAGS = (zuint8)( /* HF, NF = 0 */ \
|
|
A_SYX | /* SF = sign; YF = Y; XF = X */ \
|
|
ZF_ZERO(A) | /* ZF = zero */ \
|
|
(IFF2 << 2) | /* PF = IFF2 */ \
|
|
F_C); /* CF unchanged */ \
|
|
\
|
|
PC += 2; \
|
|
return 9
|
|
|
|
|
|
#define LD_VWORD_COMMON(insn_size) \
|
|
zuint16 n; \
|
|
\
|
|
Q_0 \
|
|
MEMPTR = (n = FETCH_16((PC += insn_size) - 2)) + 1
|
|
|
|
|
|
#define EX_VSP(rhs, pc_increment) \
|
|
zuint16 sp, t = rhs; \
|
|
\
|
|
Q_0 \
|
|
pc_increment; \
|
|
rhs = MEMPTR = READ_16(sp = SP); \
|
|
WRITE_16B(sp, t); \
|
|
return 19
|
|
|
|
|
|
#define LDX(operator) \
|
|
zuint8 t = READ(HL operator); \
|
|
\
|
|
WRITE(DE operator, t); \
|
|
t += A; \
|
|
\
|
|
FLAGS = (zuint8)( /* HF, NF = 0 */ \
|
|
F_SZC | /* SF, ZF, CF unchanged */ \
|
|
((t & 2) << 4) | /* YF = (A + [HLi]).1 */ \
|
|
(t & XF) | /* XF = (A + [HLi]).3 */ \
|
|
(!!(--BC) << 2)); /* PF = !!BCo */ \
|
|
\
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define LDXR(operator) \
|
|
zuint8 t = READ(HL operator); \
|
|
\
|
|
WRITE(DE operator, t); \
|
|
t += A; \
|
|
\
|
|
if (--BC) \
|
|
{ /* HF, NF = 0 */ \
|
|
FLAGS = F_SZC | /* SF, ZF, CF unchanged */ \
|
|
((PC >> 8) & YXF) | /* YF = PCi.13; XF = PCi.11 */ \
|
|
PF; /* PF = 1 */ \
|
|
\
|
|
MEMPTR = PC + 1; \
|
|
return 21; \
|
|
} \
|
|
\
|
|
FLAGS = (zuint8)( /* HF, PF, NF = 0 */ \
|
|
F_SZC | /* SF, ZF, CF unchanged */ \
|
|
((t & 2) << 4) | /* YF = (A + [HLi]).1 */ \
|
|
(t & XF)); /* XF = (A + [HLi]).3 */ \
|
|
\
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define CPX(operator) \
|
|
zuint8 n = READ(HL operator); \
|
|
zuint8 t0 = A - n; \
|
|
zuint8 hf = (A ^ n ^ t0) & HF; \
|
|
zuint8 t1 = t0 - (hf >> 4); \
|
|
\
|
|
FLAGS = (zuint8)( \
|
|
(t0 & SF) | /* SF = sign */ \
|
|
ZF_ZERO(t0) | /* ZF = zero */ \
|
|
hf | /* HF = half-borrow */ \
|
|
((t1 & 2) << 4) | /* YF = (A - [HLi] - HFo).1 */ \
|
|
(t1 & XF) | /* XF = (A - [HLi] - HFo).3 */ \
|
|
(!!(--BC) << 2) | /* PF = !!BCo */ \
|
|
NF | /* NF = 1 */ \
|
|
F_C); /* CF unchanged */ \
|
|
\
|
|
MEMPTR operator; \
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define CPXR(operator) \
|
|
zuint8 n = READ(HL operator); \
|
|
zuint8 t0 = A - n; \
|
|
zuint8 hf = (A ^ n ^ t0) & HF; \
|
|
zuint8 t1 = t0 - (hf >> 4); \
|
|
\
|
|
zuint8 f = (zuint8)( \
|
|
(t0 & SF) | /* SF = sign */ \
|
|
ZF_ZERO(t0) | /* ZF = zero */ \
|
|
hf | /* HF = half-borrow */ \
|
|
(!!(--BC) << 2) | /* PF = !!BCo */ \
|
|
NF | /* NF = 1 */ \
|
|
F_C); /* CF unchanged */ \
|
|
\
|
|
if (t0 && BC) \
|
|
{ \
|
|
/* YF = PCi.13; XF = PCi.11 */ \
|
|
FLAGS = f | ((PC >> 8) & YXF); \
|
|
MEMPTR = PC + 1; \
|
|
return 21; \
|
|
} \
|
|
\
|
|
FLAGS = (zuint8)( \
|
|
f | \
|
|
((t1 & 2) << 4) | /* YF = (A - [HLi] - HFo).1 */ \
|
|
(t1 & XF)); /* XF = (A - [HLi] - HFo).3 */ \
|
|
\
|
|
MEMPTR operator; \
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define ADD_16(lhs, rhs, pc_increment) \
|
|
zuint16 n = rhs; \
|
|
zuint32 t = lhs + n; \
|
|
\
|
|
FLAGS = (zuint8)( /* NF = 0 */ \
|
|
F_SZP | /* SF, ZF, PF unchanged */ \
|
|
((t >> 8) & YXF) | /* YF = high-Y; XF = high-X */ \
|
|
(((lhs ^ n ^ t) >> 8) & HF) | /* HF = high-half-carry */ \
|
|
((t >> 16) & 1)); /* CF = carry */ \
|
|
\
|
|
MEMPTR = lhs + 1; \
|
|
lhs = (zuint16)t; \
|
|
pc_increment; \
|
|
return 11
|
|
|
|
|
|
#define ADC_SBC_HL_SS(operator, pf_overflow_rhs, or_nf) \
|
|
zuint16 ss = SS1; \
|
|
zuint32 t = HL operator ss operator F_C; \
|
|
\
|
|
FLAGS = (zuint8)( \
|
|
((t >> 8) & SYXF) /* SF = sign; YF = high-Y; XF = high-X */ \
|
|
| (zuint8)ZF_ZERO((zuint16)t) /* ZF = zero */ \
|
|
/* HF = high-half-carry (adc), high-half-borrow (sbc) */ \
|
|
| (((HL ^ ss ^ t) >> 8) & HF) \
|
|
/* PF = overflow */ \
|
|
| PF_OVERFLOW(16, (zuint16)t, HL, pf_overflow_rhs) \
|
|
| ((t >> 16) & 1) /* CF = carry (adc), borrow (sbc) */ \
|
|
or_nf); /* NF = 0 (adc), 1 (sbc) */ \
|
|
\
|
|
MEMPTR = HL + 1; \
|
|
HL = (zuint16)t; \
|
|
PC += 2; \
|
|
return 15
|
|
|
|
|
|
/* rla .-------------------------. rra .-------------------------.
|
|
| .----. .----A----. | | .----A----. .----. |
|
|
'--| CF |<--| 7 <-- 0 |<--' '-->| 7 --> 0 |-->| CF |--'
|
|
'----' '---------' '---------' '---*/
|
|
|
|
#define RXA(a_to_cf, operator, fc_to_a) \
|
|
zuint8 cf = a_to_cf; \
|
|
\
|
|
A = (zuint8)((A operator 1) | fc_to_a); \
|
|
\
|
|
FLAGS = F_SZP | /* SF, ZF, PF unchanged */ \
|
|
A_YX | /* YF = Y; XF = X */ \
|
|
cf; /* CF = Ai.7 (rla), Ai.0 (rra) */ \
|
|
/* HF, NF = 0 */ \
|
|
PC++; \
|
|
return 4
|
|
|
|
|
|
/* rld .------------------------. rrd .------------------------.
|
|
| | | |
|
|
.--------|--. .-------------. | .--------V--. .-------------. |
|
|
| 7-4 | 3-0 |<--| 7-4 <-- 3-0 |<--' | 7-4 | 3-0 |-->| 7-4 --> 3-0 |---'
|
|
'-----A-----' '----(HL)-----' '-----A-----' '----(HL)----*/
|
|
|
|
#define RXD(vhl_to_vhl, a_to_vhl, vhl_to_a) \
|
|
zuint8 t = READ(HL); \
|
|
\
|
|
MEMPTR = HL + 1; \
|
|
WRITE(HL, (zuint8)((t vhl_to_vhl) | (A a_to_vhl))); \
|
|
A = (A & 0xF0) | (t vhl_to_a); \
|
|
\
|
|
FLAGS = (zuint8)( /* HF, NF = 0 */ \
|
|
A_SYX | /* SF = sign; YF = Y; XF = X */ \
|
|
ZF_ZERO(A) | /* ZF = zero */ \
|
|
PF_PARITY(A) | /* PF = parity */ \
|
|
F_C); /* CF unchanged */ \
|
|
\
|
|
PC += 2; \
|
|
return 18
|
|
|
|
|
|
#define DJNZ_JR_Z(condition, cycles_if_true, cycles_if_false) \
|
|
zsint8 offset; \
|
|
\
|
|
Q_0 \
|
|
offset = (zsint8)FETCH(PC + 1); /* Always read */ \
|
|
\
|
|
if (condition) \
|
|
{ \
|
|
MEMPTR = (PC += 2 + offset); \
|
|
return cycles_if_true; \
|
|
} \
|
|
\
|
|
PC += 2; \
|
|
return cycles_if_false
|
|
|
|
|
|
#define RETX(mnemonic) \
|
|
DATA[2] = IFF1; \
|
|
NOTIFY(mnemonic); \
|
|
Q_0 \
|
|
RET; \
|
|
if ((IFF1 = IFF2) && INT_LINE) REQUEST |= Z80_REQUEST_INT; \
|
|
return 14
|
|
|
|
|
|
#define IN_VC(set_lhs) \
|
|
zuint8 t; \
|
|
\
|
|
MEMPTR = BC + 1; \
|
|
t = IN(BC); \
|
|
\
|
|
FLAGS = (zuint8)( /* HF, NF = 0 */ \
|
|
(t & SYXF) | /* SF = sign; YF = Y; XF = X */ \
|
|
ZF_ZERO(t) | /* ZF = zero */ \
|
|
PF_PARITY(t) | /* PF = parity */ \
|
|
F_C); /* CF unchanged */ \
|
|
\
|
|
set_lhs \
|
|
PC += 2; \
|
|
return 12
|
|
|
|
|
|
#define INX_OUTX_COMMON(io) \
|
|
FLAGS = (zuint8)( \
|
|
(B & SYXF) | /* SF = Bo.7; YF = Bo.5; XF = Bo.3 */ \
|
|
ZF_ZERO(B) | /* ZF = !Bo */ \
|
|
PF_PARITY((t & 7) ^ B) | /* PF = ((T & 7) ^ Bo).parity */ \
|
|
((t > 255) ? HCF : 0) | /* HF, CF = T > 255 */ \
|
|
((io >> 6) & NF)); /* NF = IO.7 */ \
|
|
\
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define INX(hl_operator, memptr_operator) \
|
|
zuint8 io = IN(BC); \
|
|
zuint t = (zuint)io + (zuint8)(C memptr_operator 1); \
|
|
\
|
|
WRITE(HL hl_operator, io); \
|
|
MEMPTR = BC memptr_operator 1; \
|
|
B--; \
|
|
INX_OUTX_COMMON(io)
|
|
|
|
|
|
#define OUTX(hl_operator, memptr_operator) \
|
|
zuint8 io = READ(HL hl_operator); \
|
|
zuint t = (zuint)io + L; \
|
|
\
|
|
B--; \
|
|
MEMPTR = BC memptr_operator 1; \
|
|
OUT(BC, io); \
|
|
INX_OUTX_COMMON(io)
|
|
|
|
|
|
/*-----------------------------------------------------------------------------.
|
|
| Block instructions produce an extra M-cycle of 5 T-states to decrement PC if |
|
|
| the loop condition is met. In 2018, David Banks (AKA hoglet) discovered that |
|
|
| the Z80 CPU performs additional flag changes during this M-cycle and managed |
|
|
| to decipher the behaviors: All block instructions copy bits 13 and 11 of PCi |
|
|
| to YF and XF, respectively [1.1], but `inir`, `indr`, `otir` and `otdr` also |
|
|
| modify HF and PF in a very complicated way [1.2]. These latter two flags are |
|
|
| not commented here because the explanation would not be simpler than the |
|
|
| code itself, so please refer to David Banks' paper [2] for more information. |
|
|
| |
|
|
| David Banks' discoveries have been corroborated thanks to Peter Helcmanovsky |
|
|
| (AKA Ped7g), who wrote a test that covers most of the cases that can be |
|
|
| verified on a ZX Spectrum [3]. |
|
|
| |
|
|
| In 2022, rofl0r discovered that the instructions `otir` and `otdr` also set |
|
|
| MEMPTR to `PCi + 1` during the extra M-cycle [4]. However, this information |
|
|
| was not announced anywhere and went unnoticed by the emulation community |
|
|
| until 2023, when Manuel Sainz de Baranda y Goñi rediscovered the same |
|
|
| behaviour in all four I/O block instructions: `inir`, `indr`, `otir` and |
|
|
| `otdr` [5]. |
|
|
| |
|
|
| References: |
|
|
| 1. https://stardot.org.uk/forums/viewtopic.php?t=15464 |
|
|
| 1. https://stardot.org.uk/forums/viewtopic.php?p=211042#p211042 |
|
|
| 2. https://stardot.org.uk/forums/viewtopic.php?p=212021#p212021 |
|
|
| 2. Banks, David (2018-08-21). "Undocumented Z80 Flags" rev. 1.0. |
|
|
| * https://github.com/hoglet67/Z80Decoder/wiki/Undocumented-Flags |
|
|
| * https://stardot.org.uk/forums/download/file.php?id=39831 |
|
|
| 3. Helcmanovsky, Peter (2021/2022). "Z80 Block Flags Test". |
|
|
| * https://github.com/MrKWatkins/ZXSpectrumNextTests |
|
|
| 4. https://github.com/hoglet67/Z80Decoder/issues/2 |
|
|
| 5. https://spectrumcomputing.co.uk/forums/viewtopic.php?t=10555 |
|
|
'=============================================================================*/
|
|
|
|
#define INXR_OTXR_COMMON \
|
|
if (B) { \
|
|
FLAGS = (zuint8)( /* ZF = 0 */ \
|
|
(B & SF) | /* SF = Bo.7 */ \
|
|
(PCH & YXF) | /* YF = PCi.13; XF = PCi.11 */ \
|
|
nf | /* NF = IO.7 */ \
|
|
(hcf ? /* CF = T > 255 */ \
|
|
CF | \
|
|
(nf ? \
|
|
(!(B & 0xF) << 4) | \
|
|
PF_PARITY(p ^ ((B - 1) & 7)) \
|
|
: \
|
|
(((B & 0xF) == 0xF) << 4) | \
|
|
PF_PARITY(p ^ ((B + 1) & 7))) \
|
|
: PF_PARITY(p ^ (B & 7)))); \
|
|
\
|
|
MEMPTR = PC + 1; \
|
|
return 21; \
|
|
} \
|
|
\
|
|
FLAGS = ZF | /* ZF = 1; SF, YF, XF = 0 */ \
|
|
hcf | /* HF, CF = T > 255 */ \
|
|
PF_PARITY(p) | /* PF = ((T & 7) ^ Bo).parity */ \
|
|
nf /* NF = IO.7 */
|
|
|
|
|
|
#define INXR(hl_operator, memptr_operator) \
|
|
zuint8 io = IN(BC); \
|
|
zuint8 nf = (io >> 6) & NF; \
|
|
zuint t; \
|
|
zuint8 hcf, p; \
|
|
\
|
|
WRITE(HL hl_operator, io); \
|
|
t = (zuint)io + (zuint8)(MEMPTR = BC memptr_operator 1); \
|
|
hcf = (t > 255) ? HCF : 0; \
|
|
p = (t & 7) ^ --B; \
|
|
INXR_OTXR_COMMON; \
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define OTXR(hl_operator, memptr_operator) \
|
|
zuint8 io = READ(HL hl_operator); \
|
|
zuint8 nf = (io >> 6) & NF; \
|
|
zuint t = (zuint)io + L; \
|
|
zuint8 hcf = (t > 255) ? HCF : 0; \
|
|
zuint8 p = (t & 7) ^ --B; \
|
|
\
|
|
OUT(BC, io); \
|
|
INXR_OTXR_COMMON; \
|
|
MEMPTR = BC memptr_operator 1; \
|
|
PC += 2; \
|
|
return 16
|
|
|
|
|
|
#define EXIT_HALT \
|
|
HALT_LINE = 0; \
|
|
if (self->halt != Z_NULL) self->halt(CONTEXT, 0)
|
|
|
|
|
|
/* MARK: - Forward Declarations */
|
|
|
|
static Insn const insn_table [256];
|
|
static Insn const cb_insn_table [256];
|
|
static Insn const ed_insn_table [256];
|
|
static Insn const xy_insn_table [256];
|
|
static Insn const xy_cb_insn_table[256];
|
|
|
|
|
|
/* MARK: - Instructions: 8-Bit Load Group */
|
|
/*----------------------------------------------------------------------------.
|
|
| 0 1 2 3 Flags T-states |
|
|
| Assembly 76543210765432107654321076543210 SZYHXPNC 12345 |
|
|
| ------------------- -------------------------------- -------- -------- |
|
|
| ld J,K 01jjjkkk ........ 4:4 |
|
|
|* ld O,P <--XY-->01oooppp ........ 8:44 |
|
|
| ld J,BYTE 00jjj110<-BYTE-> ........ 7:43 |
|
|
|* ld O,BYTE <--XY-->00ooo110<-BYTE-> ........ 11:443 |
|
|
| ld J,(hl) 01jjj110 ........ 7:43 |
|
|
| ld J,(XY+OFFSET) <--XY-->01jjj110<OFFSET> ........ 19:44353 |
|
|
| ld (hl),K 01110kkk ........ 7:43 |
|
|
| ld (XY+OFFSET),K <--XY-->01110kkk<OFFSET> ........ 19:44353 |
|
|
| ld (hl),BYTE <--36--><-BYTE-> ........ 10:433 |
|
|
| ld (XY+OFFSET),BYTE <--XY--><--36--><OFFSET><-BYTE-> ........ 19:44353 |
|
|
| ld a,(bc) <--0A--> ........ 7:43 |
|
|
| ld a,(de) <--1A--> ........ 7:43 |
|
|
| ld a,(WORD) <--3A--><-----WORD-----> ........ 13:4333 |
|
|
| ld (bc),a <--02--> ........ 7:43 |
|
|
| ld (de),a <--12--> ........ 7:43 |
|
|
| ld (WORD),a <--32--><-----WORD-----> ........ 13:4333 |
|
|
| ld a,i <--ED--><--57--> szy0x*0. 9:45 |
|
|
| ld a,r <--ED--><--5F--> szy0x*0. 9:45 |
|
|
| ld i,a <--ED--><--47--> ........ 9:45 |
|
|
| ld r,a <--ED--><--4F--> ........ 9:45 |
|
|
|-----------------------------------------------------------------------------|
|
|
| (*) Undocumented instruction. |
|
|
'============================================================================*/
|
|
|
|
INSN(ld_J_K ) {Q_0 J0 = K0; PC++; return 4;}
|
|
INSN(ld_O_P ) {Q_0 O = P; PC += 2; return 4;}
|
|
INSN(ld_J_BYTE ) {Q_0 J0 = FETCH((PC += 2) - 1); return 7;}
|
|
INSN(ld_O_BYTE ) {Q_0 O = FETCH((PC += 3) - 1); return 7;}
|
|
INSN(ld_J_vhl ) {Q_0 J0 = READ(HL); PC++; return 7;}
|
|
INSN(ld_J_vXYpOFFSET) {Q_0 J1 = READ(FETCH_XY_EA((PC += 3) - 1)); return 15;}
|
|
INSN(ld_vhl_K ) {Q_0 PC++; WRITE(HL, K0); return 7;}
|
|
INSN(ld_vXYpOFFSET_K) {Q_0 WRITE(FETCH_XY_EA((PC += 3) - 1), K1); return 15;}
|
|
INSN(ld_vhl_BYTE ) {Q_0 WRITE(HL, FETCH((PC += 2) - 1)); return 10;}
|
|
INSN(ld_a_vbc ) {Q_0 MEMPTR = BC + 1; A = READ(BC); PC++; return 7;}
|
|
INSN(ld_a_vde ) {Q_0 MEMPTR = DE + 1; A = READ(DE); PC++; return 7;}
|
|
INSN(ld_a_vWORD ) {Q_0 MEMPTR = FETCH_16((PC += 3) - 2); A = READ(MEMPTR++); return 13;}
|
|
INSN(ld_vbc_a ) {Q_0 PC++; MEMPTRL = C + 1; WRITE(BC, MEMPTRH = A); return 7;}
|
|
INSN(ld_vde_a ) {Q_0 PC++; MEMPTRL = E + 1; WRITE(DE, MEMPTRH = A); return 7;}
|
|
INSN(ld_a_i ) {LD_A_IR(I); }
|
|
INSN(ld_a_r ) {LD_A_IR(R_ALL); }
|
|
INSN(ld_i_a ) {NOTIFY(ld_i_a); Q_0 I = A; PC += 2; return 9;}
|
|
INSN(ld_r_a ) {NOTIFY(ld_r_a); Q_0 R = R7 = A; PC += 2; return 9;}
|
|
|
|
|
|
INSN(ld_vXYpOFFSET_BYTE)
|
|
{
|
|
zuint16 ea;
|
|
|
|
Q_0
|
|
ea = FETCH_XY_EA((PC += 4) - 2);
|
|
WRITE(ea, FETCH(PC - 1));
|
|
return 15;
|
|
}
|
|
|
|
|
|
INSN(ld_vWORD_a)
|
|
{
|
|
zuint16 ea;
|
|
|
|
Q_0
|
|
MEMPTRL = (zuint8)((ea = FETCH_16((PC += 3) - 2)) + 1);
|
|
WRITE(ea, MEMPTRH = A);
|
|
return 13;
|
|
}
|
|
|
|
|
|
/* MARK: - Instructions: 16-Bit Load Group */
|
|
/*----------------------------------------------------------------------.
|
|
| 0 1 2 3 Flags T-states |
|
|
| Assembly 76543210765432107654321076543210 SZYHXPNC 123456 |
|
|
| ------------ -------------------------------- -------- --------- |
|
|
| ld SS,WORD 00ss0001<-----WORD-----> ........ 10:433 [1]
|
|
| ld XY,WORD <--XY--><--21--><-----WORD-----> ........ 14:4433 |
|
|
| ld hl,(WORD) <--2A--><-----WORD-----> ........ 16:43333 |
|
|
| ld SS,(WORD) <--ED-->01ss1011<-----WORD-----> ........ 20:443333 |
|
|
| ld XY,(WORD) <--XY--><--2A--><-----WORD-----> ........ 20:443333 |
|
|
| ld (WORD),hl <--22--><-----WORD-----> ........ 16:43333 |
|
|
| ld (WORD),SS <--ED-->01ss0011<-----WORD-----> ........ 20:443333 |
|
|
| ld (WORD),XY <--XY--><--22--><-----WORD-----> ........ 20:443333 |
|
|
| ld sp,hl <--F9--> ........ 6:6 |
|
|
| ld sp,XY <--XY--><--F9--> ........ 10:46 |
|
|
| push TT 11tt0101 ........ 11:533 |
|
|
| push XY <--XY--><--E5--> ........ 15:4533 |
|
|
| pop TT 11tt0001 ........ 10:433 |
|
|
| pop XY <--XY--><--E1--> ........ 14:4433 |
|
|
|-----------------------------------------------------------------------|
|
|
| 1. All versions of Zilog's "Z80 CPU User Manual" have a typo in the |
|
|
| M-cycles of the instruction. |
|
|
'======================================================================*/
|
|
|
|
INSN(ld_SS_WORD ) {Q_0 SS0 = FETCH_16((PC += 3) - 2); return 10;}
|
|
INSN(ld_XY_WORD ) {Q_0 XY = FETCH_16((PC += 4) - 2); return 10;}
|
|
INSN(ld_hl_vWORD) {LD_VWORD_COMMON(3); HL = READ_16(n); return 16;}
|
|
INSN(ld_SS_vWORD) {LD_VWORD_COMMON(4); SS1 = READ_16(n); return 20;}
|
|
INSN(ld_XY_vWORD) {LD_VWORD_COMMON(4); XY = READ_16(n); return 16;}
|
|
INSN(ld_vWORD_hl) {LD_VWORD_COMMON(3); WRITE_16F(n, HL ); return 16;}
|
|
INSN(ld_vWORD_SS) {LD_VWORD_COMMON(4); WRITE_16F(n, SS1); return 20;}
|
|
INSN(ld_vWORD_XY) {LD_VWORD_COMMON(4); WRITE_16F(n, XY ); return 16;}
|
|
INSN(ld_sp_hl ) {Q_0 SP = HL; PC++; return 6;}
|
|
INSN(ld_sp_XY ) {Q_0 SP = XY; PC += 2; return 6;}
|
|
INSN(push_TT ) {Q_0 PC++; PUSH(TT); return 11;}
|
|
INSN(push_XY ) {Q_0 PC += 2; PUSH(XY); return 11;}
|
|
INSN(pop_TT ) {Q_0 TT = READ_16(SP); SP += 2; PC++; return 10;}
|
|
INSN(pop_XY ) {Q_0 XY = READ_16(SP); SP += 2; PC += 2; return 10;}
|
|
|
|
|
|
/* MARK: - Instructions: Exchange, Block Transfer and Search Groups */
|
|
/*-------------------------------------------------------------.
|
|
| 0 1 Flags T-states |
|
|
| Assembly 7654321076543210 SZYHXPNC !0 123456 =0 1234 |
|
|
| ---------- ---------------- -------- ------------------ |
|
|
| ex de,hl <--EB--> ........ 4:4 |
|
|
| ex af,af' <--08--> ........ 4:4 |
|
|
| exx <--D9--> ........ 4:4 |
|
|
| ex (sp),hl <--E3--> ........ 19:43435 |
|
|
| ex (sp),XY <--XY--><--E3--> ........ 23:443435 |
|
|
| ldi <--ED--><--A0--> ..*0**0. 16:4435 |
|
|
| ldir <--ED--><--B0--> ..*0*00. 21:44355 16:4435 |
|
|
| ldd <--ED--><--A8--> ..*0**0. 16:4435 |
|
|
| lddr <--ED--><--B8--> ..*0*00. 21:44355 16:4435 |
|
|
| cpi <--ED--><--A1--> sz*b**1. 16:4435 |
|
|
| cpir <--ED--><--B1--> sz*b**1. 21:44355 16:4435 |
|
|
| cpd <--ED--><--A9--> sz*b**1. 16:4435 |
|
|
| cpdr <--ED--><--B9--> sz*b**1. 21:44355 16:4435 |
|
|
'=============================================================*/
|
|
|
|
INSN(ex_de_hl ) {zuint16 t; Q_0 EX(DE, HL ); PC++; return 4;}
|
|
INSN(ex_af_af_) {zuint16 t; Q_0 EX(AF, AF_); PC++; return 4;}
|
|
INSN(exx ) {zuint16 t; Q_0 EX(BC, BC_); EX(DE, DE_); EX(HL, HL_); PC++; return 4;}
|
|
INSN(ex_vsp_hl) {EX_VSP(HL, PC++ ); }
|
|
INSN(ex_vsp_XY) {EX_VSP(XY, PC += 2); }
|
|
INSN(ldi ) {LDX (++); }
|
|
INSN(ldir ) {LDXR(++); }
|
|
INSN(ldd ) {LDX (--); }
|
|
INSN(lddr ) {LDXR(--); }
|
|
INSN(cpi ) {CPX (++); }
|
|
INSN(cpir ) {CPXR(++); }
|
|
INSN(cpd ) {CPX (--); }
|
|
INSN(cpdr ) {CPXR(--); }
|
|
|
|
|
|
/* MARK: - Instructions: 8-Bit Arithmetic and Logical Group */
|
|
/*-------------------------------------------------------------------.
|
|
| 0 1 2 Flags T-states |
|
|
| Assembly 765432107654321076543210 SZYHXPNC 123456 |
|
|
| ----------------- ------------------------ -------- --------- |
|
|
| U [a,]K 10uuukkk sz|||||| 4:4 |
|
|
|* U [a,]P <--XY-->10uuuppp sz|||||| 8:44 |
|
|
| U [a,]BYTE 11uuu110<-BYTE-> sz|||||| 7:43 |
|
|
| U [a,](hl) 10uuu110 sz|||||| 7:43 |
|
|
| U [a,](XY+OFFSET) <--XY-->10uuu110<OFFSET> sz|||||| 19:44353 |
|
|
| V J 00jjj10v szy|xv|. 4:4 |
|
|
|* V O <--XY-->00ooo10v szy|xv|. 8:44 |
|
|
| V (hl) 0011010v szy|xv|. 11:443 |
|
|
| V (XY+OFFSET) <--XY-->0011010v<OFFSET> szy|xv|. 23:443543 |
|
|
|--------------------------------------------------------------------|
|
|
| (*) Undocumented instruction. |
|
|
| (|) The flag is explained in table U/V. |
|
|
'===================================================================*/
|
|
|
|
INSN(U_a_K ) {U0(K0); PC++; return 4;}
|
|
INSN(U_a_P ) {U1(P ); PC += 2; return 4;}
|
|
INSN(U_a_BYTE ) {U0(FETCH((PC += 2) - 1)); return 7;}
|
|
INSN(U_a_vhl ) {U0(READ(HL)); PC++; return 7;}
|
|
INSN(U_a_vXYpOFFSET) {U1(READ(FETCH_XY_EA((PC += 3) - 1))); return 15;}
|
|
INSN(V_J ) {zuint8 *j = &J0; *j = V0(*j); PC++; return 4;}
|
|
INSN(V_O ) {zuint8 *o = &O; *o = V1(*o); PC += 2; return 4;}
|
|
INSN(V_vhl ) {PC++; WRITE(HL, V0(READ(HL))); return 11;}
|
|
INSN(V_vXYpOFFSET ) {zuint16 ea = FETCH_XY_EA((PC += 3) - 1); WRITE(ea, V1(READ(ea))); return 19;}
|
|
|
|
|
|
/* MARK: - Instructions: General-Purpose Arithmetic and CPU Control Groups */
|
|
/*-------------------------------------------------.
|
|
| 0 1 Flags T-states |
|
|
| Assembly 7654321076543210 SZYHXPNC 12 |
|
|
| -------- ---------------- -------- -------- |
|
|
| daa <--27--> szy^xp.* 4:4 |
|
|
| cpl <--2F--> ..y1x.1. 4:4 |
|
|
|- neg <--ED-->01***100 szybxv1b 8:44 |
|
|
| ccf <--3F--> ..***.0~ 4:4 |
|
|
| scf <--37--> ..*0*.01 4:4 |
|
|
| nop <--00--> ........ 4:4 |
|
|
| halt <--76--> ........ 4:4 |
|
|
| di <--F3--> ........ 4:4 |
|
|
| ei <--FB--> ........ 4:4 |
|
|
|- im 0 <--ED-->01*0*110 ........ 8:44 |
|
|
|- im 1 <--ED-->01*10110 ........ 8:44 |
|
|
|- im 2 <--ED-->01*11110 ........ 8:44 |
|
|
|--------------------------------------------------|
|
|
| (-) The instruction has undocumented opcodes. |
|
|
'=================================================*/
|
|
|
|
INSN(nop ) {Q_0 PC++; return 4;}
|
|
INSN(im_0) {Q_0 IM = 0; PC += 2; return 8;}
|
|
INSN(im_1) {Q_0 IM = 1; PC += 2; return 8;}
|
|
INSN(im_2) {Q_0 IM = 2; PC += 2; return 8;}
|
|
|
|
|
|
INSN(daa)
|
|
{
|
|
# ifdef Z80_WITH_PRECOMPUTED_DAA
|
|
zuint16 afi = AF;
|
|
|
|
# ifdef Z80_WITH_Q
|
|
Q = (zuint8)
|
|
# endif
|
|
(AF = daa_af_table[
|
|
( afi >> 8) |
|
|
((afi & (CF | NF)) << 8) |
|
|
((afi & HF) << 6)]);
|
|
# else
|
|
zuint8 cf = A > 0x99, t = ((F & HF) || (A & 0xF) > 9) ? 6 : 0;
|
|
|
|
if (F_C || cf) t |= 0x60;
|
|
t = (F & NF) ? A - t : A + t;
|
|
|
|
FLAGS = (zuint8)(
|
|
(F & (NF | CF)) | /* NF unchanged; CF dominant */
|
|
(t & SYXF) | /* SF = sign; YF = Y; XF = X */
|
|
ZF_ZERO(t) | /* ZF = zero */
|
|
((A ^ t) & HF) | /* HF = Ai.4 != Ao.4 */
|
|
PF_PARITY(t) | /* PF = parity */
|
|
cf); /* CF |= 1 (if BCD carry) */
|
|
|
|
A = t;
|
|
# endif
|
|
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
|
|
INSN(cpl)
|
|
{
|
|
FLAGS = F_SZPC | /* SF, ZF, PF, CF unchanged */
|
|
((A = (zuint8)~A) & YXF) | /* YF = Y; XF = X */
|
|
HF | NF; /* HF, NF = 1 */
|
|
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
|
|
INSN(neg)
|
|
{
|
|
zuint8 t = (zuint8)-A;
|
|
|
|
FLAGS = (zuint8)(
|
|
(t & SYXF) | /* SF = sign; YF = Y; XF = X */
|
|
ZF_ZERO(t) | /* ZF = zero */
|
|
((A ^ t) & HF) | /* HF = half-borrow */
|
|
((t == 128) << 2) | /* PF = overflow */
|
|
NF | /* NF = 1 */
|
|
!!A); /* CF = borrow (not 0) */
|
|
|
|
A = t;
|
|
PC += 2;
|
|
return 8;
|
|
}
|
|
|
|
|
|
/*---------------------------------------------------------------------------.
|
|
| `ccf` and `scf` are the only instructions in which Q affects the flags. |
|
|
| Patrik Rak cracked the behavior of YF and XF in 2012, confirming that they |
|
|
| are taken, respectively, from bits 5 and 3 of the result of `(Q ^ F) | A` |
|
|
| [1, 2]. This applies to all Zilog Z80 models, both NMOS and CMOS. In 2018, |
|
|
| David Banks (AKA hoglet) discovered that at least some ST CMOS models do |
|
|
| not set XF according to this formula and instead take this flag from bit 3 |
|
|
| of A, whereas NEC NMOS models take both flags from A [3]. |
|
|
| |
|
|
| References: |
|
|
| 1. https://worldofspectrum.org/forums/discussion/20345 |
|
|
| 2. https://worldofspectrum.org/forums/discussion/41704 |
|
|
| 3. Banks, David (2018-08-21). "Undocumented Z80 Flags" rev. 1.0. |
|
|
| * https://github.com/hoglet67/Z80Decoder/wiki/Undocumented-Flags |
|
|
| * https://stardot.org.uk/forums/download/file.php?id=39831 |
|
|
'===========================================================================*/
|
|
|
|
INSN(ccf)
|
|
{
|
|
FLAGS = (zuint8)(
|
|
(F_SZPC ^ CF) | /* SF, ZF, PF unchanged; CF = ~CFi */
|
|
/* Zilog: YF = A.5 | (YFi ^ YQi); XF = A.3 | (XFi ^ XQi) */
|
|
/* ST CMOS: YF = A.5 | (YFi ^ YQi); XF = A.3 */
|
|
/* NEC NMOS: YF = A.5; XF = A.3 */
|
|
# ifdef Z80_WITH_Q
|
|
((((F ^ Q) & OPTIONS) | A) & YXF) |
|
|
# else
|
|
(A & YXF) |
|
|
# endif
|
|
(F_C << 4)); /* HF = CFi */
|
|
/* NF = 0 */
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
|
|
INSN(scf)
|
|
{
|
|
FLAGS = F_SZP | /* SF, ZF, PF unchanged */
|
|
/* Zilog: YF = A.5 | (YFi ^ YQi); XF = A.3 | (XFi ^ XQi) */
|
|
/* ST CMOS: YF = A.5 | (YFi ^ YQi); XF = A.3 */
|
|
/* NEC NMOS: YF = A.5; XF = A.3 */
|
|
# ifdef Z80_WITH_Q
|
|
((((F ^ Q) & OPTIONS) | A) & YXF) |
|
|
# else
|
|
(A & YXF) |
|
|
# endif
|
|
CF; /* CF = 1 */
|
|
/* HF, NF = 0 */
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------.
|
|
| The `halt` instruction enables the HALT state after PC is incremented during |
|
|
| the opcode fetch. The CPU neither decrements nor avoids incrementing PC "so |
|
|
| that the instruction is re-executed" as Sean Young writes in section 5.4 of |
|
|
| "The Undocumented Z80 Documented". During the HALT state, the CPU repeatedly |
|
|
| executes an internal NOP operation. Each NOP consists of 1 M1 cycle of 4 |
|
|
| T-states that fetches (and disregards) the next opcode after `halt` without |
|
|
| incrementing PC. This opcode is read again and again until an exit condition |
|
|
| occurs (i.e., INT, NMI or RESET). |
|
|
| |
|
|
| This was first documented by Tony Brewer in 2014, and was later verified on |
|
|
| real hardware with the HALT2INT test written by Mark Woodmass (AKA Woody) in |
|
|
| 2021. |
|
|
| |
|
|
| References: |
|
|
| * Brewer, Tony (2014-12). "Z80 Special Reset". |
|
|
| * http://primrosebank.net/computers/z80/z80_special_reset.htm |
|
|
| * https://stardot.org.uk/forums/viewtopic.php?p=357136#p357136 |
|
|
'=============================================================================*/
|
|
|
|
INSN(halt)
|
|
{
|
|
if (!HALT_LINE)
|
|
{
|
|
if (!RESUME)
|
|
{
|
|
Q_0
|
|
PC++;
|
|
|
|
if ((self->cycles += 4) >= self->cycle_limit)
|
|
{
|
|
RESUME = Z80_RESUME_HALT;
|
|
return 0;
|
|
}
|
|
|
|
if (REQUEST) return 0;
|
|
RESUME = Z80_RESUME_HALT;
|
|
}
|
|
|
|
HALT_LINE = 1;
|
|
|
|
if (self->halt != Z_NULL)
|
|
{
|
|
self->halt(CONTEXT, 1);
|
|
if (self->cycles >= self->cycle_limit) return 0;
|
|
}
|
|
}
|
|
|
|
if (self->nop == Z_NULL || (OPTIONS & Z80_OPTION_HALT_SKIP))
|
|
{
|
|
zusize nop_cycles = self->cycle_limit - self->cycles;
|
|
|
|
nop_cycles += (4 - (nop_cycles & 3)) & 3;
|
|
R += (zuint8)(nop_cycles >> 2);
|
|
self->cycles += nop_cycles;
|
|
}
|
|
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
else {
|
|
zuint8 opcode;
|
|
|
|
do {
|
|
# ifdef Z80_WITH_RESET_SIGNAL
|
|
if (self->reset_signal != Z_NULL && (*self->reset_signal & self->reset_signal_mask))
|
|
return 0;
|
|
# endif
|
|
|
|
R++;
|
|
opcode = self->nop(CONTEXT, PC);
|
|
self->cycles += 4;
|
|
|
|
if (REQUEST)
|
|
{
|
|
RESUME = 0;
|
|
|
|
if (REQUEST & Z80_REQUEST_SPECIAL_RESET)
|
|
{
|
|
HALT_LINE = 0;
|
|
|
|
if (self->halt != Z_NULL)
|
|
self->halt(CONTEXT, Z80_HALT_EXIT_EARLY);
|
|
|
|
if ((DATA[0] = opcode) != Z80_HALT)
|
|
{
|
|
self->cycles -= 4;
|
|
PC--;
|
|
return insn_table[opcode](self);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
while (self->cycles < self->cycle_limit);
|
|
|
|
DATA[2] = opcode;
|
|
}
|
|
|
|
# else
|
|
else do {
|
|
# ifdef Z80_WITH_RESET_SIGNAL
|
|
if (self->reset_signal != Z_NULL && (*self->reset_signal & self->reset_signal_mask))
|
|
return 0;
|
|
# endif
|
|
|
|
R++;
|
|
(void)self->nop(CONTEXT, PC);
|
|
self->cycles += 4;
|
|
|
|
if (REQUEST)
|
|
{
|
|
RESUME = 0;
|
|
return 0;
|
|
}
|
|
}
|
|
while (self->cycles < self->cycle_limit);
|
|
# endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
INSN(di)
|
|
{
|
|
Q_0
|
|
IFF1 = IFF2 = 0;
|
|
REQUEST &= ~(zuint8)Z80_REQUEST_INT;
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
|
|
INSN(ei)
|
|
{
|
|
Q_0
|
|
IFF1 = IFF2 = 1;
|
|
if (INT_LINE) REQUEST |= Z80_REQUEST_INT;
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
|
|
/* MARK: - Instructions: 16-Bit Arithmetic Group */
|
|
/*--------------------------------------------------.
|
|
| 0 1 Flags T-states |
|
|
| Assembly 7654321076543210 SZYHXPNC 1234 |
|
|
| --------- ---------------- -------- -------- |
|
|
| add hl,SS 00ss1001 ..YCX.0c 11:443 |
|
|
| adc hl,SS <--ED-->01ss1010 szYCXv0c 15:4443 |
|
|
| sbc hl,SS <--ED-->01ss0010 szYBXv1b 15:4443 |
|
|
| add XY,WW <--XY-->00ww1001 ..YCX.0c 15:4443 |
|
|
| inc SS 00ss0011 ........ 6:6 |
|
|
| inc XY <--XY--><--23--> ........ 10:46 |
|
|
| dec SS 00ss1011 ........ 6:6 |
|
|
| dec XY <--XY--><--2B--> ........ 10:46 |
|
|
'==================================================*/
|
|
|
|
INSN(add_hl_SS) {ADD_16(HL, SS0, PC++); }
|
|
INSN(adc_hl_SS) {ADC_SBC_HL_SS(+, ~ss, Z_EMPTY);}
|
|
INSN(sbc_hl_SS) {ADC_SBC_HL_SS(-, ss, | NF );}
|
|
INSN(add_XY_WW) {ADD_16(XY, WW, PC += 2); }
|
|
INSN(inc_SS ) {Q_0 (SS0)++; PC++; return 6;}
|
|
INSN(inc_XY ) {Q_0 XY++; PC += 2; return 6;}
|
|
INSN(dec_SS ) {Q_0 (SS0)--; PC++; return 6;}
|
|
INSN(dec_XY ) {Q_0 XY--; PC += 2; return 6;}
|
|
|
|
|
|
/* MARK: - Instructions: Rotate and Shift Group */
|
|
/*-------------------------------------------------------------------------.
|
|
| 0 1 2 3 Flags T-states |
|
|
| Assembly 76543210765432107654321076543210 SZYHXPNC 123456 |
|
|
| --------------- -------------------------------- -------- --------- |
|
|
| rlca <--07--> ..y0x.0= 4:4 |
|
|
| rla <--17--> ..y0x.0= 4:4 |
|
|
| rrca <--0F--> ..y0x.0= 4:4 |
|
|
| rra <--1F--> ..y0x.0= 4:4 |
|
|
|- G K <--CB-->00gggkkk szy0xp0= 8:44 |
|
|
|- G (hl) <--CB-->00ggg110 szy0xp0= 15:4443 |
|
|
|- G (XY+OFFSET) <--XY--><--CB--><OFFSET>00ggg110 szy0xp0= 23:443543 |
|
|
|* G (XY+OFFSET),K <--XY--><--CB--><OFFSET>00gggkkk szy0xp0= 23:443543 |
|
|
| rld <--ED--><--6F--> szy0xp0. 18:44343 |
|
|
| rrd <--ED--><--67--> szy0xp0. 18:44343 |
|
|
|--------------------------------------------------------------------------|
|
|
| (-) The instruction has undocumented [pseudo-]opcodes. |
|
|
| (*) Undocumented instruction. |
|
|
'=========================================================================*/
|
|
|
|
INSN(rlca ) {A = ROL(A); FLAGS = F_SZP | (A & YXCF); PC++; return 4;}
|
|
INSN(rla ) {RXA(A >> 7, <<, F_C); }
|
|
INSN(rrca ) {A = ROR(A); FLAGS = F_SZP | A_YX | (A >> 7); PC++; return 4;}
|
|
INSN(rra ) {RXA(A & 1, >>, (F << 7)); }
|
|
INSN(G_K ) {zuint8 *k = &K1; *k = G1(*k); return 8;}
|
|
INSN(G_vhl ) {WRITE(HL, G1(READ(HL))); return 15;}
|
|
INSN(G_vXYpOFFSET ) {zuint16 ea = MEMPTR; WRITE(ea, G3(READ(ea))); return 19;}
|
|
INSN(G_vXYpOFFSET_K) {zuint16 ea = MEMPTR; WRITE(ea, K3 = G3(READ(ea))); return 19;}
|
|
INSN(rld ) {RXD(<< 4, & 0xF, >> 4); }
|
|
INSN(rrd ) {RXD(>> 4, << 4, & 0xF); }
|
|
|
|
|
|
/* MARK: - Instructions: Bit Set, Reset and Test Group */
|
|
/*---------------------------------------------------------------------------.
|
|
| 0 1 2 3 Flags T-states |
|
|
| Assembly 76543210765432107654321076543210 SZYHXPNC 123456 |
|
|
| ----------------- -------------------------------- -------- --------- |
|
|
| bit N,K <--CB-->01nnnkkk sz*1*z0. 8:44 |
|
|
| bit N,(hl) <--CB-->01nnn110 sz*1*z0. 12:444 [1]
|
|
|- bit N,(XY+OFFSET) <--XY--><--CB--><OFFSET>01nnn*** sz*1*z0. 20:44354 |
|
|
| M N,K <--CB-->1mnnnkkk ........ 8:44 |
|
|
| M N,(hl) <--CB-->1mnnn110 ........ 15:4443 |
|
|
| M N,(XY+OFFSET) <--XY--><--CB--><OFFSET>1mnnn110 ........ 23:443543 |
|
|
|* M N,(XY+OFFSET),K <--XY--><--CB--><OFFSET>1mnnnkkk ........ 23:443543 |
|
|
|----------------------------------------------------------------------------|
|
|
| (-) The instruction has undocumented pseudo-opcodes. |
|
|
| (*) Undocumented instruction. |
|
|
|----------------------------------------------------------------------------|
|
|
| 1. All versions of Zilog's "Z80 CPU User Manual" have a typo in the |
|
|
| T-states of the instruction. |
|
|
'===========================================================================*/
|
|
|
|
INSN(M_N_K ) {zuint8 *k = &K1; *k = M1(*k); return 8;}
|
|
INSN(M_N_vhl ) {WRITE(HL, M1(READ(HL))); return 15;}
|
|
INSN(M_N_vXYpOFFSET ) {zuint16 ea = MEMPTR; WRITE(ea, M3(READ(ea))); return 19;}
|
|
INSN(M_N_vXYpOFFSET_K) {zuint16 ea = MEMPTR; WRITE(ea, K3 = M3(READ(ea))); return 19;}
|
|
|
|
|
|
INSN(bit_N_K)
|
|
{
|
|
zuint8 k = K1;
|
|
zuint8 t = k & (1U << N(1));
|
|
|
|
/*----------------------------------------------------------.
|
|
| In section 4.1 of "The Undocumented Z80 Documented" (all |
|
|
| versions), Sean Young says that YF and XF are taken from |
|
|
| the value resulting from the bit test operation, but this |
|
|
| seems not to be true. They are copies of bits 5 and 3 of |
|
|
| the register containing the value to be tested (K). |
|
|
'==========================================================*/
|
|
FLAGS = (t ? t & SF : ZPF) | /* SF = sign; ZF, PF = zero */
|
|
(k & YXF) | /* YF = K.5; XF = K.3 */
|
|
HF | /* HF = 1 */
|
|
F_C; /* CF unchanged */
|
|
/* NF = 0 */
|
|
return 8;
|
|
}
|
|
|
|
|
|
INSN(bit_N_vhl)
|
|
{
|
|
zuint8 t = READ(HL) & (1U << N(1));
|
|
|
|
/*----------------------------------------------------------------.
|
|
| This is the only instruction in which MEMPTR affects the flags. |
|
|
| YF and XF are taken, respectively, from bits 13 and 11 of this |
|
|
| internal register whose behavior was cracked in 2006 by boo_boo |
|
|
| and Vladimir Kladov. Official schematics refer to this register |
|
|
| as WZ, but this emulator uses the name "MEMPTR" to honor those |
|
|
| who cracked it. |
|
|
| |
|
|
| References: |
|
|
| * https://zxpress.ru/zxnet/zxnet.pc/5909 |
|
|
| * boo_boo; Kladov, Vladimir (2006-03-29). "MEMPTR, Esoteric |
|
|
| Register of the Zilog Z80 CPU". |
|
|
| * https://zx-pk.ru/showpost.php?p=43688 |
|
|
| * https://zx-pk.ru/attachment.php?attachmentid=2984 |
|
|
| * https://zx-pk.ru/showpost.php?p=43800 |
|
|
| * https://zx-pk.ru/attachment.php?attachmentid=2989 |
|
|
'================================================================*/
|
|
FLAGS = (t ? t & SF : ZPF) | /* SF = sign; ZF, PF = zero */
|
|
(MEMPTRH & YXF) | /* YF = MEMPTRH.5; XF = MEMPTRH.3 */
|
|
HF | /* HF = 1 */
|
|
F_C; /* CF unchanged */
|
|
/* NF = 0 */
|
|
return 12;
|
|
}
|
|
|
|
|
|
INSN(bit_N_vXYpOFFSET)
|
|
{
|
|
zuint8 t = READ(MEMPTR) & (1U << N(3));
|
|
|
|
FLAGS = (t ? t & SF : ZPF) | /* SF sign; ZF, PF = zero */
|
|
(MEMPTRH & YXF) | /* YF = EA.13; XF = EA.11 */
|
|
HF | /* HF = 1 */
|
|
F_C; /* CF unchanged */
|
|
/* NF = 0 */
|
|
return 16;
|
|
}
|
|
|
|
|
|
/* MARK: - Instructions: Jump Group */
|
|
/*----------------------------------------------------------------.
|
|
| 0 1 2 Flags T-states |
|
|
| Assembly 765432107654321076543210 SZYHXPNC Y 123 N 12 |
|
|
| ----------- ------------------------ -------- ------------ |
|
|
| jp WORD <--C3--><-----WORD-----> ........ 10:433 |
|
|
| jp Z,WORD 11zzz010<-----WORD-----> ........ 10:433 |
|
|
| jr OFFSET <--18--><OFFSET> ........ 12:435 |
|
|
| jr Z,OFFSET 001zz000<OFFSET> ........ 12:435 7:43 |
|
|
| jp (hl) <--E9--> ........ 4:4 |
|
|
| jp (XY) <--XY--><--E9--> ........ 8:44 |
|
|
| djnz OFFSET <--10--><OFFSET> ........ 13:535 8:53 |
|
|
'================================================================*/
|
|
|
|
INSN(jp_WORD ) {Q_0 MEMPTR = PC = FETCH_16(PC + 1); return 10;}
|
|
INSN(jp_Z_WORD ) {Q_0 MEMPTR = FETCH_16(PC + 1); PC = Z(7) ? MEMPTR : PC + 3; return 10;}
|
|
INSN(jr_OFFSET ) {Q_0 MEMPTR = (PC += 2 + (zsint8)FETCH(PC + 1)); return 12;}
|
|
INSN(jr_Z_OFFSET) {DJNZ_JR_Z(Z(3), 12, 7); }
|
|
INSN(jp_hl ) {Q_0 PC = HL; return 4;}
|
|
INSN(jp_XY ) {Q_0 PC = XY; return 4;}
|
|
INSN(djnz_OFFSET) {DJNZ_JR_Z(--B, 13, 8); }
|
|
|
|
|
|
/* MARK: - Instructions: Call and Return Group */
|
|
/*--------------------------------------------------------------------.
|
|
| 0 1 2 Flags T-states |
|
|
| Assembly 765432107654321076543210 SZYHXPNC Y 123 N 123 |
|
|
| ----------- ------------------------ -------- ---------------- |
|
|
| call WORD <--CD--><-----WORD-----> ........ 17:43433 |
|
|
| call Z,WORD 11zzz100<-----WORD-----> ........ 17:43433 10:433 |
|
|
| ret <--C9--> ........ 10:433 |
|
|
| ret Z 11zzz000 ........ 11:533 5:5 |
|
|
|- reti/retn <--ED-->01***101 ........ 14:4433 |
|
|
| rst N 11nnn111 ........ 11:533 |
|
|
|---------------------------------------------------------------------|
|
|
| (-) The instruction has undocumented opcodes. The `reti` mnemonic |
|
|
| is used to represent the ED4Dh opcode, which is recognized by |
|
|
| the Z80 CTC chip. All other opcodes are represented as `retn`. |
|
|
'====================================================================*/
|
|
|
|
INSN(call_WORD) {zuint16 pci = PC; Q_0 MEMPTR = PC = FETCH_16(pci + 1); PUSH(pci + 3); return 17;}
|
|
INSN(ret ) {Q_0 RET; return 10;}
|
|
INSN(ret_Z ) {Q_0 if (Z(7)) {RET; return 11;} PC++; return 5;}
|
|
INSN(reti ) {RETX(reti); }
|
|
INSN(retn ) {RETX(retn); }
|
|
INSN(rst_N ) {zuint16 pci = PC; Q_0 MEMPTR = PC = DATA[0] & 56; PUSH(pci + 1); return 11;}
|
|
|
|
|
|
INSN(call_Z_WORD)
|
|
{
|
|
zuint16 pci;
|
|
|
|
Q_0
|
|
MEMPTR = FETCH_16((pci = PC) + 1); /* Always read */
|
|
|
|
if (Z(7))
|
|
{
|
|
PC = MEMPTR;
|
|
PUSH(pci + 3);
|
|
return 17;
|
|
}
|
|
|
|
PC += 3;
|
|
return 10;
|
|
}
|
|
|
|
|
|
/* MARK: - Instructions: Input and Output Group */
|
|
/*--------------------------------------------------------------.
|
|
| 0 1 Flags T-states |
|
|
| Assembly 7654321076543210 SZYHXPNC !0 12345 =0 1234 |
|
|
| ------------ ---------------- -------- ----------------- |
|
|
| in a,(BYTE) <--DB--><-BYTE-> ........ 11:434 |
|
|
| in J,(c) <--ED-->01jjj000 szy0xp0. 12:444 |
|
|
|* in (c) <--ED--><--70--> szy0xp0. 12:444 |
|
|
| ini <--ED--><--A2--> ******** 16:4543 [1]
|
|
| inir <--ED--><--B2--> ******** 21:45435 16:4543 [1]
|
|
| ind <--ED--><--AA--> ******** 16:4543 [1]
|
|
| indr <--ED--><--BA--> ******** 21:45435 16:4543 [1]
|
|
| out (BYTE),a <--D3--><-BYTE-> ........ 11:434 |
|
|
| out (c),J <--ED-->01jjj001 ........ 12:444 |
|
|
|* out (c),0 <--ED--><--71--> ........ 12:444 |
|
|
| outi <--ED--><--A3--> ******** 16:4534 |
|
|
| otir <--ED--><--B3--> ******** 21:45345 16:4534 |
|
|
| outd <--ED--><--AB--> ******** 16:4534 |
|
|
| otdr <--ED--><--BB--> ******** 21:45345 16:4534 |
|
|
|---------------------------------------------------------------|
|
|
| (*) Undocumented instruction. |
|
|
|---------------------------------------------------------------|
|
|
| 1. All versions of Zilog's "Z80 CPU User Manual" have typos |
|
|
| in the T-states of the instruction. |
|
|
'==============================================================*/
|
|
|
|
INSN(in_J_vc ) {IN_VC(J1 = t;); }
|
|
INSN(in_vc ) {IN_VC(Z_EMPTY); }
|
|
INSN(ini ) {INX (++, +); }
|
|
INSN(inir ) {INXR(++, +); }
|
|
INSN(ind ) {INX (--, -); }
|
|
INSN(indr ) {INXR(--, -); }
|
|
INSN(out_vc_J) {Q_0 PC += 2; MEMPTR = BC + 1; OUT(BC, J1); return 12;}
|
|
INSN(outi ) {OUTX(++, +); }
|
|
INSN(otir ) {OTXR(++, +); }
|
|
INSN(outd ) {OUTX(--, -); }
|
|
INSN(otdr ) {OTXR(--, -); }
|
|
|
|
|
|
INSN(in_a_vBYTE)
|
|
{
|
|
zuint16 t;
|
|
|
|
Q_0
|
|
|
|
/*--------------------------------------------------------------------.
|
|
| In "MEMPTR, Esoteric Register of the Zilog Z80 CPU", boo_boo says |
|
|
| that MEMPTR is set to `((A << 8) | BYTE) + 1`. This causes a carry |
|
|
| from the LSbyte of the port number to MEMPTRH if BYTE is 255, which |
|
|
| differs from all other instructions where MEMPTRH is set to A, but |
|
|
| it has been verified on real hardware with the IN-MEMPTR test. |
|
|
'====================================================================*/
|
|
MEMPTR = (t = (zuint16)(((zuint16)A << 8) | FETCH((PC += 2) - 1))) + 1;
|
|
|
|
A = IN(t);
|
|
return 11;
|
|
}
|
|
|
|
|
|
INSN(out_vBYTE_a)
|
|
{
|
|
zuint8 t;
|
|
|
|
Q_0
|
|
MEMPTRL = (t = FETCH((PC += 2) - 1)) + 1;
|
|
MEMPTRH = A;
|
|
OUT((zuint16)(((zuint16)A << 8) | t), A);
|
|
return 11;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------.
|
|
| The `out (c),0` instruction behaves as `out (c),255` on the Zilog Z80 CMOS. |
|
|
| This was first discovered by Simon Cooke, who reported it on Usenet in 1996 |
|
|
| [1, 2]. Later, in 2004, Colin Piggot rediscovered it with his SAM Coupé when |
|
|
| running a demo for SCPDU 6, coincidentally written by Simon Cooke [1]. In |
|
|
| 2008, this was once again rediscovered by the MSX community [1, 3]. |
|
|
| |
|
|
| References: |
|
|
| 1. https://sinclair.wiki.zxnet.co.uk/wiki/Z80 |
|
|
| 2. https://groups.google.com/g/comp.os.cpm/c/HfSTFpaIkuU/m/KotvMWu3bZoJ |
|
|
| 3. https://msx.org/forum/development/msx-development/bug-z80-emulation-or-tr |
|
|
| -hw |
|
|
'=============================================================================*/
|
|
|
|
INSN(out_vc_0)
|
|
{
|
|
Q_0
|
|
PC += 2;
|
|
MEMPTR = BC + 1;
|
|
OUT(BC, (zuint8)0 - (OPTIONS & (zuint8)Z80_OPTION_OUT_VC_255));
|
|
return 12;
|
|
}
|
|
|
|
|
|
/* MARK: - Instructions: Optimizations */
|
|
|
|
INSN(nop_nop) {Q_0 PC += 2; return 4;}
|
|
|
|
|
|
/* MARK: - Instructions: Prefix Handling */
|
|
|
|
INSN(cb_prefix)
|
|
{
|
|
R++;
|
|
return cb_insn_table[DATA[1] = FETCH_OPCODE((PC += 2) - 1)](self);
|
|
}
|
|
|
|
|
|
INSN(ed_prefix)
|
|
{
|
|
R++;
|
|
return ed_insn_table[DATA[1] = FETCH_OPCODE(PC + 1)](self);
|
|
}
|
|
|
|
|
|
#define XY_PREFIX(index_register) \
|
|
zuint8 cycles; \
|
|
\
|
|
if ((self->cycles += 4) >= self->cycle_limit) \
|
|
{ \
|
|
RESUME = Z80_RESUME_XY; \
|
|
return 0; \
|
|
} \
|
|
\
|
|
R++; \
|
|
XY = index_register; \
|
|
cycles = xy_insn_table[DATA[1] = FETCH_OPCODE(PC + 1)](self); \
|
|
index_register = XY; \
|
|
return cycles;
|
|
|
|
|
|
INSN(dd_prefix) {XY_PREFIX(IX)}
|
|
INSN(fd_prefix) {XY_PREFIX(IY)}
|
|
|
|
|
|
/*-----------------------------------------------------------------------.
|
|
| Instructions with the two-byte prefix DDCBh or FDCBh increment R by 2, |
|
|
| as only the prefix is fetched by opcode fetch operations (M1 cycles). |
|
|
| The remaining two bytes are fetched by normal memory read operations. |
|
|
'=======================================================================*/
|
|
|
|
INSN(xy_cb_prefix)
|
|
{
|
|
FETCH_XY_EA((PC += 4) - 2);
|
|
return xy_cb_insn_table[DATA[3] = FETCH(PC - 1)](self);
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------.
|
|
| In a sequence of DDh and/or FDh prefixes, it is the last one that counts, as |
|
|
| each prefix overrides the previous one. No matter how long the sequence is, |
|
|
| interrupts can only be responded to after executing the final instruction |
|
|
| once all the prefixes have been fetched. Each prefix takes 4 T-states. |
|
|
'=============================================================================*/
|
|
|
|
INSN(xy_xy)
|
|
{
|
|
ZInt16 *xy;
|
|
zuint16 t;
|
|
zuint8 cycles;
|
|
|
|
do {
|
|
PC++;
|
|
DATA[0] = DATA[1];
|
|
|
|
if ((self->cycles += 4) >= self->cycle_limit)
|
|
{
|
|
RESUME = Z80_RESUME_XY;
|
|
return 0;
|
|
}
|
|
|
|
R++;
|
|
}
|
|
while (IS_XY_PREFIX(DATA[1] = FETCH_OPCODE(PC + 1)));
|
|
|
|
t = XY;
|
|
XY = (xy = &self->ix_iy[(DATA[0] >> 5) & 1])->uint16_value;
|
|
cycles = xy_insn_table[DATA[1]](self);
|
|
xy->uint16_value = XY;
|
|
XY = t;
|
|
return cycles;
|
|
}
|
|
|
|
|
|
/* MARK: - Instructions: Illegal */
|
|
|
|
/*------------------------------------------------------------------------.
|
|
| Illegal opcodes prefixed with EDh are ignored by the CPU. Functionally, |
|
|
| these instructions are equivalent to two consecutive `nop` instructions |
|
|
| and take a total of 8 T-states. |
|
|
'========================================================================*/
|
|
|
|
INSN(ed_illegal)
|
|
{
|
|
if (self->illegal != Z_NULL)
|
|
{
|
|
DATA[2] = 0;
|
|
return self->illegal(self, DATA[1]);
|
|
}
|
|
|
|
Q_0
|
|
PC += 2;
|
|
return 8;
|
|
}
|
|
|
|
|
|
/*-------------------------------------------------------------------------.
|
|
| Illegal opcodes prefixed with DDh or FDh make the CPU ignore the prefix, |
|
|
| As a result, the byte that immediately follows the prefix is treated as |
|
|
| the first byte of a new instruction. The prefix takes 4 T-states. |
|
|
'=========================================================================*/
|
|
|
|
INSN(xy_illegal)
|
|
{
|
|
PC++;
|
|
return insn_table[DATA[0] = DATA[1]](self);
|
|
}
|
|
|
|
|
|
#ifdef Z80_WITH_Q
|
|
INSN(xy_xcf)
|
|
{
|
|
Q_0
|
|
PC++;
|
|
return insn_table[DATA[0] = DATA[1]](self);
|
|
}
|
|
#else
|
|
# define xy_xcf xy_illegal
|
|
#endif
|
|
|
|
|
|
/* MARK: - Instructions: Hooking */
|
|
|
|
INSN(hook)
|
|
{
|
|
if (self->hook == Z_NULL)
|
|
{
|
|
Q_0
|
|
PC++;
|
|
return 4;
|
|
}
|
|
|
|
return ((DATA[0] = self->hook(CONTEXT, PC)) != Z80_HOOK)
|
|
? insn_table[DATA[0]](self) : 0;
|
|
}
|
|
|
|
|
|
/* MARK: - Instruction Function Tables */
|
|
|
|
#ifdef Z80_WITH_UNOFFICIAL_RETI
|
|
# define reti_retn reti
|
|
#else
|
|
# define reti_retn retn
|
|
#endif
|
|
|
|
static Insn const insn_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ nop, ld_SS_WORD, ld_vbc_a, inc_SS, V_J, V_J, ld_J_BYTE, rlca, ex_af_af_, add_hl_SS, ld_a_vbc, dec_SS, V_J, V_J, ld_J_BYTE, rrca,
|
|
/* 1 */ djnz_OFFSET, ld_SS_WORD, ld_vde_a, inc_SS, V_J, V_J, ld_J_BYTE, rla, jr_OFFSET, add_hl_SS, ld_a_vde, dec_SS, V_J, V_J, ld_J_BYTE, rra,
|
|
/* 2 */ jr_Z_OFFSET, ld_SS_WORD, ld_vWORD_hl, inc_SS, V_J, V_J, ld_J_BYTE, daa, jr_Z_OFFSET, add_hl_SS, ld_hl_vWORD, dec_SS, V_J, V_J, ld_J_BYTE, cpl,
|
|
/* 3 */ jr_Z_OFFSET, ld_SS_WORD, ld_vWORD_a, inc_SS, V_vhl, V_vhl, ld_vhl_BYTE, scf, jr_Z_OFFSET, add_hl_SS, ld_a_vWORD, dec_SS, V_J, V_J, ld_J_BYTE, ccf,
|
|
/* 4 */ nop, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_vhl, ld_J_K, ld_J_K, nop, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_vhl, ld_J_K,
|
|
/* 5 */ ld_J_K, ld_J_K, nop, ld_J_K, ld_J_K, ld_J_K, ld_J_vhl, ld_J_K, ld_J_K, ld_J_K, ld_J_K, nop, ld_J_K, ld_J_K, ld_J_vhl, ld_J_K,
|
|
/* 6 */ ld_J_K, ld_J_K, ld_J_K, ld_J_K, hook, ld_J_K, ld_J_vhl, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_K, nop, ld_J_vhl, ld_J_K,
|
|
/* 7 */ ld_vhl_K, ld_vhl_K, ld_vhl_K, ld_vhl_K, ld_vhl_K, ld_vhl_K, halt, ld_vhl_K, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_K, ld_J_vhl, nop,
|
|
/* 8 */ U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K,
|
|
/* 9 */ U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K,
|
|
/* A */ U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K,
|
|
/* B */ U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_K, U_a_vhl, U_a_K,
|
|
/* C */ ret_Z, pop_TT, jp_Z_WORD, jp_WORD, call_Z_WORD, push_TT, U_a_BYTE, rst_N, ret_Z, ret, jp_Z_WORD, cb_prefix, call_Z_WORD, call_WORD, U_a_BYTE, rst_N,
|
|
/* D */ ret_Z, pop_TT, jp_Z_WORD, out_vBYTE_a, call_Z_WORD, push_TT, U_a_BYTE, rst_N, ret_Z, exx, jp_Z_WORD, in_a_vBYTE, call_Z_WORD, dd_prefix, U_a_BYTE, rst_N,
|
|
/* E */ ret_Z, pop_TT, jp_Z_WORD, ex_vsp_hl, call_Z_WORD, push_TT, U_a_BYTE, rst_N, ret_Z, jp_hl, jp_Z_WORD, ex_de_hl, call_Z_WORD, ed_prefix, U_a_BYTE, rst_N,
|
|
/* F */ ret_Z, pop_TT, jp_Z_WORD, di, call_Z_WORD, push_TT, U_a_BYTE, rst_N, ret_Z, ld_sp_hl, jp_Z_WORD, ei, call_Z_WORD, fd_prefix, U_a_BYTE, rst_N};
|
|
|
|
static Insn const cb_insn_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K, G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K,
|
|
/* 1 */ G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K, G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K,
|
|
/* 2 */ G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K, G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K,
|
|
/* 3 */ G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K, G_K, G_K, G_K, G_K, G_K, G_K, G_vhl, G_K,
|
|
/* 4 */ bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K,
|
|
/* 5 */ bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K,
|
|
/* 6 */ bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K,
|
|
/* 7 */ bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_K, bit_N_vhl, bit_N_K,
|
|
/* 8 */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* 9 */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* A */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* B */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* C */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* D */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* E */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K,
|
|
/* F */ M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_K, M_N_vhl, M_N_K};
|
|
|
|
static Insn const ed_insn_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* 1 */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* 2 */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* 3 */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* 4 */ in_J_vc, out_vc_J, sbc_hl_SS, ld_vWORD_SS, neg, retn, im_0, ld_i_a, in_J_vc, out_vc_J, adc_hl_SS, ld_SS_vWORD, neg, reti, im_0, ld_r_a,
|
|
/* 5 */ in_J_vc, out_vc_J, sbc_hl_SS, ld_vWORD_SS, neg, retn, im_1, ld_a_i, in_J_vc, out_vc_J, adc_hl_SS, ld_SS_vWORD, neg, reti_retn, im_2, ld_a_r,
|
|
/* 6 */ in_J_vc, out_vc_J, sbc_hl_SS, ld_vWORD_SS, neg, retn, im_0, rrd, in_J_vc, out_vc_J, adc_hl_SS, ld_SS_vWORD, neg, reti_retn, im_0, rld,
|
|
/* 7 */ in_vc, out_vc_0, sbc_hl_SS, ld_vWORD_SS, neg, retn, im_1, ed_illegal, in_J_vc, out_vc_J, adc_hl_SS, ld_SS_vWORD, neg, reti_retn, im_2, ed_illegal,
|
|
/* 8 */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* 9 */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* A */ ldi, cpi, ini, outi, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ldd, cpd, ind, outd, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* B */ ldir, cpir, inir, otir, ed_illegal, ed_illegal, ed_illegal, ed_illegal, lddr, cpdr, indr, otdr, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* C */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* D */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* E */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal,
|
|
/* F */ ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal, ed_illegal};
|
|
|
|
static Insn const xy_insn_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ nop_nop, xy_illegal, xy_illegal, xy_illegal, V_O, V_O, ld_O_BYTE, xy_illegal, xy_illegal, add_XY_WW, xy_illegal, xy_illegal, V_O, V_O, ld_O_BYTE, xy_illegal,
|
|
/* 1 */ xy_illegal, xy_illegal, xy_illegal, xy_illegal, V_O, V_O, ld_O_BYTE, xy_illegal, xy_illegal, add_XY_WW, xy_illegal, xy_illegal, V_O, V_O, ld_O_BYTE, xy_illegal,
|
|
/* 2 */ xy_illegal, ld_XY_WORD, ld_vWORD_XY, inc_XY, V_O, V_O, ld_O_BYTE, xy_illegal, xy_illegal, add_XY_WW, ld_XY_vWORD, dec_XY, V_O, V_O, ld_O_BYTE, xy_illegal,
|
|
/* 3 */ xy_illegal, xy_illegal, xy_illegal, xy_illegal, V_vXYpOFFSET, V_vXYpOFFSET, ld_vXYpOFFSET_BYTE, xy_xcf, xy_illegal, add_XY_WW, xy_illegal, xy_illegal, V_O, V_O, ld_O_BYTE, xy_xcf,
|
|
/* 4 */ nop_nop, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_J_vXYpOFFSET, ld_O_P, ld_O_P, nop_nop, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_J_vXYpOFFSET, ld_O_P,
|
|
/* 5 */ ld_O_P, ld_O_P, nop_nop, ld_O_P, ld_O_P, ld_O_P, ld_J_vXYpOFFSET, ld_O_P, ld_O_P, ld_O_P, ld_O_P, nop_nop, ld_O_P, ld_O_P, ld_J_vXYpOFFSET, ld_O_P,
|
|
/* 6 */ ld_O_P, ld_O_P, ld_O_P, ld_O_P, nop_nop, ld_O_P, ld_J_vXYpOFFSET, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_O_P, nop_nop, ld_J_vXYpOFFSET, ld_O_P,
|
|
/* 7 */ ld_vXYpOFFSET_K, ld_vXYpOFFSET_K, ld_vXYpOFFSET_K, ld_vXYpOFFSET_K, ld_vXYpOFFSET_K, ld_vXYpOFFSET_K, xy_illegal, ld_vXYpOFFSET_K, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_O_P, ld_J_vXYpOFFSET, nop_nop,
|
|
/* 8 */ U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P,
|
|
/* 9 */ U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P,
|
|
/* A */ U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P,
|
|
/* B */ U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_P, U_a_vXYpOFFSET, U_a_P,
|
|
/* C */ xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_cb_prefix, xy_illegal, xy_illegal, xy_illegal, xy_illegal,
|
|
/* D */ xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_xy, xy_illegal, xy_illegal,
|
|
/* E */ xy_illegal, pop_XY, xy_illegal, ex_vsp_XY, xy_illegal, push_XY, xy_illegal, xy_illegal, xy_illegal, jp_XY, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal,
|
|
/* F */ xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, xy_illegal, ld_sp_XY, xy_illegal, xy_illegal, xy_illegal, xy_xy, xy_illegal, xy_illegal};
|
|
|
|
static Insn const xy_cb_insn_table[256] = {
|
|
/* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
|
|
/* 0 */ G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K,
|
|
/* 1 */ G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K,
|
|
/* 2 */ G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K,
|
|
/* 3 */ G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET_K, G_vXYpOFFSET, G_vXYpOFFSET_K,
|
|
/* 4 */ bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET,
|
|
/* 5 */ bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET,
|
|
/* 6 */ bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET,
|
|
/* 7 */ bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET, bit_N_vXYpOFFSET,
|
|
/* 8 */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* 9 */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* A */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* B */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* C */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* D */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* E */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K,
|
|
/* F */ M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET_K, M_N_vXYpOFFSET, M_N_vXYpOFFSET_K};
|
|
|
|
|
|
/* MARK: - Public Functions */
|
|
|
|
/*-----------------------------------------------------------------------.
|
|
| On POWER-ON, the CPU zeroes PC, I and R, sets SP, IX, IY, AF, BC, DE, |
|
|
| HL, AF', BC', DE' and HL' to FFFFh [1, 2], resets the interrupt enable |
|
|
| flip-flops (IFF1 and IFF2) and selects interrupt mode 0 [3]. On Zilog |
|
|
| NMOS models, F is sometimes set to FDh (NF reset) [1]. |
|
|
| |
|
|
| There is no information about the initial state of MEMPTR and Q, so |
|
|
| they are assumed to be 0. |
|
|
| |
|
|
| References: |
|
|
| 1. https://baltazarstudios.com/webshare/Z80-undocumented-behavior.htm |
|
|
| 2. https://worldofspectrum.org/forums/discussion/34574 |
|
|
| 3. Young, Sean (2005-09-18). "Undocumented Z80 Documented, The" v0.91, |
|
|
| p. 20. |
|
|
'=======================================================================*/
|
|
|
|
Z80_API void z80_power(Z80 *self, zbool state)
|
|
{
|
|
MEMPTR = PC = R = I = IFF1 = IFF2 = IM = Q =
|
|
DATA[0] = HALT_LINE = INT_LINE = RESUME = REQUEST = 0;
|
|
|
|
SP = IX = IY = AF = BC = DE = HL = AF_ = BC_ = DE_ = HL_ =
|
|
state ? Z_UINT16(0xFFFF) : 0;
|
|
|
|
# ifdef Z80_WITH_RESET_SIGNAL
|
|
self->reset_signal = Z_NULL;
|
|
self->reset_signal_mask = 0;
|
|
# endif
|
|
}
|
|
|
|
|
|
/*-------------------------------------------------------------------------.
|
|
| The normal RESET zeroes PC, I, and R [1, 2, 3, 4, 5, 6], resets the |
|
|
| interrupt enable flip-flops (IFF1 and IFF2) [1, 2, 3, 4, 5] and selects |
|
|
| interrupt mode 0 [1, 2, 3, 4, 7]. |
|
|
| |
|
|
| References: |
|
|
| 1. Zilog (2016-09). "Z80 CPU User Manual" rev. 11, p. 6. |
|
|
| 2. SGS-Thomson (1990-01). "Z80 Microprocessor Family" 1st ed., p. 33. |
|
|
| 3. Brewer, Tony (2014-12). "Z80 Special Reset". |
|
|
| * http://primrosebank.net/computers/z80/z80_special_reset.htm |
|
|
| 4. Flammenkamp, Achim. "Interrupt Behaviour of the Z80 CPU". |
|
|
| * http://z80.info/interrup.htm |
|
|
| 5. https://baltazarstudios.com/webshare/Z80-undocumented-behavior.htm |
|
|
| 6. https://worldofspectrum.org/forums/discussion/34574 |
|
|
| 7. Zilog (1978-05). "Z80 Family Program Interrupt Structure, The", p. 8. |
|
|
'=========================================================================*/
|
|
|
|
Z80_API void z80_instant_reset(Z80 *self)
|
|
{
|
|
if (HALT_LINE) {EXIT_HALT;}
|
|
|
|
PC = R = I = IFF1 = IFF2 = IM =
|
|
DATA[0] = HALT_LINE = RESUME = REQUEST = 0;
|
|
}
|
|
|
|
|
|
#ifdef Z80_WITH_SPECIAL_RESET
|
|
Z80_API void z80_special_reset(Z80 *self)
|
|
{REQUEST |= Z80_REQUEST_SPECIAL_RESET;}
|
|
#endif
|
|
|
|
|
|
Z80_API void z80_int(Z80 *self, zbool state)
|
|
{
|
|
if (!(INT_LINE = state)) REQUEST &= ~(zuint8)Z80_REQUEST_INT;
|
|
else if (IFF1) REQUEST |= Z80_REQUEST_INT;
|
|
}
|
|
|
|
|
|
Z80_API void z80_nmi(Z80 *self)
|
|
{REQUEST |= Z80_REQUEST_NMI;}
|
|
|
|
|
|
#ifdef Z80_WITH_EXECUTE
|
|
Z80_API zusize z80_execute(Z80 *self, zusize cycles)
|
|
{
|
|
ZInt16 *xy;
|
|
|
|
R7 = R;
|
|
self->cycles = 0;
|
|
self->cycle_limit = cycles;
|
|
|
|
if (RESUME && cycles) switch (RESUME)
|
|
{
|
|
case Z80_RESUME_HALT:
|
|
(void)halt(self);
|
|
break;
|
|
|
|
case Z80_RESUME_XY:
|
|
RESUME = 0;
|
|
R++;
|
|
XY = (xy = &self->ix_iy[(DATA[0] >> 5) & 1])->uint16_value;
|
|
self->cycles += xy_insn_table[DATA[1] = FETCH_OPCODE(PC + 1)](self);
|
|
xy->uint16_value = XY;
|
|
break;
|
|
}
|
|
|
|
while (self->cycles < self->cycle_limit)
|
|
{
|
|
R++;
|
|
self->cycles += insn_table[DATA[0] = FETCH_OPCODE(PC)](self);
|
|
}
|
|
|
|
R = R_ALL;
|
|
return self->cycles;
|
|
}
|
|
#endif
|
|
|
|
|
|
Z80_API zusize z80_run(Z80 *self, zusize cycles)
|
|
{
|
|
ZInt16 *xy;
|
|
zuint8 ird;
|
|
|
|
/*---------------------------------------------------------------------.
|
|
| The CPU increments R during each M1 cycle without altering the most |
|
|
| significant bit, commonly known as R7. This behavior is not emulated |
|
|
| in every increment for obvious speed reasons. Instead, a copy of R |
|
|
| is used to preserve R7, which is restored before returning from this |
|
|
| function. The emulation of `ld {a,r|r,a}` takes this into account. |
|
|
'=====================================================================*/
|
|
R7 = R;
|
|
|
|
self->cycles = 0;
|
|
self->cycle_limit = cycles;
|
|
|
|
if (RESUME && cycles) switch (RESUME)
|
|
{
|
|
/*------------------------------------------------------------.
|
|
| The CPU is halted. To avoid affecting the speed of the main |
|
|
| execution loop, this state is executed by a dedicated loop |
|
|
| in the function that emulates the `halt` instruction. |
|
|
'============================================================*/
|
|
case Z80_RESUME_HALT:
|
|
if (REQUEST)
|
|
{
|
|
RESUME = 0;
|
|
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
if ((REQUEST & Z80_REQUEST_SPECIAL_RESET) && HALT_LINE)
|
|
{
|
|
zuint8 opcode;
|
|
|
|
HALT_LINE = 0;
|
|
|
|
if (self->halt != Z_NULL)
|
|
self->halt(CONTEXT, Z80_HALT_EXIT_EARLY);
|
|
|
|
if (IS_XY_PREFIX(DATA[0] = opcode = DATA[2]))
|
|
self->cycles += insn_table[FETCH_OPCODE(PC)](self);
|
|
|
|
else if (opcode != Z80_HALT)
|
|
{
|
|
PC--;
|
|
self->cycles += insn_table[opcode](self) - 4;
|
|
}
|
|
}
|
|
# endif
|
|
}
|
|
|
|
else (void)halt(self);
|
|
break;
|
|
|
|
/*-------------------------------------------------------.
|
|
| The CPU is in normal operation state; the emulator ran |
|
|
| out of clock cycles by fetching a prefix DDh or FDh. |
|
|
'=======================================================*/
|
|
case Z80_RESUME_XY:
|
|
RESUME = 0;
|
|
R++;
|
|
XY = (xy = &self->ix_iy[(DATA[0] >> 5) & 1])->uint16_value;
|
|
self->cycles += xy_insn_table[DATA[1] = FETCH_OPCODE(PC + 1)](self);
|
|
xy->uint16_value = XY;
|
|
break;
|
|
|
|
/*---------------------------------------------------------.
|
|
| The CPU is responding to an INT in mode 0; the emulator |
|
|
| ran out of clock cycles by fetching a prefix DDh or FDh. |
|
|
'=========================================================*/
|
|
# ifdef Z80_WITH_FULL_IM0
|
|
case Z80_RESUME_IM0_XY:
|
|
ird = DATA[0];
|
|
goto im0_begin;
|
|
# endif
|
|
}
|
|
|
|
while (self->cycles < self->cycle_limit) /* main execution loop */
|
|
{
|
|
# ifdef Z80_WITH_RESET_SIGNAL
|
|
/* Check external reset signal at the instruction boundary,
|
|
matching real Z80 behaviour where /RESET is sampled at the
|
|
rising edge of the last T-state. A non-zero value causes
|
|
immediate exit before the next instruction fetch. */
|
|
if (self->reset_signal != Z_NULL && (*self->reset_signal & self->reset_signal_mask))
|
|
break;
|
|
# endif
|
|
|
|
if (REQUEST)
|
|
{
|
|
/*-------------------------------------------------------------------------.
|
|
| After detecting a special RESET signal, the CPU completes the ongoing |
|
|
| instruction or interrupt response and then zeroes PC during the falling |
|
|
| edge of the next M1T1. The special RESET can be used in conjunction with |
|
|
| an interrupt, in which case PC is zeroed during the subsequent interrupt |
|
|
| acknowledge M-cycle. Otherwise, if no interrupt has been accepted at the |
|
|
| TLAST of the instruction or interrupt response in which the special |
|
|
| RESET has been detected, the CPU produces an internal NOP of 4 T-states |
|
|
| to allow for the fetch-execute overlap to take place, during which it |
|
|
| fetches the next opcode and zeroes PC. |
|
|
| |
|
|
| References: |
|
|
| * Brewer, Tony (2014-12). "Z80 Special Reset". |
|
|
| * http://primrosebank.net/computers/z80/z80_special_reset.htm |
|
|
| * US Patent 4486827. |
|
|
| * Checked with "Visual Z80 Remix". |
|
|
'=========================================================================*/
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
zuint8 special_reset = REQUEST & Z80_REQUEST_SPECIAL_RESET;
|
|
# endif
|
|
|
|
/*-------------------------------------------------------------------------.
|
|
| NMI Response: Execute `rst 66h` | T-states: 11:533 |
|
|
|--------------------------------------------------------------------------|
|
|
| The non-maskable interrupt takes priority over the maskable interrupt |
|
|
| and cannot be disabled under software control. Its usual function is to |
|
|
| provide immediate response to important signals. The CPU responds to an |
|
|
| NMI request by pushing PC onto the stack and jumping to the ISR located |
|
|
| at address 0066h. The interrupt enable flip-flop #1 (IFF1) is reset to |
|
|
| prevent any maskable interrupt from being accepted during the execution |
|
|
| of this routine, which is usually exited by using a `reti` or `retn` |
|
|
| instruction to restore the original state of IFF1 [1]. |
|
|
| |
|
|
| Some technical documents from Zilog include an erroneous timing diagram |
|
|
| showing an NMI acknowledge cycle of 4 T-states. However, documents from |
|
|
| other manufacturers and third parties specify that this M-cycle has 5 |
|
|
| T-states, as has been confirmed by low-level tests [2] and electronic |
|
|
| simulations [3]. |
|
|
| |
|
|
| In 2022, Manuel Sainz de Baranda y Goñi discovered that the CPU does not |
|
|
| accept a second NMI during the NMI response [4, 5]. Therefore, it is not |
|
|
| possible to chain two NMI responses in a row without executing at least |
|
|
| one instruction between them [3]. |
|
|
| |
|
|
| References: |
|
|
| 1. Zilog (1978-05). "Z80 Family Program Interrupt Structure, The", |
|
|
| pp. 4-5. |
|
|
| 2. https://baltazarstudios.com/webshare/Z80-undocumented-behavior.htm |
|
|
| 3. Checked with "Visual Z80 Remix". |
|
|
| 4. https://spectrumcomputing.co.uk/forums/viewtopic.php?p=91405#p91405 |
|
|
| 5. https://stardot.org.uk/forums/viewtopic.php?p=356579#p356579 |
|
|
'=========================================================================*/
|
|
if (REQUEST & Z80_REQUEST_REJECT_NMI)
|
|
REQUEST = 0;
|
|
|
|
else if (REQUEST & Z80_REQUEST_NMI)
|
|
{
|
|
REQUEST = Z80_REQUEST_REJECT_NMI;
|
|
IFF1 = 0;
|
|
if (HALT_LINE) {EXIT_HALT;}
|
|
R++;
|
|
if (self->nmia != Z_NULL) (void)self->nmia(CONTEXT, PC);
|
|
DATA[0] = 0;
|
|
Q_0
|
|
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
PUSH(PC >> special_reset);
|
|
# else
|
|
PUSH(PC);
|
|
# endif
|
|
|
|
MEMPTR = PC = 0x66;
|
|
self->cycles += 11;
|
|
continue;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------.
|
|
| INT Response |
|
|
|--------------------------------------------------------------------------|
|
|
| The maskable interrupt is enabled and disabled by using, respectively, |
|
|
| the instructions `ei` and `di`, which control the state of the interrupt |
|
|
| enable flip-flops (IFF1 and IFF2). The CPU does not accept this kind of |
|
|
| interrupt during an `ei` instruction. This allows ISRs to return without |
|
|
| the danger of being interrupted immediately after re-enabling interrupts |
|
|
| if the /INT line is still active, which could cause a stack overflow. |
|
|
| |
|
|
| In 2021, Andre Weissflog (aka Floh) discovered that `reti` and `retn` do |
|
|
| not accept the maskable interrupt if IFF1 and IFF2 do not have the same |
|
|
| state prior to the execution of the instruction, which can only be |
|
|
| caused by an earlier NMI response [1]. This behavior was rediscovered in |
|
|
| 2022 by Manuel Sainz de Baranda y Goñi [2, 3]. |
|
|
| |
|
|
| References: |
|
|
| 1. Weissflog, Andre (2021-12-17). "New Cycle-Stepped Z80 Emulator, A". |
|
|
| * https://floooh.github.io/2021/12/17/cycle-stepped-z80.html |
|
|
| 2. https://spectrumcomputing.co.uk/forums/viewtopic.php?t=7086 |
|
|
| 3. https://stardot.org.uk/forums/viewtopic.php?t=24662 |
|
|
'=========================================================================*/
|
|
else if (
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
(REQUEST & Z80_REQUEST_INT) &&
|
|
# endif
|
|
/* If the previous instruction is not `ei` and... */
|
|
DATA[0] != 0xFB &&
|
|
/* the previous instruction is not `reti/retn`,
|
|
or IFF1 has not changed. */
|
|
(self->data.uint32_value & Z_UINT32_BIG_ENDIAN(Z_UINT32(0xFFC70100)))
|
|
!= Z_UINT32_BIG_ENDIAN(Z_UINT32(0xED450000))
|
|
)
|
|
{
|
|
# ifdef Z80_WITH_FULL_IM0
|
|
Z80Read hook;
|
|
IM0 im0;
|
|
# endif
|
|
|
|
REQUEST = IFF1 = IFF2 = 0;
|
|
if (HALT_LINE) {EXIT_HALT;}
|
|
|
|
/*----------------------------------------------------------------------.
|
|
| Due to a bug, the Zilog Z80 NMOS resets PF when an INT is accepted |
|
|
| during the execution of the `ld a,{i|r}` instructions. |
|
|
| |
|
|
| References: |
|
|
| * Zilog (1989-01). "Z80 Family Data Book", pp. 412-413. |
|
|
| * Roshchin, Ivan (1998). "Undocumented Feature of the Z80 Processor". |
|
|
| * https://zxpress.ru/article.php?id=7820 |
|
|
| * http://code-zx.zxnet-archive.ru/id/123456 |
|
|
'======================================================================*/
|
|
# ifdef Z80_WITH_ZILOG_NMOS_LD_A_IR_BUG
|
|
if ( (OPTIONS & Z80_OPTION_LD_A_IR_BUG) &&
|
|
(self->data.uint16_array[0] & Z_UINT16_BIG_ENDIAN(Z_UINT16(0xFFF7)))
|
|
== Z_UINT16_BIG_ENDIAN(Z_UINT16(0xED57))
|
|
)
|
|
FLAGS = F & ~(zuint8)PF;
|
|
# endif
|
|
|
|
/*-----------------------------------------------------------------------.
|
|
| The INT acknowledge M-cycle (INTA) indicates that the interrupting I/O |
|
|
| device can write to the data bus. The CPU adds 2 wait T-states to this |
|
|
| M-cycle, allowing sufficient time to identify which device must insert |
|
|
| the interrupt response data (IRD). The first and possibly sole byte of |
|
|
| the IRD is read from the data bus during this special M1 cycle. |
|
|
| |
|
|
| The value FFh is assumed when the `Z80::inta` callback is not used. |
|
|
| This is the most convenient default IRD, since an `rst 38h` will be |
|
|
| executed if the interrupt mode is 0. |
|
|
'=======================================================================*/
|
|
R++;
|
|
ird = (self->inta != Z_NULL) ? self->inta(CONTEXT, PC) : 0xFF;
|
|
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
PC >>= special_reset;
|
|
# endif
|
|
|
|
switch (IM)
|
|
{
|
|
/*-------------------------------------------------------------------------.
|
|
| Interrupt Mode 0: Execute Instruction | T-states: 2*n + instruction |
|
|
|--------------------------------------------------------------------------|
|
|
| An instruction supplied via the data bus is executed. Its first byte is |
|
|
| read during the INTA M-cycle and, if it is an opcode prefix, additional |
|
|
| M-cycles of this kind are produced until the final opcode of the |
|
|
| instruction is fetched [1]. Each INTA M-cycle takes as many T-states as |
|
|
| its normal M1 counterpart (the opcode fetch M-cycle) plus the 2 wait |
|
|
| T-states mentioned in the previous comment [1]. Subsequent bytes of the |
|
|
| instruction are fetched by using normal memory read M-cycles [1, 2], |
|
|
| during which the interrupting I/O device must still supply the data [2]. |
|
|
| The PC register, however, remains at its pre-interrupt state, not being |
|
|
| incremented as a result of the instruction fetch [1, 2]. |
|
|
| |
|
|
| References: |
|
|
| 1. Checked with "Visual Z80 Remix". |
|
|
| 2. Zilog (1978-05). "Z80 Family Program Interrupt Structure, The", |
|
|
| pp. 6, 8. |
|
|
'=========================================================================*/
|
|
case 0:
|
|
DATA[0] = ird;
|
|
|
|
# ifdef Z80_WITH_FULL_IM0
|
|
im0_begin:
|
|
|
|
/*-----------------------------------------------------.
|
|
| The `Z80::hook` callback is temporarily disabled, as |
|
|
| traps are ignored during the INT response in mode 0. |
|
|
'=====================================================*/
|
|
hook = self->hook;
|
|
self->hook = Z_NULL;
|
|
|
|
/*-----------------------------------------------------------------------.
|
|
| The `Z80::fetch` callback is temporarily replaced with a trampoline |
|
|
| that invokes `Z80::int_fetch`. This trampoline needs access to the |
|
|
| callback pointer as well as the initial, non-incremented value of PC. |
|
|
| To provide this, the value of `Z80::context` is temporarily replaced |
|
|
| with a pointer to an `IM0` object that holds the real context and all |
|
|
| required data. As a consequence, other callbacks must also be replaced |
|
|
| with trampolines so that the real context can be passed to them. |
|
|
| |
|
|
| The main idea is that the instruction code invokes trampolines rather |
|
|
| than callbacks. The trampoline assigned to `Z80::fetch` ignores the |
|
|
| received fetch address and passes the initial, non-incremented value |
|
|
| of PC to `Z80::int_fetch` instead. |
|
|
'=======================================================================*/
|
|
im0.z80 = self;
|
|
im0.context = CONTEXT;
|
|
im0.fetch = self->fetch;
|
|
im0.read = self->read;
|
|
im0.write = self->write;
|
|
im0.in = self->in;
|
|
im0.out = self->out;
|
|
im0.pc = PC;
|
|
self->context = &im0;
|
|
self->fetch = (Z80Read )im0_fetch;
|
|
self->read = (Z80Read )im0_read;
|
|
self->write = (Z80Write)im0_write;
|
|
self->in = (Z80Read )im0_in;
|
|
self->out = (Z80Write)im0_out;
|
|
|
|
im0_execute:
|
|
|
|
/*-----------------------------------------------------------------------.
|
|
| `call`, `djnz`, `jr` and `rst` increment PC before pushing it onto the |
|
|
| stack or using it as the base address. This makes it necessary to |
|
|
| decrement PC before executing any of these instructions so that the |
|
|
| final address is correct. `jmp` and `ret` are handled here as well |
|
|
| because in this case the pre-decrement has no effect and PC must not |
|
|
| be corrected either after executing the instruction. This group of |
|
|
| instructions is identified using a table of decrements. Note that |
|
|
| `jmp (XY)`, `reti` and `retn` are prefixed and will be handled later. |
|
|
'=======================================================================*/
|
|
if (im0_pc_decrement_table[ird])
|
|
{
|
|
PC -= im0_pc_decrement_table[ird];
|
|
self->cycles += 2 + insn_table[ird](self);
|
|
}
|
|
|
|
/* `halt` */
|
|
else if (ird == Z80_HALT) HALT_LINE = 1;
|
|
|
|
/*---------------------------------------------------------------.
|
|
| Instructions with the CBh prefix are called directly from here |
|
|
| after fetching the opcode in the 2nd INTA. This bypasses the |
|
|
| `cb_prefix` function, so PC is never incremented. |
|
|
'===============================================================*/
|
|
else if (ird == 0xCB)
|
|
{
|
|
R++;
|
|
self->cycles += 4 + cb_insn_table[DATA[1] = self->inta(im0.context, im0.pc)](self);
|
|
}
|
|
|
|
/* Instructions with the EDh prefix. */
|
|
else if (ird == 0xED)
|
|
{
|
|
Insn insn;
|
|
|
|
R++;
|
|
|
|
if ((insn = ed_insn_table[DATA[1] = ird = self->inta(im0.context, im0.pc)]) != ed_illegal)
|
|
{
|
|
im0.ld_i_a = self->ld_i_a;
|
|
im0.ld_r_a = self->ld_r_a;
|
|
im0.reti = self->reti;
|
|
im0.retn = self->retn;
|
|
self->ld_i_a = (Z80Notify)im0_ld_i_a;
|
|
self->ld_r_a = (Z80Notify)im0_ld_r_a;
|
|
|
|
# ifdef Z80_WITH_IM0_RETX_NOTIFICATIONS
|
|
self->reti = (Z80Notify)im0_reti;
|
|
self->retn = (Z80Notify)im0_retn;
|
|
# else
|
|
self->reti = Z_NULL;
|
|
self->retn = Z_NULL;
|
|
# endif
|
|
|
|
PC -= ((ird & 0xC7) == 0x43)
|
|
? 4 /* `ld SS,(WORD)` and `ld (WORD),SS`. */
|
|
: 2 /* All other instructions. */;
|
|
|
|
self->cycles += 4 + insn(self);
|
|
|
|
self->ld_i_a = im0.ld_i_a;
|
|
self->ld_r_a = im0.ld_r_a;
|
|
self->reti = im0.reti;
|
|
self->retn = im0.retn;
|
|
}
|
|
|
|
else if (self->illegal == Z_NULL)
|
|
self->cycles += 4 + 8;
|
|
|
|
else {
|
|
DATA[2] = 4; /* Notify wait T-states. */
|
|
self->cycles += 4 + self->illegal(self, ird);
|
|
}
|
|
}
|
|
|
|
/* Instructions with the prefix DDh, FDh, DDCBh or FDCBh. */
|
|
else if (IS_XY_PREFIX(ird))
|
|
{
|
|
Insn insn;
|
|
|
|
if (RESUME) RESUME = 0;
|
|
|
|
else {
|
|
im0_advance_xy:
|
|
if ((self->cycles += 6) >= self->cycle_limit)
|
|
{
|
|
RESUME = Z80_RESUME_IM0_XY;
|
|
goto im0_finalize;
|
|
}
|
|
}
|
|
|
|
R++;
|
|
|
|
if (IS_XY_PREFIX(ird = self->inta(im0.context, im0.pc)))
|
|
{
|
|
DATA[0] = ird;
|
|
goto im0_advance_xy;
|
|
}
|
|
|
|
if ((insn = xy_insn_table[ird]) == xy_illegal)
|
|
{
|
|
DATA[0] = ird;
|
|
PC++;
|
|
goto im0_execute;
|
|
}
|
|
|
|
DATA[1] = ird;
|
|
XY = (xy = &self->ix_iy[(DATA[0] >> 5) & 1])->uint16_value;
|
|
self->cycles += 2 + insn(self);
|
|
xy->uint16_value = XY;
|
|
|
|
/* Restore PC, except for `jp (XY)`. */
|
|
if (ird != 0xE9) PC = im0.pc;
|
|
}
|
|
|
|
else {
|
|
self->cycles += 2 + insn_table[ird](self);
|
|
PC = im0.pc;
|
|
}
|
|
|
|
im0_finalize:
|
|
self->context = im0.context;
|
|
self->fetch = im0.fetch;
|
|
self->read = im0.read;
|
|
self->write = im0.write;
|
|
self->in = im0.in;
|
|
self->out = im0.out;
|
|
self->hook = hook;
|
|
|
|
if (HALT_LINE)
|
|
{
|
|
if (self->halt != Z_NULL) self->halt(im0.context, 1);
|
|
RESUME = Z80_RESUME_HALT;
|
|
Q_0
|
|
self->cycles += 6;
|
|
(void)halt(self);
|
|
}
|
|
|
|
continue;
|
|
|
|
# else
|
|
switch (ird)
|
|
{
|
|
case 0xC3: /* `jp WORD` */
|
|
Q_0
|
|
MEMPTR = PC = int_fetch_16(self);
|
|
self->cycles += 2 + 10;
|
|
continue;
|
|
|
|
case 0xCD: /* `call WORD` */
|
|
Q_0
|
|
MEMPTR = int_fetch_16(self);
|
|
PUSH(PC);
|
|
PC = MEMPTR;
|
|
self->cycles += 2 + 17;
|
|
continue;
|
|
|
|
default: /* `rst N` is assumed for all other instructions. */
|
|
Q_0
|
|
PUSH(PC);
|
|
MEMPTR = PC = ird & 56;
|
|
self->cycles += 2 + 11;
|
|
continue;
|
|
}
|
|
# endif
|
|
|
|
/*----------------------------------------------------------.
|
|
| Interrupt Mode 1: Execute `rst 38h` | T-states: 13:733 |
|
|
|-----------------------------------------------------------|
|
|
| An internal `rst 38h` is executed. The interrupt response |
|
|
| data read from the data bus is disregarded. |
|
|
'==========================================================*/
|
|
case 1:
|
|
DATA[0] = 0;
|
|
Q_0
|
|
PUSH(PC);
|
|
MEMPTR = PC = 0x38;
|
|
self->cycles += 13;
|
|
continue;
|
|
|
|
/*---------------------------------------------------------------------.
|
|
| Interrupt Mode 2: Execute `call (i:BYTE)` | T-states: 19:73333 |
|
|
|----------------------------------------------------------------------|
|
|
| An indirect call is executed. The pointer to the ISR is loaded from |
|
|
| the memory address formed by taking the I register as the most |
|
|
| significant byte, and the interrupt response vector (IRD) read from |
|
|
| the data bus as the least significant byte. |
|
|
| |
|
|
| Zilog's official documentation states that the least significant bit |
|
|
| of the interrupt response vector "must be a zero", since the address |
|
|
| formed "is used to get two adjacent bytes to form a complete 16-bit |
|
|
| service routine starting address and the addresses must always start |
|
|
| in even locations" [1]. However, Sean Young's tests found that there |
|
|
| is no such limitation [2]; the CPU fetches the ISR pointer from the |
|
|
| specified location regardless of the value of bit 0 of the IRD. |
|
|
| |
|
|
| References: |
|
|
| 1. Zilog (2005-03). "Z80 CPU User Manual" rev. 5, pp. 25-26. |
|
|
| 2. Young, Sean (2005-09-18). "Undocumented Z80 Documented, The" |
|
|
| v0.91, p. 20. |
|
|
'=====================================================================*/
|
|
case 2:
|
|
DATA[0] = 0;
|
|
Q_0
|
|
PUSH(PC);
|
|
MEMPTR = PC = READ_16((zuint16)(((zuint16)I << 8) | ird));
|
|
self->cycles += 19;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
# ifdef Z80_WITH_SPECIAL_RESET
|
|
if (special_reset)
|
|
{
|
|
REQUEST = 0;
|
|
|
|
/*---------------------------------------------------------.
|
|
| The /HALT line goes low and then high during TLAST if a |
|
|
| special RESET is detected during the `halt` instruction. |
|
|
'=========================================================*/
|
|
if (DATA[0] == Z80_HALT && self->halt != Z_NULL)
|
|
self->halt(CONTEXT, Z80_HALT_CANCEL);
|
|
|
|
R++;
|
|
if (self->nop != Z_NULL) (void)self->nop(CONTEXT, PC);
|
|
DATA[0] = 0;
|
|
Q_0
|
|
PC = 0;
|
|
self->cycles += 4;
|
|
continue;
|
|
}
|
|
# endif
|
|
}
|
|
|
|
R++;
|
|
|
|
# ifdef Z80_WITH_RESET_SIGNAL
|
|
/* Fetch the opcode first (generates M1 bus cycle), then
|
|
re-check the reset signal before executing. A real Z80
|
|
checks /RESET at M-cycle boundaries — if /RESET went LOW
|
|
during the M1 fetch, the CPU should not execute this
|
|
opcode. */
|
|
DATA[0] = FETCH_OPCODE(PC);
|
|
|
|
if (self->reset_signal != Z_NULL && (*self->reset_signal & self->reset_signal_mask))
|
|
break;
|
|
|
|
self->cycles += insn_table[DATA[0]](self);
|
|
# else
|
|
self->cycles += insn_table[DATA[0] = FETCH_OPCODE(PC)](self);
|
|
# endif
|
|
}
|
|
|
|
R = R_ALL; /* Restore R7 bit. */
|
|
return self->cycles;
|
|
}
|
|
|
|
|
|
#ifdef Z80_WITH_DLL_MAIN_CRT_STARTUP
|
|
int Z_MICROSOFT_STD_CALL _DllMainCRTStartup(void *hDllHandle, unsigned long dwReason, void *lpReserved)
|
|
{return 1;}
|
|
#endif
|
|
|
|
|
|
/* Z80.c EOF */
|