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.
Clone the rust source and checkout this specific commit:
git clone https://github.com/rust-lang/rust.git
cd rust
git checkout 4c5f6e6277b89e47d73a192078697f7a5f3dc0ac
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
sed -i 's/#lld = false/lld = true/' config.toml
sed -i 's/#targets.*$/targets = "Mips;X86"/' config.toml
sed -i 's/#experimental-targets.*$/experimental-targets = ""/' config.toml ```
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
Patch the rust compiler. Applying this patch to a different commit may require manual intervention:
git apply /path/to/patches/rustc_psx.patch
Build the rust compiler:
```
./x.py build -i library/std
./x.py build -i library/std --keep-stage 1 ```
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.
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
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 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 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 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
.
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
// This can be any import from the psx
crate, but there must be at least one.
use psx;
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 !
.
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.
Use the following to run the tests in the psx crate.
cd psx/
cargo psx test