/mini-gdbstub

An implementation of the GDB Remote Serial Protocol to help you adding debug mode on emulator

Primary LanguageCMIT LicenseMIT

mini-gdbstub

mini-gdbstub is an implementation of the GDB Remote Serial Protocol that gives your emulators debugging capabilities.

Usage

The very first thing you should do is to build the statically-linked library libgdbstub.a.

make

To use the library in your project, you should include the file gdbstub.h first. Then, you have to initialize the pre-allocated structure gdbstub_t with gdbstub_init.

bool gdbstub_init(gdbstub_t *gdbstub, struct target_ops *ops, arch_info_t arch, char *s);

The parameters s is the easiest one to understand. It is a string of the socket which your emulator would like to bind as gdb server.

The struct target_ops is made up of function pointers. Each member function represents an abstraction of your emulator's operation. The following lists the requirement that should be provided for each method:

Method Description
cont Run the emulator until hitting breakpoint or exit.
stepi Do one step on the emulator. You may define your own step for the emulator. For example, the common design is executing one instruction.
read_reg Read the value of the register specified by regno to *value. Return zero if the operation success, otherwise return an errno for the corresponding error.
write_reg Write value value to the register specified by regno. Return zero if the operation success, otherwise return an errno for the corresponding error.
read_mem Read the memory according to the address specified by addr with size len to the buffer *val. Return zero if the operation success, otherwise return an errno for the corresponding error.
write_mem Write data in the buffer val with size len to the memory which address is specified by addr. Return zero if the operation success, otherwise return an errno for the corresponding error.
set_bp Set type type breakpoint on the address specified by addr. Return true if we set the breakpoint successfully, otherwise return false.
del_bp Delete type type breakpoint on the address specified by addr. Return true if we delete the breakpoint successfully, otherwise return false.
on_interrupt Do something when receiving interrupt from GDB client. This method will run concurrently with cont, so you should be careful if there're shared data between them. You will need a lock or something similar to avoid data race.
struct target_ops {
    gdb_action_t (*cont)(void *args);
    gdb_action_t (*stepi)(void *args);
    int (*read_reg)(void *args, int regno, size_t *value);
    int (*write_reg)(void *args, int regno, size_t value);
    int (*read_mem)(void *args, size_t addr, size_t len, void *val);
    int (*write_mem)(void *args, size_t addr, size_t len, void *val);
    bool (*set_bp)(void *args, size_t addr, bp_type_t type);
    bool (*del_bp)(void *args, size_t addr, bp_type_t type);
    void (*on_interrupt)(void *args);
};

For cont and stepi which are used to process the execution of emulator, their return type should be gdb_action_t. After performing the relevant operation, you should return ACT_RESUME to continue debugging; otherwise, return ACT_SHUTDOWN to finish debugging. The library typically uses ACT_NONE to take no action.

typedef enum {
    ACT_NONE,
    ACT_RESUME,
    ACT_SHUTDOWN,
} gdb_action_t;

For set_bp and del_bp, the type of breakpoint which should be set or deleted is described in the type bp_type_t. In fact, its value will always be BP_SOFTWARE currently.

typedef enum {
    BP_SOFTWARE = 0,
} bp_type_t;

Another structure you have to declare is arch_info_t. You must explicitly specify "mini-gdbstub" about size in bytes (reg_byte) and the number of target registers (reg_num) within arch_info_t structure while integrating into your emulator. The target_desc is an optional member which could be TARGET_RV32 or TARGET_RV64 if the emulator is RISC-V 32-bit or 64-t instruction set architecture, otherwise you would simply set it to NULL.

  • Although the value of reg_num and reg_byte may be determined by target_desc, those members are still required to be filled correctly.
typedef struct {
    char *target_desc;
    int reg_num;
    size_t reg_byte;
} arch_info_t;

After startup, we can use gdbstub_run to run the emulator as gdbstub. The args can be used to pass the argument to any function in struct target_ops.

bool gdbstub_run(gdbstub_t *gdbstub, void *args);

When exiting from gdbstub_run, gdbstub_close should be called to recycle the resource on the initialization.

void gdbstub_close(gdbstub_t *gdbstub);

Finally, you can build you project with the statically-linked library libgdbstub.a now! Additionally, it is advised that you check the reference emulator in the directory emu, which demonstrates how to integrate mini-gdbstub into your project.

Reference

Project

Article