mirror of
https://github.com/HarbourMasters/Shipwright.git
synced 2024-12-11 02:32:18 -05:00
479 lines
17 KiB
Markdown
479 lines
17 KiB
Markdown
|
## Installing
|
||
|
Run `make` for a single-threaded build, or `make MT=y` for a multi-threaded
|
||
|
build. Copy `libgfxd.a` to your lib directory, and `gfxd.h` to your include
|
||
|
directory.
|
||
|
|
||
|
## Example usage
|
||
|
Example source code:
|
||
|
```
|
||
|
#include <stdio.h>
|
||
|
#include <gfxd.h>
|
||
|
|
||
|
static int macro_fn(void)
|
||
|
{
|
||
|
/* Print a tab before each macro, and a comma and newline after each
|
||
|
macro */
|
||
|
gfxd_puts("\t");
|
||
|
gfxd_macro_dflt(); /* Execute the default macro handler */
|
||
|
gfxd_puts(",\n");
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int main()
|
||
|
{
|
||
|
/* Read from stdin and write to stdout */
|
||
|
gfxd_input_fd(fileno(stdin));
|
||
|
gfxd_output_fd(fileno(stdout));
|
||
|
|
||
|
/* Override the default macro handler to make the output prettier */
|
||
|
gfxd_macro_fn(macro_fn);
|
||
|
|
||
|
/* Select F3DEX as the target microcode */
|
||
|
gfxd_target(gfxd_f3dex);
|
||
|
|
||
|
/* Set the input endianness to big endian, and the word size to 4 */
|
||
|
gfxd_endian(gfxd_endian_big, 4);
|
||
|
|
||
|
/* Print an opening brace */
|
||
|
gfxd_puts("{\n");
|
||
|
|
||
|
/* Execute until the end of input, or until encountering an invalid
|
||
|
command */
|
||
|
gfxd_execute();
|
||
|
|
||
|
/* Print a closing brace */
|
||
|
gfxd_puts("}\n");
|
||
|
}
|
||
|
```
|
||
|
|
||
|
Example input (binary):
|
||
|
```
|
||
|
0xe7000000, 0x00000000,
|
||
|
0xfc127e03, 0xfffffdf8,
|
||
|
0xb900031d, 0xc8112078,
|
||
|
0xb6000000, 0x000e0000,
|
||
|
0xb7000000, 0x00012000,
|
||
|
0xfa000000, 0xffffffff,
|
||
|
0x040030bf, 0x000002e0,
|
||
|
0xb1000204, 0x00020604,
|
||
|
0xb1080a0c, 0x000a0e0c,
|
||
|
0xb10a1012, 0x000a120e,
|
||
|
0xb1140200, 0x00140016,
|
||
|
0xb8000000, 0x00000000,
|
||
|
```
|
||
|
|
||
|
Example output:
|
||
|
```
|
||
|
{
|
||
|
gsDPPipeSync(),
|
||
|
gsDPSetCombineLERP(TEXEL0, 0, SHADE, 0, 0, 0, 0, 1, COMBINED, 0, PRIMITIVE, 0, 0, 0, 0, COMBINED),
|
||
|
gsDPSetRenderMode(G_RM_FOG_SHADE_A, G_RM_AA_ZB_OPA_SURF2),
|
||
|
gsSPClearGeometryMode(G_LIGHTING | G_TEXTURE_GEN | G_TEXTURE_GEN_LINEAR),
|
||
|
gsSPSetGeometryMode(G_CULL_BACK | G_FOG),
|
||
|
gsDPSetPrimColor(0, 0, 0xFF, 0xFF, 0xFF, 0xFF),
|
||
|
gsSPVertex(0x000002E0, 12, 0),
|
||
|
gsSP2Triangles(0, 1, 2, 0, 1, 3, 2, 0),
|
||
|
gsSP2Triangles(4, 5, 6, 0, 5, 7, 6, 0),
|
||
|
gsSP1Quadrangle(5, 8, 9, 7, 0),
|
||
|
gsSP1Quadrangle(10, 1, 0, 11, 0),
|
||
|
gsSPEndDisplayList(),
|
||
|
}
|
||
|
```
|
||
|
|
||
|
## Input/output methods
|
||
|
The input consists of any number of `Gfx` packets, and the output is the
|
||
|
decompiled macros in plain-text. The endianness and microcode type of the input
|
||
|
can be set using `gfxd_endian` and `gfxd_target`.
|
||
|
|
||
|
Several methods of doing I/O are available. No method is selected by default,
|
||
|
meaning there will be no input, and any output will be discarded.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_input_buffer(const void *buf, int size)`
|
||
|
##### `void gfxd_output_buffer(char *buf, int size)`
|
||
|
Use the buffer pointed to by `buf`, of `size` bytes.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_input_fd(int fd)`
|
||
|
##### `void gfxd_output_fd(int fd)`
|
||
|
Use `read()` / `write()` with the provided file descriptor, `fd`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_input_fn_t(void *buf, int count)`
|
||
|
##### `typedef int gfxd_output_fn_t(const char *buf, int count)`
|
||
|
##### `void gfxd_input_callback(gfxd_input_fn_t *fn)`
|
||
|
##### `void gfxd_output_callback(gfxd_output_fn_t *fn)`
|
||
|
Use the provided callback function, `fn`. `fn` should copy at most `count`
|
||
|
bytes to/from `buf`, and return the number of bytes actually copied. The input
|
||
|
callback should return 0 to signal end of input.
|
||
|
|
||
|
## Handlers
|
||
|
The macro handler function is responsible for writing the output of each
|
||
|
decompiled macro. The default macro handler is `gfxd_macro_dflt`, but this can
|
||
|
be changed with `gfxd_macro_fn`. The new handler can extend the default
|
||
|
function by calling `gfxd_macro_dflt` within it, or it can override it
|
||
|
completely.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_macro_dflt()`
|
||
|
The default macro handler. Outputs the macro name, dynamic display list pointer
|
||
|
if one has been specified, and then each argument in order using the function
|
||
|
registered using `gfxd_arg_fn` (`gfxd_arg_dflt` by default), and returns zero.
|
||
|
Because it is designed to be extended, it only outputs the macro text, without
|
||
|
any whitespace or punctuation before or after. When this function is used as
|
||
|
the sole macro handler, it will output the entire display list on one line
|
||
|
without any separation between macros, which is probably not what you want.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_macro_fn(gfxd_macro_fn_t *fn)`
|
||
|
Set `fn` to be the macro handler function. `fn` can be null, in which case the
|
||
|
handler is reset to the default.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_arg_dflt(int arg_num)`
|
||
|
The default argument handler for `gfxd_macro_dflt`. For the argument with index
|
||
|
`arg_num`, calls `gfxd_arg_callbacks`, and prints the argument value if the
|
||
|
callback returns zero, or if there is no callback for the given argument.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_arg_fn(gfxd_arg_fn_t *fn)`
|
||
|
Set `fn` to be the argument handler function, called by `gfxd_macro_dflt`,
|
||
|
for each argument in the current macro, not counting the dynamic display list
|
||
|
pointer if one has been specified. `fn` can be null, in which case the handler
|
||
|
is reset to the default. This only affects the output of `gfxd_macro_dflt`, and
|
||
|
has no observable effect if `gfxd_macro_dflt` is overridden (not extended).
|
||
|
|
||
|
## Argument callbacks
|
||
|
Callbacks can be registered that will be executed when an argument of a certain
|
||
|
type is encountered. The default argument handler `gfxd_arg_dflt` will execute
|
||
|
callbacks as needed using `gfxd_arg_callbacks`. If a callback returns non-zero,
|
||
|
`gfxd_arg_dflt` will not output anything. This is to allow callbacks to
|
||
|
override the default argument output. Otherwise, `gfxd_arg_dflt` will output
|
||
|
the argument value after the callback function's output.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_arg_callbacks(int arg_num)`
|
||
|
Examines the argument with index `arg_num` and executes the callback function
|
||
|
for that argument type, if such a callback is supported and has been
|
||
|
registered. This function returns the value that was returned by the callback
|
||
|
function. If no callback function has been registered for the argument type,
|
||
|
zero is returned.
|
||
|
|
||
|
Most argument callbacks have some extra parameters containing information that
|
||
|
might be relevant to the argument that triggered the callback. The extra
|
||
|
information is extracted only from the current macro, as gfxd does not retain
|
||
|
any context information from previous or subsequent macros. If any of the extra
|
||
|
parameter values is not available in the current macro, the value for that
|
||
|
parameter is substituted with `-1` for signed parameters, and zero for unsigned
|
||
|
parameters.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_tlut_fn_t(uint32_t tlut, int32_t idx, int32_t count)`
|
||
|
##### `void gfxd_tlut_callback(gfxd_tlut_fn_t *fn)`
|
||
|
Set the callback function for palette arguments. The argument type is
|
||
|
`gfxd_Tlut`. The palette index is in `idx` and the number of colors in `count`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_timg_fn_t(uint32_t timg, int32_t fmt, int32_t siz, int32_t width, int32_t height, int32_t pal)`
|
||
|
##### `void gfxd_timg_callback(gfxd_timg_fn_t *fn)`
|
||
|
Set the callback function for texture arguments. The argument type is
|
||
|
`gfxd_Timg`. The image format is in `fmt` and `siz`, the dimensions in `width`
|
||
|
and `height`, and the palette index in `pal`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_cimg_fn_t(uint32_t cimg, int32_t fmt, int32_t siz, int32_t width)`
|
||
|
##### `void gfxd_cimg_callback(gfxd_cimg_fn_t *fn)`
|
||
|
Set the callback function for frame buffer arguments. The argument type is
|
||
|
`gfxd_Cimg`. The image format is in `fmt` and `siz`, and the horizontal
|
||
|
resolution in `width`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_zimg_fn_t(uint32_t zimg)`
|
||
|
##### `void gfxd_zimg_callback(gfxd_zimg_fn_t *fn)`
|
||
|
Set the callback function for depth buffer arguments. The argument type is
|
||
|
`gfxd_Zimg`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_dl_fn_t(uint32_t dl)`
|
||
|
##### `void gfxd_dl_callback(gfxd_dl_fn_t *fn)`
|
||
|
Set the callback function for display list arguments. The argument type is
|
||
|
`gfxd_Dl`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_mtx_fn_t(uint32_t mtx)`
|
||
|
##### `void gfxd_mtx_callback(gfxd_mtx_fn_t *fn)`
|
||
|
Set the callback function for matrix arguments. The argument type is
|
||
|
`gfxd_Mtxptr`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_lookat_fn_t(uint32_t lookat, int32_t count)`
|
||
|
##### `void gfxd_lookat_callback(gfxd_lookat_fn_t *fn)`
|
||
|
Set the callback function for lookat array arguments. The argument type is
|
||
|
`gfxd_Lookatptr`. The number of lookat structures (1 or 2) is in `count`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_light_fn_t(uint32_t light, int32_t count)`
|
||
|
##### `void gfxd_light_callback(gfxd_light_fn_t *fn)`
|
||
|
Set the callback function for light array arguments. The argument type is
|
||
|
`gfxd_Lightptr`. The number of light structures is in `count`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_seg_fn_t(uint32_t seg, int32_t num)`
|
||
|
##### `void gfxd_seg_callback(gfxd_seg_fn_t *fn)`
|
||
|
Set the callback function for segment base arguments. The argument type is
|
||
|
`gfxd_Segptr`. The segment number is in `num`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_vtx_fn_t(uint32_t vtx, int32_t num)`
|
||
|
##### `void gfxd_vtx_callback(gfxd_vtx_fn_t *fn)`
|
||
|
Set the callback function for vertex array arguments. The argument type is
|
||
|
`gfxd_Vtxptr`. The number of vertex structures is in `num`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_vp_fn_t(uint32_t vp)`
|
||
|
##### `void gfxd_vp_callback(gfxd_vp_fn_t *fn)`
|
||
|
Set the callback function for viewport arguments. The argument type is
|
||
|
`gfxd_Vp`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_uctext_fn_t(uint32_t text, uint32_t size)`
|
||
|
##### `void gfxd_uctext_callback(gfxd_uctext_fn_t *fn)`
|
||
|
Set the callback function for microcode text arguments. The argument type is
|
||
|
`gfxd_Uctext`. The size of the text segment is in `size`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_ucdata_fn_t(uint32_t data, uint32_t size)`
|
||
|
##### `void gfxd_ucdata_callback(gfxd_ucdata_fn_t *fn)`
|
||
|
Set the callback function for microcode data arguments. The argument type is
|
||
|
`gfxd_Ucdata`. The size of the data segment is in `size`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `typedef int gfxd_dram_fn_t(uint32_t dram, uint32_t size)`
|
||
|
##### `void gfxd_dram_callback(gfxd_dram_fn_t *fn)`
|
||
|
Set the callback function for generic pointer arguments. The argument type is
|
||
|
`gfxd_Dram`. The size of the data is in `size`.
|
||
|
|
||
|
## General settings
|
||
|
These functions control general input and output settings.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_target(gfxd_ucode_t ucode)`
|
||
|
Select `ucode` as the target microcode. `ucode` can be `gfxd_f3d`, `gfxd_f3db`,
|
||
|
`gfxd_f3dex`, `gfxd_f3dexb`, or `gfxd_f3dex2`. The microcode must be selected
|
||
|
before `gfxd_execute`, as no microcode is selected by default.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_endian(int endian, int wordsize)`
|
||
|
Select `endian` as the endianness of the input, and `wordsize` as the size of
|
||
|
each word in number of bytes. `endian` can be `gfxd_endian_big`,
|
||
|
`gfxd_endian_little`, or `gfxd_endian_host` (the endianness of the host
|
||
|
machine). `wordsize` can be 1, 2, 4, or 8. Big endian is selected by default,
|
||
|
with a word size of 4.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_dynamic(const char *arg)`
|
||
|
Enable or disable the use of dynamic `g` macros instead of static `gs` macros,
|
||
|
and select the dynamic display list pointer argument to be used. `arg` will be
|
||
|
used by `gfxd_macro_dflt` as the first argument to dynamic macros. If `arg` is
|
||
|
null, dynamic macros are disabled, and `gs` macros are used. Also affects the
|
||
|
result of `gfxd_macro_name`, as it will return either the dynamic or static
|
||
|
version of the macro name as selected by this setting.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_enable(int cap)`
|
||
|
##### `void gfxd_disable(int cap)`
|
||
|
Enable or disable the feature specified by `cap`. Can be one of the following;
|
||
|
- `gfxd_stop_on_invalid`: Stop execution when encountering an invalid macro.
|
||
|
Enabled by default.
|
||
|
- `gfxd_stop_on_end`: Stop execution when encountering a `SPBranchList` or
|
||
|
`SPEndDisplayList`. Enabled by default.
|
||
|
- `gfxd_emit_dec_color`: Print color components as decimal instead of
|
||
|
hexadecimal. Disabled by default.
|
||
|
- `gfxd_emit_q_macro`: Print fixed-point conversion `q` macros for fixed-point
|
||
|
values. Disabled by default.
|
||
|
- `gfxd_emit_ext_macro`: Emit non-standard macros. Some commands are valid
|
||
|
(though possibly meaningless), but have no macros associated with them, such as
|
||
|
a standalone `G_RDPHALF_1`. When this feature is enabled, such a command will
|
||
|
produce a non-standard `gsDPHalf1` macro instead of a raw hexadecimal command.
|
||
|
Also enables some non-standard multi-packet texture loading macros. Disabled by
|
||
|
default.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `void gfxd_udata_set(void *ptr)`
|
||
|
##### `void *gfxd_udata_get(void)`
|
||
|
Set or get a generic pointer that can be used to pass user-defined data in and
|
||
|
out of callback functions.
|
||
|
|
||
|
## Execution
|
||
|
Decompilation is started using the `gfxd_execute` function. When gfxd is
|
||
|
executing (i.e. after `gfxd_execute` has been entered, and before it returns),
|
||
|
the general settings and the I/O settings should not be changed.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_execute()`
|
||
|
Start executing gfxd with the current settings. For each macro, the macro
|
||
|
handler registered with `gfxd_macro_fn` is called. Execution ends when the
|
||
|
input ends, the macro handler returns non-zero, when an invalid macro is
|
||
|
encountered and `gfxd_stop_on_invalid` is enabled, or when `SPBranchList` or
|
||
|
`SPEndDisplayList` is encountered and `gfxd_stop_on_end` is enabled. If
|
||
|
execution ends due to an invalid macro, `-1` is returned. If execution ends
|
||
|
because the macro handler returns non-zero, the return value from the macro
|
||
|
handler is returned. Otherwise zero is returned.
|
||
|
|
||
|
## Macro information
|
||
|
The following functions can be used to obtain information about the current
|
||
|
macro and its arguments. They should only be used in custom handlers and
|
||
|
callbacks from within `gfxd_execute`. If used elsewhere, their behavior is
|
||
|
undefined.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_macro_offset()`
|
||
|
Returns the offset in the input data of the current macro. The offset starts
|
||
|
at zero when `gfxd_execute` is called.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_macro_packets()`
|
||
|
Returns the number of `Gfx` packets within the current macro.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `const void *gfxd_macro_data()`
|
||
|
Returns a pointer to the input data for the current macro. The data is not
|
||
|
byte-swapped. The data has a length of `sizeof(Gfx) * gfxd_macro_packets()`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_macro_id()`
|
||
|
Returns a number that uniquely identifies the current macro. The number will
|
||
|
be one of the constants in `gfxd.h`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `const char *gfxd_macro_name()`
|
||
|
Returns the name of the current macro. If the macro does not have a name (i.e.
|
||
|
it's invalid), null is returned. If a dynamic display list pointer has been
|
||
|
specified, the dynamic `g` version is returned. Otherwise the static `gs`
|
||
|
version is returned. The returned pointer is invalidated by a subsequent call
|
||
|
to `gfxd_macro_name`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_arg_count()`
|
||
|
Returns the number of arguments to the current macro, not including a dynamic
|
||
|
display list pointer if one has been specified.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_arg_type(int arg_num)`
|
||
|
Returns a number that identifies the type of the argument with index `arg_num`.
|
||
|
The number will be one of the constants in `gfxd.h`.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `const char *gfxd_arg_name(int arg_num)`
|
||
|
Returns the name of the argument with index `arg_num`. Argument names are not
|
||
|
canonical, nor are they needed for macro disassembly, but they can be useful
|
||
|
for informational and diagnostic purposes.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_arg_fmt(int arg_num)`
|
||
|
Returns the data format of the argument with index `arg_num`. The return value
|
||
|
will be `gfxd_argfmt_i` for `int32_t`, `gfxd_argfmt_u` for `uint32_t`, or
|
||
|
`gfxd_argfmt_f` for `float`. When accessing the value of the argument with
|
||
|
`gfxd_arg_value`, the member with the corresponding type should be used.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `const gfxd_value_t *gfxd_arg_value(int arg_num)`
|
||
|
Returns a pointer to the value of the argument with index `arg_num`. The value
|
||
|
is a union of type `gfxd_value_t` with the following layout;
|
||
|
```
|
||
|
typedef union
|
||
|
{
|
||
|
int32_t i;
|
||
|
uint32_t u;
|
||
|
float f;
|
||
|
} gfxd_value_t
|
||
|
```
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `const gfxd_value_t *gfxd_value_by_type(int type, int idx)`
|
||
|
Returns a pointer to the value of the argument that is of `type`, and has order
|
||
|
`idx` in all arguments of that type. An `idx` of zero returns the first
|
||
|
argument that has the specified type. If there is no argument with the given
|
||
|
type and order, null is returned.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_arg_valid(int arg_num)`
|
||
|
Returns non-zero if the argument with index `arg_num` is "valid", for some
|
||
|
definition of valid. An invalid argument generally means that the disassembler
|
||
|
found inconsistencies in the input data, or that the data can not be reproduced
|
||
|
by the current macro type. The argument still has a value that can be printed,
|
||
|
though the value is not guaranteed to make any sense.
|
||
|
|
||
|
## Custom output
|
||
|
When the default handlers are overridden or extended, the custom handler
|
||
|
functions will want to do some output of their own. The following methods are
|
||
|
available for inserting custom text into the gfxd output.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_write(const void *buf, int count)`
|
||
|
Insert `count` bytes from the buffer at `buf` into the output. The number of
|
||
|
characters written is returned.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_puts(const char *str)`
|
||
|
Insert the null-terminated string at `str` into the output. The number of
|
||
|
characters written is returned.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_printf(const char *fmt, ...)`
|
||
|
Insert the printf-formatted string described by `fmt` and additional arguments
|
||
|
into the output. Limited to 255 characters. The number of characters written is
|
||
|
returned.
|
||
|
|
||
|
---
|
||
|
|
||
|
##### `int gfxd_print_value(int type, const gfxd_value_t *value)`
|
||
|
Insert the type-formatted value into the output. The type should be one of the
|
||
|
constants in `gfxd.h`. The number of characters written is returned. The
|
||
|
macro argument with index `n` can be printed with
|
||
|
`gfxd_print_value(gfxd_arg_type(n), gfxd_arg_value(n))`.
|