psx-sdk-rs

This is a basic SDK to run custom Rust code on a PlayStation 1. You'll need to build the rust compiler from source. Building the compiler is computationally expensive, so it may take quite a bit of time. See the system requirements for building the rust compiler for more specifics.

Building the compiler

  1. Clone the rust source and checkout this specific commit:

    git clone https://github.com/rust-lang/rust.git cd rust git checkout 4c5f6e6277b89e47d73a192078697f7a5f3dc0ac

  2. Configure the build script to use rust-lld and optionally remove unnecessary targets to speed up the LLVM build:

    ``` cp config.toml.example config.toml

    Set lld to true

    sed -i 's/#lld = false/lld = true/' config.toml

    Only build the MIPS and X86 targets

    sed -i 's/#targets.*$/targets = "Mips;X86"/' config.toml

    Don't build any experimental targets

    sed -i 's/#experimental-targets.*$/experimental-targets = ""/' config.toml ```

  3. Optionally patch LLVM to enable the nightlier feature. To use this feature set max_atomic_width: Some(0), to Some(16) in patches/rustc_psx.patch before applying it.

    git submodule update --init --progress src/llvm-project cd src/llvm-project git apply /path/to/patches/llvm_atomic_fence.patch

  4. Patch the rust compiler. Applying this patch to a different commit may require manual intervention:

    git apply /path/to/patches/rustc_psx.patch

  5. Build the rust compiler:

    ```

    For the initial build

    ./x.py build -i library/std

    To rebuild

    ./x.py build -i library/std --keep-stage 1 ```

  6. Create a new toolchain with the patched compiler:

    rustup toolchain link psx build/x86_64-unknown-linux-gnu/stage1

    where psx is the name for the new toolchain.

Installing cargo-psx

cargo-psx is an optional wrapper for cargo that sets some commonly required flags and arguments. Basically this lets you just run cargo psx run instead of

RUSTFLAGS="-Ccodegen-units=1 -Crelocation-model=static \ -Clink-arg=-Tpsexe.ld -Clink-arg=--oformat=binary" \ cargo run +psx -Zbuild-std=core \ -Zbuild-std-features=compiler-builtins-mem \ --target mipsel-sony-psx

which has the minimum number of flags required to create and run an exe.

To install:

cargo install cargo-psx

To uninstall:

cargo uninstall cargo-psx

For more options:

cargo psx --help

Usage

The examples directory has some demos which have been tested with the SCPH7001 NA BIOS in mednafen, pcsx-redux and/or duckstation. Other BIOS versions/regions may work but have not been tested. To try out a demo first check that you have the emulator in examples/$DEMO/.cargo/config.toml installed, then run cargo psx run from its directory.

To manually select the executable in your emulator, first build with cargo psx build then open the .exe in /path/to/crate/target/mipsel-sony-psx/release/. To use cargo psx run with other emulators change the runner for the mipsel-sony-psx target. To package the executable into an ISO (e.g. to use the CD-ROM filesystem) use mkpsxiso or similar.

Mednafen

Mednafen tends to just work with the following .cargo/config.toml. Getting stdout may require setting psx.dbg_level in mednafen.cfg to at least 2. When loading ISOs there doesn't seem to be a fastboot (skip BIOS) option, so ISOs must be licensed properly since the BIOS validates some checksums.

[target.mipsel-sony-psx] runner = "mednafen"

PCSX-Redux

PCSX-Redux works with the following .cargo/config.toml. Note that -loadexe must be the last option since cargo psx run passes the executable filename at the end. See this page for more command line options.

[target.mipsel-sony-psx] runner = ["pcsx-redux", "-stdout", "-run", "-loadexe"]

To run an ISO from the command line use pcsx-redux -stdout -run -iso $ISO_FILENAME.

DuckStation

DuckStation works and is extremely useful for debugging, but it doesn't work with cargo psx run. The issue is that the command passes the runner a relative path to the executable, but DuckStation only accepts absolute paths on the command line. To use DuckStation, build with cargo psx build then open the executable manually. To run an ISO use duckstation-qt -fastboot /full/path/to/$ISO_FILENAME.

Program template

To create a new program just use cargo init and add psx = "*" or psx = { path = "/path/to/psx/crate" } to Cargo.toml under [dependencies]. Then replace src/main.rs with the following template

```rust

![no_std]

![no_main]

// This can be any import from the psx crate, but there must be at least one. use psx;

[no_mangle]

fn main() { } ```

no_std is required to link the core crate instead of std. no_main tells rustc to make no assumption about the program's entry point. The attribute on main is required since the entry point defined in the psx crate expects to call an unmangled function. The main function should not return, but the return type can be either omitted (i.e. ()) or !.

Documentation

See docs.rs for documentation. To regenerate documentation for the latest version of psx:

cd psx cargo doc --target mipsel-unknown-linux-gnu

Then open target/mipsel-unknown-linux-gnu/doc/psx/index.html in a browser.

For documentation on PlayStation internals see the nocash specs.

Testing

Use the following to run the tests in the psx crate.

cd psx/ cargo psx test