Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add MC support for CHERI memory colouring instructions. #508

Open
wants to merge 6 commits into
base: dev
Choose a base branch
from

Conversation

davidchisnall
Copy link
Member

This uses @jrtc27's proposed name for the atomic RMW operation. @rmn30, let me know if you need anything beyond this to assemble test cases for Sail. @jrtc27, I've not put this behind a feature flag but it's likely to remain an optional extension on top of xCHERI for quite a while. We can either define these instructions to always exist but fail (most of them have a mechanism for reporting failure), always enable them in CodeGen but have them trap on some targets, or hide them behind a separate feature flag. I have no strong opinions either way - they're not likely to be used in IR lowering except via explicit use of intrinsics, so there isn't much of a problem enabling them unconditionally in the compiler and require code that uses them to explicitly check first.

Based on the draft addition to the ISA spec.  This is MC-only for now,
though the CodeGen parts of this are likely to only be intrinsics and so
not very difficult to implement.  This should be sufficient to write
tests for the Sail model.
@davidchisnall davidchisnall requested review from jrtc27, nwf and rmn30 January 28, 2021 11:25
(ins GPR:$rs1, GPCR:$rs2),
"cstoreversion", "$rs1, $rs2">;
let mayLoad = 1, mayStore = 1 in
def CAmoCDecVersion : Cheri_rr<0x3, "camocdecversion", GPR>;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it important to indicate that $rs1 is atomic?

Suggested change
def CAmoCDecVersion : Cheri_rr<0x3, "camocdecversion", GPR>;
def CAmoCDecVersion : Cheri_rr<0x3, "camocdecversion", GPR, GPR, GPCRMemAtomic>;

Copy link
Member

@arichardson arichardson Jan 28, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe that type exists only to allow using 0(ca0) in the assembler despite the instruction not having an immediate operand.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've not seen that before, but in other back ends you never specify anything explicitly on operands to indicate that they are the memory addresses for atomics.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes it's solely a parser thing. RISC-V's atomic operands are the only memory-accessing instructions that don't take an immediate, and have (rs1) as the canonical form but also accept 0(rs1), primarily to make it easy for generated assembly. Since our explicit loads and stores end up looking the same in terms of operands I reused the existing class (and its corresponding capability version), and did not rename it to avoid merge conflicts just for something cosmetic. In this case you do want the suggested change, but only so its assembly syntax matches the other AMOs, i.e. that 0(cs1) is permitted in the assembly.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This being the case I am tempted to (ab)use GPCRMemAtomic for the address operands to cloadversion and cstore version too so that they get brackets around the address. Thoughts?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, that's what I would do, then it matches the explicit loads/stores. Maybe one day I'll rename it to something more general (and do so upstream too, even if it currently makes less sense, but I think the hypervisor load/store-as-guest instructions are also immediate-less and will end up wanting the same syntax).

Copy link
Member

@jrtc27 jrtc27 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I would like this to be a separate (dependent) extension, otherwise people may come along, see the instructions and be surprised that they don't work. I don't mind them living in the XCheri file though, especially as that'll reduce churn if and when they become less experimental and get folded into XCheri itself.

llvm/lib/Target/RISCV/RISCVInstrInfoXCheri.td Outdated Show resolved Hide resolved
llvm/lib/Target/RISCV/RISCVInstrInfoXCheri.td Outdated Show resolved Hide resolved
@@ -1554,3 +1554,21 @@ let Predicates = [HasCheri, IsRV64, IsCapMode] in {
defm : CheriLdPat<load, CLC_128>;
defm : CheriStPat<store, CSC_128, GPCR>;
} // Predicates = [HasCheri, IsRV64, IsCapMode]


/// CHERI memory colouring extensions
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instructions go in the first half of the file to keep MC and CodeGen cleanly separated*

(* pseudos are more complicated, but placed based on whether they're used for assembly or just CodeGen)

llvm/test/MC/RISCV/cheri/rv64xcheri-version.s Outdated Show resolved Hide resolved
@@ -0,0 +1,23 @@
# RUN: llvm-mc %s -triple=riscv64 -mattr=+xcheri -show-encoding \
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These belong in rv64xcheri-valid.s as it stands

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(though a separate extension means rv64xname-valid.s for some other name)

davidchisnall and others added 2 commits January 29, 2021 09:36
Add a +xcherimemversion feature that enables them.
1. Swap the position of CStoreVersion operands so that it conforms
with other store operations (which are a bit irregular)

2. CAmoCDecVersion now takes a capability for expected version
argument.
@rmn30
Copy link
Contributor

rmn30 commented Mar 17, 2021

I've updated this to reflect some changes to the memory versioning spec.

… and camocdecversion.

This slightly uses GPCRMemAtomic which exists only to support memory operands without an immediate and could be renamed.
let mayLoad = 1, mayStore = 1, hasSideEffects = 0 in
def CAmoCDecVersion : RVInstR<0x3, 0, OPC_CHERI, (outs GPR:$rd),
(ins GPCRMemAtomic:$rs1, GPCR:$rs2),
"camocdecversion", "$rd, $rs1, $rs2">;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
"camocdecversion", "$rd, $rs1, $rs2">;
"camocdecversion", "$rd, $rs2, $rs1">;

to match stores and atomics

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants