Skip to content

Latest commit

 

History

History
431 lines (325 loc) · 15.9 KB

README.md

File metadata and controls

431 lines (325 loc) · 15.9 KB

Project moved

This is the java version of Spice86. It is not maintained anymore.

We had to rewrite the project in C# to add automated code generation (java doesn't have the goto keyword, making automated ASM translation challenging)

New repo is here: https://github.com/OpenRakis/Spice86/

Spice86 - A PC emulator for real mode reverse engineering

Spice86 is a tool to execute, reverse engineer and rewrite real mode dos programs for which source code is not available.

It requires java 17.

Approach

Rewriting a program from only the binary is a hard task.

Spice86 is a tool that helps you do so with a methodic divide and conquer approach.

General process:

  • You start by emulating the program in the Spice86 emulator.
  • This emulator allows you to gradually reimplement the assembly code with your java methods
  • This is helpful because:
    • Small sequences of assembly can be statically analyzed and are generally easy to translate to a higher level language.
    • You work all the time with a fully working version of the program so it is relatively easy to catch mistakes early.
    • Rewriting code function by function allows you to discover the intent of the author.

Running your exe

This is a java program, you run it with the regular java -jar command. Example with running a program called file.exe:

java -jar spice86-1.27.0.jar file.exe

com files and bios files are also supported.

Dynamic analysis

Spice86 speaks the GDB remote protocol:

  • it supports most of the commands you need to debug.
  • it also provides custom GDB commands to do dynamic analysis. This is where the magic happens.

Connecting

You need to specify a port for the GDB server to start when launching Spice86:

java -jar spice86-1.27.0.jar file.exe --gdbPort=10000

Spice86 will wait for GDB to connect before starting execution so that you can setup breakpoints and so on.

Here is how to connect from GDB command line client and how to set the architecture:

(gdb) target remote localhost:10000
(gdb) set architecture i8086

Vanilla GDB

You can add breakpoints, step, view memory and so on.

Example with a breakpoint on VGA VRAM writes:

(gdb) watch *0xA0000

Viewing assembly:

(gdb) layout asm

Removing a breakpoint:

(gdb) remove 1

Searching for a sequence of bytes in memory (start address 0, length F0000, ascii bytes of 'Spice86' string):

(gdb) find /b 0x0, 0xF0000, 0x53, 0x70, 0x69, 0x63, 0x65, 0x38, 0x36

GDB does not support x86 real mode segmented addressing, so pointers need to refer to the actual physical address in memory. VRAM at address A000:0000 would be 0xA0000 in GDB.

Similarly, The $pc variable in GDB will be exposed by Spice86 as the physical address pointed by CS:IP.

Custom commands (where the magic happens)

The list of custom commands can be displayed like this:

(gdb) monitor help

Dump everything

(gdb) monitor dumpall

Dumps everything described below in one shot. Files are created in the current execution folder.

Dump the memory to a file

(gdb) monitor dumpmemory path/to/dump.bin

DOS programs can rewrite some of their instructions / load additional modules in memory. It's a good idea to get a memory dump to see the actual assembly being executed.

Dump the functions

(gdb) monitor dumpfunctions path/to/functions.txt

This will dump dynamic information about the functions that have been encountered in the program you are reverse engineering. For each function:

  • Their address (both in segmented and physical addressing)
  • Their name if you provided an override (more about that later)
  • The addresses of the returns that have been reached and their type (NEAR / FAR / INTERRUPT / MACHINE)
  • The addresses of the returns that did not make the RET instruction point to the expected caller (some programs use RET as jump ...)
  • The list of functions that calls it
  • The list of functions it calls

Example:

function unknown_0x2538_0x151_0x254D1 returns:3 callers:1 called: 4 calls:3 approximateSize:11482
 - ret: FAR 0x2538:0x26AF/0x27A2F
 - ret: FAR 0x2538:0x2D41/0x280C1
 - ret: FAR 0x2538:0x2E2B/0x281AB
 - caller: unknown_0x1ED_0xC108_0xDFD8
 - call: vgaDriver.loadPalette_0x2538_0xB68_0x25EE8 overriden
 - call: vgaDriver.waitForRetraceInTransitions_0x2538_0x2572_0x278F2 overriden
 - call: unknown_0x2538_0x2596_0x27916

Here you can see that:

  • The generated name unknown_0x2538_0x151_0x254D1 can be copy pasted directly in java to start overriding it.
  • The physical address of the function is 0x254D1 in RAM (2538:0151 segmented)
  • It spawns 11482 bytes (estimated distance between the entry point and the returns)
  • Emulator encounterd several returns and it is called by one caller only
  • It calls 3 other methods and 2 are overriden already

You can also dump the functions as CSV for import and processing in a spreadsheet:

(gdb) monitor dumpfunctionscsv path/to/functions.txt

Generate Java / Kotlin code

#Java
(gdb) monitor dumpJavaStubs path/to/stub.java

#Kotlin
(gdb) monitor dumpKotlinStubs path/to/stub.kt

This will generate java / kotlin source code with:

  • The function calls and how to override them
  • Accessors for global variables (memory bytes accessed via hardcoded address)

Special breakpoints

Break after x emulated CPU Cycles:

(gdb) monitor breakCycles 1000

Break at the end of the emulated program:

(gdb) monitor breakStop

Displaying additional buffers

Sometimes it's useful to see what kind of images programs are generating in memory before display.

# Displays the content of memory at address 0x123AB as a buffer of resolution 320x200
(gdb) monitor vbuffer add 0x123AB 320x200

# Displays the content of memory at address 0x123AB as a buffer of resolution 16x16, scaling it 10 times so that it's not tiny
(gdb) monitor vbuffer add 0x123AB 16x16 10

# Remove the buffer display
(gdb) monitor vbuffer remove 0x123AB

# List information about currently displayed buffers
(gdb) monitor vbuffer list

#Refreshing screen or buffers while debugging
(gdb) monitor vbuffer refresh

Detailed reverse engineering process

Concrete example with Cryo Dune here.

First run your program and make sure everything works fine in Spice86. If you encounter issues it could be due to unimplemented hardware / DOS / BIOS features.

Then run your it with the GDB server enabled and set a breakpoint on emulated program stop:

(gdb) monitor breakStop
(gdb) continue

Run some interesting actions in your emulated program and quit it from the program. If you don't have this option, you can also break after a defined number of Cycles with breakCycles.

When GDB gives you control due to breakpoint being reached:

  • Dump the memory with monitor dumpmemory
  • Dump the functions with monitor dumpfunctions
  • Dump the functions as CSV with monitor dumpfunctionscsv
  • Dump the java stubs with monitor dumpjavastubs

Open the CSV file in a spreadsheet, filter functions that are overridable and not overriden:

Overridable means that the function calls no other function, or that it calls only overridden functions as calling Java from ASM is supported but ASM from Java is not.

If you sort by approximate size, you are likely to get the easiest targets first.

Note that approximate size does not always reflect the real size of the function as it is the difference between entry point address and encountered return address. A small function can sometimes be a monster that only got partially executed.

Also note that other values like callers can be wrong because sometimes the programs use returns to do jumps and this messes up the call stack analysis.

Open the memory dump in a disassembler / decompiler (I personally use ghidra).

In the screenshot, physical address of unknown_0x1ED_0xA1E8_0xC0B8 will be C0B8. The name contains both the segment:offset and physical addresses.

Go to this address:

As you can see, it is 2 lines and is very simple:

  • Instruction at address C0B8 increases the byte at address DS:47A8 by one
  • Instruction at address C0BC does a near ret

From there you can re-implement (override) the function and continue with the next one (see next chapter on how to do so).

Once you have a stub or the function implemented, you can put a java breakpoint in it to get a better understanding on how when the function is called and how it interacts with the rest of the code.

It is useful to document the relevant inputs and outputs.

To get a better understanding of the environment of a function, and especially what it calls, you can check it in the dump provided by dumpfunctions

Overriding emulated code with Java code

This is where things start to get fun!

You can provide your own Java code to override the program original assembly code.

Defining overrides

Spice86 can take in input an instance of spice86.emulator.function.OverrideSupplier that builds a mapping between the memory address of functions and their java overrides.

For a complete example you can check the source code of Cryo Dune RE.

Here is a simple example of how it would look like:

package my.program;

// This class is responsible for providing the overrides to spice86.
// There is only one per program you reimplement.
public class MyProgramOverrideSupplier implements OverrideSupplier {
  @Override
  public Map<SegmentedAddress, FunctionInformation> generateFunctionInformations(int programStartSegment,
                                                                                 Machine machine) {
    Map<SegmentedAddress, FunctionInformation> res = new HashMap<>();
    // In more complex examples, overrides may call each other
    new MyOverrides(res, programStartSegment, machine);
    return res;
  }

  @Override
  public String toString() {
    return "Overrides My program exe. class is " + this.getClass().getCanonicalName();
  }
}

// This class contains the actual overrides. As the project grows, you will probably need to split the reverse engineered code in several classes.
public class MyOverrides extends JavaOverrideHelper {
  private MyOverridesGlobalsOnDs globalsOnDs;

  public MyOverrides(Map<SegmentedAddress, FunctionInformation> functionInformations, int segment, Machine machine) {
    // "myOverides" is a prefix that will be appended to all the function names defined in this class
    super(functionInformations, "myOverides", machine);
    globalsOnDs = new MyOverridesGlobalsOnDs(machine);
    // incUnknown47A8_0x1ED_0xA1E8_0xC0B8 will get executed instead of the assembly code when a call to 1ED:A1E8 is performed.
    // Also when dumping functions, the name myOverides.incUnknown47A8 or instead of unknown
    // Note: the segment is provided in parameter as spice86 can load executables in different places depending on the configuration
    defineFunction(segment, 0xA1E8, "incDialogueCount47A8", this::incDialogueCount47A8_0x1ED_0xA1E8_0xC0B8);
    defineFunction(segment, 0x0100, "addOneToAX", this::addOneToAX_0x1ED_0x100_0x1FD0);
  }

  public Runnable incDialogueCount47A8_0x1ED_0xA1E8_0xC0B8() {
    // Accessing the memory via accessors
    globalsOnDs.setDialogueCount47A8(globalsOnDs.getDialogueCount47A8() + 1);
    // Depends on the actual return instruction performed by the function, needed to be called from the emulated code as
    // some programs like to mess with the stack ...
    return nearRet();
  }

  private Runnable addOneToAX_0x1ED_0x100_0x1FD0() {
    // Assembly for this would be
    // INC AX
    // RETF
    // Note that you can access the whole emulator to change the state in the overrides.
    state.setAX(state.getAX() + 1);
    return nearRet();
  }
}

// Memory accesses can be encapsulated into classes like this to give names to addresses and make the code shorter.
public class MyOverridesGlobalsOnDs extends MemoryBasedDataStructureWithDsBaseAddress {
  public DialoguesGlobalsOnDs(Machine machine) {
    super(machine);
  }

  public void setDialogueCount47A8(int value) {
    this.setUint8(0x47A8, value);
  }

  public int getDialogueCount47A8() {
    return this.getUint8(0x47A8);
  }
}

To avoid mistakes, you can copy paste the java stubs generated by monitor dumpjavastubs

Loading overrides

Let's suppose that the overrides defined in the previous paragraph are in overrides.jar. Here is how you could launch Spice86 to use them:

java -cp 'overrides.jar:spice86-1.27.0.jar' spice86.main.Main file.exe --overrideSupplierClassName=my.program.MyProgramOverrideSupplier

If you just want to use the function names and not the overrides, you could add --useCodeOverride=false to the command line.

If you build a project around this, just call Spice86 like this in your main:

public static void main(String[] args) {
  Spice86Application.runWithOverrides(args, MyProgramOverrideSupplier.class);
}

Generating overrides

The command dumpJavaStubs generates a text file with some java stubs that could be generated automatically.

(gdb) monitor dumpJavaStubs path/to/stubs.txt

Generated stub look like this:

...
// defineFunction(0x2538, 0x151, "unknown", this::unknown_0x2538_0x151_0x254D1);
public Runnable unknown_0x2538_0x151_0x254D1() {
  return farRet();
}
        ...

You can copy paste the stub to your code.

Misc

C Drive

It is possible to provide a C: Drive for emulated DOS functions with the option --cDrive. Default is current folder. For some games you may need to set the C drive to the game folder.

Emulated program arguments

You can pass arguments (max 127 chars!) to the emulated program with the option --exeArgs. Default is empty.

Time

The emulated Timer hardware of the PC (Intel 8259) supports measuring time from either:

  • The real elapsed time. Speed can be altered with parameter --timeMultiplier.
  • The number of instructions the emulated CPU executed. This is the behaviour is activated with parameter --instructionsPerSecond and is forced when in GDB mode so that you can debug with peace of mine without the timer triggering.

Screen refresh

Screen is refreshed 30 times per second and each time a VGA retrace wait is detected (see VideoBiosServicesDispatcher::tick3DA).

Emulator features

CPU:

  • Only 16 bits instructions are supported, memory size is 1MB
  • The only supported addressing mode is real mode. 286/386 Protected mode and the related instructions are not implemented.
  • Instruction set is (hopefully!) fully implemented for 8086, and validated via automated tests.
  • For 80186, BOUND, ENTER and LEAVE instructions are missing.
  • For 80286, instructions related to protected mode are not implemented
  • For 80386, protected mode and 32 bits instructions are not implemented. FS and GS registers are supported.
  • No FPU instruction implemented apart those used for FPU detection.

Graphics:

  • Only VGA mode 0x13 is implemented

DOS:

  • Part of int 21 is implemented. Identifies itself as dos 5.0 for now.

Input:

  • Keyboard
  • Mouse (callback in mouse driver not implemented yet)
  • No joystick for now

CD-ROM:

  • No MSCDEX support for now. Some games, like DUNE, can be copied entirely from the CD and run from the hard drive.

Sound:

  • No sound for now, games will not detect sound blaster or adlib.

Compatibility list available here.

How to build

Locally you will need:

  • Java 17
  • Maven 3.6+
mvn clean install

How to debug

The main class is

spice86.main.Main

If you get this error:

Error: JavaFX runtime components are missing, and are required to run this application

Then the selected main class is the wrong one.

Some screenshots

Cryo dune:

Prince of persia:

Stunts: