-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathchap-isaref-riscv.tex
314 lines (263 loc) · 10.4 KB
/
chap-isaref-riscv.tex
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
\chapter{The CHERI-RISC-V Instruction-Set Reference}
\label{chap:isaref-riscv}
\input{def-riscv-insns}
\def\rvcheriasminsnref#1{#1}
\def\rvcheriasminsnnoref#1{#1}
\providecommand{\rvcheriasmfmt}{}
\renewcommand{\rvcheriasmfmt}[2][]{%
#2%
\ifthenelse{\equal{#1}{}}{%
}{%
~{\textit{\footnotesize{(#1)}}}%
}%
}
In this chapter, we specify each instruction via both informal descriptions
and code in the Sail language.
To allow for more succinct code descriptions, we rely on a number of
common function definitions and constants also described in this chapter.
\section{Sail language used in instruction descriptions}
The instruction descriptions contained in this chapter are accompanied
by code in the Sail language taken from the Sail
CHERI-RISC-V implementation~\cite{sail-cheri-riscv}.
Sail is a domain specific imperative language designed for describing
processor architectures. It has a compiler that can output executable
code in OCaml or C for building executable models, and can also
translate to various theorem prover languages for automated reasoning
about the ISA.
A brief description of the Sail language features used in this chapter can be
found in \cref{chap:sail}.
For a full description see the Sail language documentation~\cite{sail-url}.
\section{Constant Definitions}
The following constants are used in various type and function definitions throughout the specification.
The concrete values listed here apply to the CHERI-RISC-V ISA with
128-bit capabilities that extends the base 64-bit RISC-V ISA.
The constants for with 64-bit CHERI capabilities (extending the 32-bit RISC-V ISA)
can be found in the CHERI-RISC-V Sail model~\cite{sail-cheri-riscv}.
\arnote{and are not listed here as they may change in the near future?}
\medskip
% Note: we can use \sailRISCVtype{foo\_bar} instead after rems-project/sail#100
\phantomsection\label{sailRISCVzxlen}
\sailRISCVtype{xlen}
\sailRISCVtypecapAddrWidth{}
\sailRISCVtypecapLenWidth{}
\sailRISCVtypecapSizze{}
\sailRISCVtypecapMantissaWidth{}
% FIXME: These extra labels are required to allow markdown saildoc references such as [cap_uperms_width] to work correctly.
\phantomsection\label{sailRISCVzcapzyhpermszywidth}
\sailRISCVtypecapHpermsWidth{}
\phantomsection\label{sailRISCVzcapzyupermszywidth}
\sailRISCVtypecapUpermsWidth{}
\phantomsection\label{sailRISCVzcapzyupermszyshift}
\sailRISCVtypecapUpermsShift{}
\phantomsection\label{sailRISCVzcapzyflagszywidth}
\sailRISCVtypecapFlagsWidth{}
\phantomsection\label{sailRISCVzcapzyotypezywidth}
\sailRISCVtypecapOtypeWidth{}
\phantomsection\label{sailRISCVzcapzymaxzyotype}
\sailRISCVletreservedOtypes{}
\sailRISCVletcapMaxOtype{}
\phantomsection\label{sailRISCVzcapszyperzycachezyline}
\sailRISCVtypecapsPerCacheLine{}
\section{Function Definitions}
This section contains descriptions of convenience functions used by the Sail code featured in this chapter.
\subsection*{Functions for integer and bit vector manipulation}
The following functions convert between bit vectors and integers and manipulate bit vectors:
\medskip
\sailRISCVval{unsigned}
\sailRISCVval{signed}
\sailRISCVval{to\_bits}
% Technically this is not a vector, but it seems appropriate to put these together
\sailRISCVval{bool\_to\_bit}
\sailRISCVval{bool\_to\_bits}
\sailRISCVval{truncate}
\sailRISCVval{pow2}
\sailRISCVval{MAX}
% The following are overloads so we can't easily use the generated latex
% Hacky hspace to get rid of unwanted hangindent
\phantomsection
\label{sailRISCVzEXTZ}
\saildocval{Adds zeros in most significant bits of vector to obtain a vector of desired length.}{\hspace{-\parindent}\isail{EXTZ}}
\label{sailRISCVzEXTS}
\saildocval{Extends the most significant bits of vector preserving the sign bit.}{\hspace{-\parindent}\isail{EXTS}}
\label{sailRISCVzzzeros}
\saildocval{Produces a bit vector of all zeros}{\hspace{-\parindent}\isail{zeros}}
\label{sailRISCVzzones}
\saildocval{Produces a bit vector of all ones}{\hspace{-\parindent}\isail{ones}}
\subsection*{Types used in function definitions}
\sailRISCVtype{CapBits}
\sailRISCVtype{CapAddrBits}
\sailRISCVtype{CapLenBits}
\sailRISCVtype{CapPermsBits}
% Expanding this looks very ugly \sailRISCVtype{Capability}
\medskip
\noindent
Many functions also use \isail{struct Capability}, a structure holding a
partially-decompressed representation of CHERI capabilities.
%
The following functions can be used to convert between the structure
representation and the raw capability bits:
\medskip%
\sailRISCVval{capBitsToCapability}
\sailRISCVval{capToBits}
\subsection*{Functions for reading and writing registers and memory}
%\label{sailRISCVzC}
\begin{lstlisting}[language=sail,label=sailRISCVzC]
C(n) : regno -> Capability
C(n) : (regno, Capability) -> unit
\end{lstlisting}
\hspace{\parindent}
The overloaded function \isail{C(n)} is used to read or write capability register \isail{n}.
\medskip
\begin{lstlisting}[language=sail,label=sailRISCVzX]
X(n) : regno -> xlenbits
X(n) : (regno, xlenbits) -> unit
\end{lstlisting}
\hspace{\parindent}
The overloaded function \isail{X(n)} is used to read or write integer register \isail{n}.
\medskip
\begin{lstlisting}[language=sail,label=sailRISCVzF]
F(n) : regno -> xlenbits
F(n) : (regno, xlenbits) -> unit
\end{lstlisting}
\hspace{\parindent}
The overloaded function \isail{F(n)} is used to read or write floating-point register \isail{n}.
\medskip
\sailRISCVval{memBitsToCapability}
\sailRISCVval{capToMemBits}
\sailRISCVval{int\_to\_cap}
\sailRISCVval{mem\_read\_cap}
\medskip
\sailRISCVval{ddc\_and\_resulting\_addr}
\sailRISCVval{get\_cheri\_mode\_cap\_addr}
\sailRISCVval{handle\_load\_cap\_via\_cap}
\sailRISCVval{handle\_load\_data\_via\_cap}
\sailRISCVval{handle\_store\_cap\_via\_cap}
\sailRISCVval{handle\_store\_data\_via\_cap}
\subsection*{Functions for ISA exception behavior}
\sailRISCVval{handle\_exception}
\sailRISCVval{handle\_illegal}
\sailRISCVval{handle\_mem\_exception}
\sailRISCVval{handle\_cheri\_cap\_exception}
\sailRISCVval{handle\_cheri\_reg\_exception}
\sailRISCVval{handle\_cheri\_pcc\_exception}
\medskip
\sailRISCVval{pcc\_access\_system\_regs}
\sailRISCVval{privLevel\_to\_bits}
\sailRISCVval{min\_instruction\_bytes}
\medskip
\sailRISCVval{legalize\_epcc}
\sailRISCVval{legalize\_tcc}
% TODO: \subsection*{Functions for control flow}
\subsection*{Functions for manipulating capabilities}
The Sail code abstracts the capability representation using the following functions for getting and setting fields in the capability.
The base of the capability is the address of the first byte of memory to which it grants access and the top is one greater than the last byte, so the set of dereferenceable addresses is:
\[
\{ a \in \mathbb{N} \mid \mathit{base} \leq a < \mathit{top}\}
\]
Note that for 128-bit capabilities $\mathit{top}$ can be up to $2^{64}$, meaning the entire 64-bit address space can be addressed.
\medskip
\sailRISCVval{getCapBounds}
\sailRISCVval{getCapBaseBits}
\sailRISCVval{getCapTop}
\sailRISCVval{getCapLength}
\sailRISCVval{inCapBounds}
\noindent The capability's address (also known as cursor) and offset (relative to base) are related by:
\[
\mathit{base} + \mathit{offset}\ \mathbf{mod}\ 2^{64} = \mathit{cursor}
\]
The following functions return the cursor and offset of a capability respectively:
\note{rmn30}{Re-name cursor to address here?}
\medskip
\sailRISCVval{getCapCursor}
\sailRISCVval{getCapOffsetBits}
\note{rmn30}{explain what happens when offset is negative? In fact it is computed modulo $2^{64}$ and always converted straight to a 64-bit vector so not important. Should maybe just return vector.}
\noindent The following functions adjust the bounds and offset of capabilities.
Not all combinations of bounds and offset are representable, so these functions return a boolean value indicating whether the requested operation was successful.
Even in the case of failure a capability is still returned, although it may not preserve the bounds of the original capability.
\medskip
\sailRISCVval{setCapBounds}
\sailRISCVval{setCapAddr}
\sailRISCVval{setCapOffset}
\sailRISCVval{incCapOffset}
\medskip
\sailRISCVval{clearTag}
\sailRISCVval{clearTagIf}
\sailRISCVval{clearTagIfSealed}
\medskip
\sailRISCVval{getRepresentableAlignmentMask}
\sailRISCVval{getRepresentableLength}
\medskip
\arnote{TODO: short description of sealing and unsealing}
\sailRISCVval{sealCap}
\sailRISCVval{unsealCap}
\sailRISCVval{isCapSealed}
\sailRISCVval{hasReservedOType}
\noindent
Capability permissions and flags are accessed using the following functions:
\medskip
\sailRISCVval{getCapPerms}
\sailRISCVval{setCapPerms}
\sailRISCVval{getCapFlags}
\sailRISCVval{setCapFlags}
\subsection*{Checking for availability of ISA features}
\sailRISCVval{haveRVC}
\sailRISCVval{haveFExt}
\sailRISCVval{haveNExt}
\sailRISCVval{haveSupMode}
\sailRISCVval{have\_pcc\_relocation}
\section{CHERI-RISC-V Instructions}
\input{insn-riscv/auipcc}
\input{insn-riscv/candperm}
\input{insn-riscv/cbuildcap}
\input{insn-riscv/cclear}
\input{insn-riscv/ccleartag}
%\input{insn-riscv/ccleartags} % missing sail
\input{insn-riscv/ccopytype}
\input{insn-riscv/ccseal}
\input{insn-riscv/cfromptr}
\input{insn-riscv/cgetbase}
\input{insn-riscv/cgetflags}
\input{insn-riscv/cgethigh}
\input{insn-riscv/cgetlen}
\input{insn-riscv/cgetoffset}
\input{insn-riscv/cgetperm}
\input{insn-riscv/cgetsealed}
\input{insn-riscv/cgettag}
\input{insn-riscv/cgettop}
\input{insn-riscv/cgettype}
\input{insn-riscv/cincoffset}
\input{insn-riscv/cincoffsetimm}
\input{insn-riscv/cinvoke}
\input{insn-riscv/cjal}
\input{insn-riscv/cjalr}
\input{insn-riscv/lc} % [c]lc
\input{insn-riscv/cloadtags}
\input{insn-riscv/cmove}
\input{insn-riscv/crepresentablealignmentmask}
\input{insn-riscv/croundrepresentablelength}
\input{insn-riscv/sc} % [c]sc
\input{insn-riscv/cseal}
\input{insn-riscv/csealentry}
\input{insn-riscv/csetaddr}
\input{insn-riscv/csetbounds}
\input{insn-riscv/csetboundsexact}
\input{insn-riscv/csetboundsimm}
\input{insn-riscv/csetequalexact}
\input{insn-riscv/csetflags}
\input{insn-riscv/csethigh}
\input{insn-riscv/csetoffset}
\input{insn-riscv/cspecialrw}
\input{insn-riscv/ctestsubset}
\input{insn-riscv/ctoptr}
\input{insn-riscv/cunseal}
\input{insn-riscv/fpclear}
\input{insn-riscv/jalrcap}
\input{insn-riscv/jalrpcc}
\input{insn-riscv/loaddatacap} % l[bhwd][u].cap
\input{insn-riscv/loaddataddc} % l[bhwd][u].ddc
\input{insn-riscv/loadcapcap} % lc.cap
\input{insn-riscv/loadcapddc} % lc.ddc
\input{insn-riscv/storedatacap} % s[bhwd].cap
\input{insn-riscv/storedataddc} % s[bhwd].ddc
\input{insn-riscv/storecapcap} % sc.cap
\input{insn-riscv/storecapddc} % sc.ddc