kolibrios/programs/emulator/dgen-sdl-1.33/cz80
turbocat ea1a60faa3 Upload DGEN port source
git-svn-id: svn://kolibrios.org@9837 a494cfbc-eb01-0410-851d-a64ba20cac60
2022-06-15 18:25:17 +00:00
..
cz80_op.inc
cz80_opcb.inc
cz80_oped.inc
cz80_opxy.inc
cz80_opxycb.inc
cz80.c
cz80.h
cz80.inc
cz80exec.inc
cz80jmp.inc
readme.txt

************************************************
*                                              *
*     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 ^^