Add the following rustflags to .cargo/config.toml
in your project. Take care not to overwrite any existing ones.
```toml rustflags = [
"-C", "target-feature=+s32c1i",
"--cfg", 'targethasatomic="8"', "--cfg", 'targethasatomic="16"', "--cfg", 'targethasatomic="32"', "--cfg", 'targethasatomic="ptr"', ] ```
Include this line of code somewhere in main.rs
rust
use xtensa_atomic_emulation_trap as _;
and build the project. The core::sync::atomic
API will now be available.
We build code for silicon that has the s32c1i
feature, then when our target attempts to execute these instructions it throws an illegal instruction exception, at which point we can decode the instruction and emulate it in software.
There is only one atomic instruction to emulate in the Xtensa ISA, S32C1I
.
However, compare values are written to the SCOMPARE1
(Special Reg No. 12) register, so in silicon without this feature it won't exist. We need to emulate the instruction and the register for sucessful atomic emulation.
See 4.3.14.2 of the ISA RM for an example atomic compare swap loop.
| Instruction | Format | Instruction composition |
| ------------ | ------ | ------------------------------- |
| WSR | RSR | 0001_0011_0000_0000_0000_0000
|
| S32C1I | RRI8 | XXXX_XXXX_1110_XXXX_XXXX_0010
|
To emulate the WSR instruction, we must first decode it and verify that the target register is 12, the SCOMPARE1
register. Once that is confirmed, we can use this crates virtual SCOMPARE1
to store the value.
Emulation of the S32C1I
instruction is a little more complicated. First we decode the entire instruction to get the following values:
target register
- this contains the new value we wish to swap tosource register
- this conaints the address in memory of the current valueoffset
- optional offset to add to the address in the source registerWe deference the source address
+ offset
to find the current value and compare it to the stored value inside our SCOMPARE1
virtual register. If they are equal, the new target value is written to memory at the source address
+ offset
. Regardless of whether the new value is written the old value is always written back into the target register.
Follow the instructions here, and also add the following.
toml
[profile.dev.package.xtensa-atomic-emulation-trap]
opt-level = 'z'