-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathreadme.txt
251 lines (159 loc) · 6.96 KB
/
readme.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
************************************************
* *
* CZ80 (Z80 CPU emulator) version 0.91 *
* Compiled with Dev-C++ *
* Copyright 2004-2005 Stéphane Dallongeville *
* *
************************************************
CZ80 is a Z80 CPU emulator, priorities were given to :
- code size
- speed
- accuracy
- portablity
It supports almost all undocumented opcodes and flags.
The emulator can be freely distribued and used for any non commercial
project as long you don't forget to credit me somewhere :)
If you want some support about the CZ80, you can contact me on
the Gens forum (http://gens.consolemul.com then go to the forum).
You should find the following files in the emulation pack :
- cz80.h -> header file (prototypes, declarations...)
- cz80.c -> contains emulation core itself
- cz80.inc -> contains most used macros
- cz80jmp.inc -> Jump table definition when Jump Table used
- cz80exec.inc -> contains the major Cz80_Exec(...) function
- cz80_op.inc -> contains code for simple Z80 opcodes emulation
- cz80_opCB.inc -> contains code for CB prefixed Z80 opcodes emulation
- cz80_opED.inc -> contains code for ED prefixed Z80 opcodes emulation
- cz80_opXY.inc -> contains code for DD/FD prefixed Z80 opcodes emulation
- cz80_opXYCB.inc -> contains code for DD/FD + CB prefixed Z80 opcodes emulation
- readme.txt -> the current file you're reading ;)
* How compile the emulator ?
****************************
The emulator has been written with Dev-CPP 4.9.X.X
You will maybe need to modify the u8, u16, u32, s8, s16, s32 and FASTCALL
definitions (cz80.h) according to your C compiler and the target system.
Then compile the cz80.c file, you should obtain a cz80.o (or cz80.obj) file...
at this moment, you're ready to use the emulator just by linking the file in your project :)
* How to use the emulator ?
***************************
1) Include the header file in your source :
------------------------------------------
#include "cz80.h"
2) Init the CZ80 core :
-----------------------
If you want to use the internal CZ80 context offered :
Cz80_Init(&CZ80);
but you can also define your own CZ80 context :
cz80_struc My_Z80;
....
Cz80_Init(&My_Z80);
You'll can emulate as many Z80 CPU as you want by defining severals CZ80 contexts.
An arbitrary pointer can also be stored in the CZ80 context.
This pointer is passed to each I/O callback:
Cz80_Set_Ctx(&CZ80, &user_context);
3) Set up your fetch region (where the Z80 will run code from) :
----------------------------------------------------------------
Cz80_Set_Fetch(&CZ80, 0x0000, 0x7FFF, (u32) your_ROM);
Cz80_Set_Fetch(&CZ80, 0xA000, 0xFFFF, (u32) your_RAM);
...
4) Set up your memory (where the Z80 will read and write data) :
----------------------------------------------------------------
Cz80_Set_ReadB(&CZ80, your_z80readbyte_function);
Cz80_Set_WriteB(&CZ80, your_z80readbyte_function);
You can improve CZ80 performance by using WORD read/write function.
For that, you need to enable the 'CZ80_USE_WORD_HANDLER' define in cz80.h file.
In this case, you'll need to add that :
#if CZ80_USE_WORD_HANDLER
Cz80_Set_ReadW(&CZ80, your_z80readword_function);
Cz80_Set_WriteW(&CZ80, your_z80readword_function);
#endif
Your read function need to be of CZ80_READ type :
typedef u32 FASTCALL CZ80_READ(u32 adr);
Your write function need to be of CZ80_WRITE type :
typedef void FASTCALL CZ80_WRITE(u32 adr, u32 data);
5) Set Up your port (where the Z80 will read and write IO data) :
-----------------------------------------------------------------
Cz80_Set_INPort(&CZ80, your_z80readport_function);
Cz80_Set_OUTPort(&CZ80, your_z80writport_function);
Your readPort function need to be of CZ80_READ type :
typedef u32 FASTCALL CZ80_READ(u32 adr);
Your writePort function need to be of CZ80_WRITE type :
typedef void FASTCALL CZ80_WRITE(u32 adr, u32 data);
6) Set Up your interrupt callback function :
--------------------------------------------
Cz80_Set_IRQ_Callback(&CZ80, your_z80irqcallback_function);
Your IRQ callback function need to be of CZ80_INT_CALLBACK type :
typedef s32 FASTCALL CZ80_INT_CALLBACK(s32 param);
If you don't understand what i am talking about here, just ignore...
it's not needed in almost case.
6) Set Up your RETI callback function :
---------------------------------------
Cz80_Set_RETI_Callback(&CZ80, your_z80reticallback_function);
Your RETI callback function need to be of CZ80_RETI_CALLBACKtype :
typedef void FASTCALL CZ80_RETI_CALLBACK();
Again, if you don't understand what i am talking about here, ignore...
7) Reset the CZ80 core before fisrt use :
-----------------------------------------
Cz80_Reset(&CZ80);
8) Do some cycles :
-------------------
Then it's time to really do some work, if you want to execute 1000 cycles, just do :
cycles_done = Cz80_Exec(&CZ80, 1000);
Cz80_Exec function return the number of cycles actually done.
Since each instruction take more than 1 cycle, Cz80_Exec will execute a bit more than
you requested, for instance here, it can return 1008 cycles instead of 1000.
In this case, adjust the number of cycle to do like that :
cycles_by_frame = 4800;
extra_cycles = 0;
while (true)
{
...
extra_cycles = CZ80_Exec(&CZ80, cycles_by_frame - extra_cycles) - cycles_by_frame;
...
}
If Cz80_Exec returns a negatif value, an error occured.
9) Do an interrupt request :
----------------------------
Cz80_Set_IRQ(&CZ80, 0);
or for a NMI :
Cz80_Set_NMI(&CZ80);
10) Cancel an interrupt request :
---------------------------------
Cz80_Clear_IRQ(&CZ80);
or for a NMI :
Cz80_Clear_NMI(&CZ80);
* Switchs
*********
There are severals switchs in the cz80.h file which permit you to configure
CZ80 depending your needs.
- CZ80_FETCH_BITS (default = 4)
This defines the number of bits to select fetch region.
This value must be 4 <= X <= 12
Greater value offers permit to have more fetch region.
In almost case, 4 is enough, but if you have fetch region smaller than 0x1000 bytes,
increase this value.
- CZ80_LITTLE_ENDIAN
Define the endianess of the target platform.
x86 CPU use Little Endian.
- CZ80_USE_JUMPTABLE
Set it to 1 to use Jump table instead of big case statement.
This can bring some small speed improvemen.
Be careful, some compiler doesn't support (computed label) so it's
saffer to not use it.
- CZ80_SIZE_OPT
Add some extras optimisation for the code size versus speed.
Minor changes anyway...
- CZ80_USE_WORD_HANDLER
See the "Set Up Memory" section for more détail.
- CZ80_EXACT
Enable accurate emulation of extended undocumented opcode and flags.
minor speed decrease when activated.
Even without that flag, CZ80 is already uite accurate, keep it
disable unless you need it or if speed isn't important for you.
- CZ80_DEBUG
Used by me, keep it disable :p
* History
*********
Version 0.90 :
--------------
* Initial release for debugging purpose ^^