Skip to content

Commit

Permalink
CAndPerms: permit clearing GL on sealed caps
Browse files Browse the repository at this point in the history
But require that all other bits in the mask provided to CAndPerms be 1
in order for a tagged sealed input to result in a tagged sealed output.

Co-authored-by: Robert Norton <[email protected]>
  • Loading branch information
nwf and ronorton committed Nov 28, 2024
1 parent acc4c92 commit bc66dbb
Show file tree
Hide file tree
Showing 2 changed files with 23 additions and 3 deletions.
4 changes: 4 additions & 0 deletions archdoc/chap-changes.tex
Original file line number Diff line number Diff line change
Expand Up @@ -45,5 +45,9 @@ \chapter{Version history}
Instead, it merely requires that the otype of the sealed input is within bounds to yield a tagged result.
The address of a sealing-root capability is now meaningful only to \rvcheriasminsnref{CSeal}.
\item[\ghissue{72},\ghpr{74}] Introduce \rvcheriasminsnref{CSetBoundsRoundDown} to facilitate constructing representable slices of buffers.
\item[\ghissue{70},\ghpr{83}] \rvcheriasminsnref{CAndPerm} can now clear \cappermG on sealed caps, so long as that is the only bit being cleared.
Previously, this was possible by round-tripping to memory, loading back through an authority lacking \cappermILG(recall \ghpr{44} and \ghissue{14} above), but not directly as a register-to-register operation.
Presently, we require that the mask provided to \rvcheriasminsnref{CAndPerm} be all-1s except possibly \cappermG;
that is, feeding the result of \rvcheriasminsnref{CGetPerm} on a sealed capability to \rvcheriasminsnref{CAndPerm} will still clear the tag of the result.
\end{description}
\end{description}
22 changes: 19 additions & 3 deletions src/cheri_insts.sail
Original file line number Diff line number Diff line change
Expand Up @@ -410,7 +410,8 @@ union clause ast = CAndPerm : (regidx, regidx, regidx)
* previous value and bits 0 to [cap_perms_width]-1 of integer register *rs2*.
* If the resulting set of permissions cannot be represented by the capability
* encoding then the result will have a (possibly empty) subset of the ANDed
* permissions. If *cs1* was sealed then *cd*.**tag** is cleared.
* permissions. If *cs1* was sealed and *rs2* codes for clearing anything other
* than Permit_Global, then *cd*.**tag** is cleared.
*/
function clause execute(CAndPerm(cd, cs1, rs2)) = {
let cs1_val = C(cs1);
Expand All @@ -419,8 +420,23 @@ function clause execute(CAndPerm(cd, cs1, rs2)) = {
let perms = getCapPerms(cs1_val);
let mask = truncate(rs2_val, cap_perms_width);

let inCap = clearTagIfSealed(cs1_val);
let newCap = setCapPerms(inCap, (perms & mask));
let newperms = perms & mask;

/*
* CAndPerm on a sealed cap clears the tag unless the mask is all ones or
* has *only* the global permission clear.
*
* Here, perm_global is a bit vector (in the architectural format used by
* [CAndPerm] and [CGetPerm]) of all zeros except for the global permission
* bit. Its formulation here relies on null_cap having no asserted
* permissions (which we verify in our SMT properties; see
* [prop_null_noperms]).
*/
let perm_global = getCapPerms({ null_cap with global = true });
let inCap = clearTagIf(cs1_val,
isCapSealed(cs1_val) & ((mask | perm_global) == ones()));

let newCap = setCapPerms(inCap, newperms);

C(cd) = newCap;
RETIRE_SUCCESS
Expand Down

0 comments on commit bc66dbb

Please sign in to comment.