1
2
3
4
5 package main
6
7 import (
8 "fmt"
9 )
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24 const (
25 riscv64REG_G = 27
26 riscv64REG_CTXT = 26
27 riscv64REG_LR = 1
28 riscv64REG_SP = 2
29 riscv64REG_GP = 3
30 riscv64REG_TP = 4
31 riscv64REG_TMP = 31
32 riscv64REG_ZERO = 0
33 )
34
35 func riscv64RegName(r int) string {
36 switch {
37 case r == riscv64REG_G:
38 return "g"
39 case r == riscv64REG_SP:
40 return "SP"
41 case 0 <= r && r <= 31:
42 return fmt.Sprintf("X%d", r)
43 case 32 <= r && r <= 63:
44 return fmt.Sprintf("F%d", r-32)
45 default:
46 panic(fmt.Sprintf("unknown register %d", r))
47 }
48 }
49
50 func init() {
51 var regNamesRISCV64 []string
52 var gpMask, fpMask, gpgMask, gpspMask, gpspsbMask, gpspsbgMask, first16Mask regMask
53 regNamed := make(map[string]regMask)
54
55
56
57
58
59 addreg := func(r int, name string) regMask {
60 mask := regMask(1) << uint(len(regNamesRISCV64))
61 if name == "" {
62 name = riscv64RegName(r)
63 }
64 regNamesRISCV64 = append(regNamesRISCV64, name)
65 regNamed[name] = mask
66 return mask
67 }
68
69
70 for r := 0; r <= 31; r++ {
71 if r == riscv64REG_LR {
72
73
74 continue
75 }
76
77 mask := addreg(r, "")
78
79
80 switch r {
81
82 case riscv64REG_ZERO, riscv64REG_GP, riscv64REG_TP, riscv64REG_TMP:
83 case riscv64REG_G:
84 gpgMask |= mask
85 gpspsbgMask |= mask
86 case riscv64REG_SP:
87 gpspMask |= mask
88 gpspsbMask |= mask
89 gpspsbgMask |= mask
90 default:
91 gpMask |= mask
92 gpgMask |= mask
93 gpspMask |= mask
94 gpspsbMask |= mask
95 gpspsbgMask |= mask
96 if r >= 5 && r < 5+16 {
97 first16Mask |= mask
98 }
99 }
100 }
101
102
103 for r := 32; r <= 63; r++ {
104 mask := addreg(r, "")
105 fpMask |= mask
106 }
107
108
109 mask := addreg(-1, "SB")
110 gpspsbMask |= mask
111 gpspsbgMask |= mask
112
113 if len(regNamesRISCV64) > 64 {
114
115 panic("Too many RISCV64 registers")
116 }
117
118 regCtxt := regNamed["X26"]
119 callerSave := gpMask | fpMask | regNamed["g"]
120 r5toR6 := regNamed["X5"] | regNamed["X6"]
121
122 var (
123 gpstore = regInfo{inputs: []regMask{gpspsbMask, gpspMask, 0}}
124 gpstore0 = regInfo{inputs: []regMask{gpspsbMask}}
125 gp01 = regInfo{outputs: []regMask{gpMask}}
126 gp11 = regInfo{inputs: []regMask{gpMask}, outputs: []regMask{gpMask}}
127 gp21 = regInfo{inputs: []regMask{gpMask, gpMask}, outputs: []regMask{gpMask}}
128 gp22 = regInfo{inputs: []regMask{gpMask, gpMask}, outputs: []regMask{gpMask, gpMask}}
129 gpload = regInfo{inputs: []regMask{gpspsbMask, 0}, outputs: []regMask{gpMask}}
130 gp11sb = regInfo{inputs: []regMask{gpspsbMask}, outputs: []regMask{gpMask}}
131 gpxchg = regInfo{inputs: []regMask{gpspsbgMask, gpgMask}, outputs: []regMask{gpMask}}
132 gpcas = regInfo{inputs: []regMask{gpspsbgMask, gpgMask, gpgMask}, outputs: []regMask{gpMask}}
133 gpatomic = regInfo{inputs: []regMask{gpspsbgMask, gpgMask}}
134
135 fp01 = regInfo{outputs: []regMask{fpMask}}
136 fp11 = regInfo{inputs: []regMask{fpMask}, outputs: []regMask{fpMask}}
137 fp21 = regInfo{inputs: []regMask{fpMask, fpMask}, outputs: []regMask{fpMask}}
138 fp31 = regInfo{inputs: []regMask{fpMask, fpMask, fpMask}, outputs: []regMask{fpMask}}
139 gpfp = regInfo{inputs: []regMask{gpMask}, outputs: []regMask{fpMask}}
140 fpgp = regInfo{inputs: []regMask{fpMask}, outputs: []regMask{gpMask}}
141 fpstore = regInfo{inputs: []regMask{gpspsbMask, fpMask, 0}}
142 fpload = regInfo{inputs: []regMask{gpspsbMask, 0}, outputs: []regMask{fpMask}}
143 fp2gp = regInfo{inputs: []regMask{fpMask, fpMask}, outputs: []regMask{gpMask}}
144
145 call = regInfo{clobbers: callerSave}
146 callClosure = regInfo{inputs: []regMask{gpspMask, regCtxt, 0}, clobbers: callerSave}
147 callInter = regInfo{inputs: []regMask{gpMask}, clobbers: callerSave}
148 )
149
150 RISCV64ops := []opData{
151 {name: "ADD", argLength: 2, reg: gp21, asm: "ADD", commutative: true},
152 {name: "ADDI", argLength: 1, reg: gp11sb, asm: "ADDI", aux: "Int64"},
153 {name: "ADDIW", argLength: 1, reg: gp11, asm: "ADDIW", aux: "Int64"},
154 {name: "NEG", argLength: 1, reg: gp11, asm: "NEG"},
155 {name: "NEGW", argLength: 1, reg: gp11, asm: "NEGW"},
156 {name: "SUB", argLength: 2, reg: gp21, asm: "SUB"},
157 {name: "SUBW", argLength: 2, reg: gp21, asm: "SUBW"},
158
159
160
161 {name: "MUL", argLength: 2, reg: gp21, asm: "MUL", commutative: true, typ: "Int64"},
162 {name: "MULW", argLength: 2, reg: gp21, asm: "MULW", commutative: true, typ: "Int32"},
163 {name: "MULH", argLength: 2, reg: gp21, asm: "MULH", commutative: true, typ: "Int64"},
164 {name: "MULHU", argLength: 2, reg: gp21, asm: "MULHU", commutative: true, typ: "UInt64"},
165 {name: "LoweredMuluhilo", argLength: 2, reg: gp22, resultNotInArgs: true},
166 {name: "LoweredMuluover", argLength: 2, reg: gp22, resultNotInArgs: true},
167
168 {name: "DIV", argLength: 2, reg: gp21, asm: "DIV", typ: "Int64"},
169 {name: "DIVU", argLength: 2, reg: gp21, asm: "DIVU", typ: "UInt64"},
170 {name: "DIVW", argLength: 2, reg: gp21, asm: "DIVW", typ: "Int32"},
171 {name: "DIVUW", argLength: 2, reg: gp21, asm: "DIVUW", typ: "UInt32"},
172 {name: "REM", argLength: 2, reg: gp21, asm: "REM", typ: "Int64"},
173 {name: "REMU", argLength: 2, reg: gp21, asm: "REMU", typ: "UInt64"},
174 {name: "REMW", argLength: 2, reg: gp21, asm: "REMW", typ: "Int32"},
175 {name: "REMUW", argLength: 2, reg: gp21, asm: "REMUW", typ: "UInt32"},
176
177 {name: "MOVaddr", argLength: 1, reg: gp11sb, asm: "MOV", aux: "SymOff", rematerializeable: true, symEffect: "Addr"},
178
179
180 {name: "MOVDconst", reg: gp01, asm: "MOV", typ: "UInt64", aux: "Int64", rematerializeable: true},
181 {name: "FMOVDconst", reg: fp01, asm: "MOVD", typ: "Float64", aux: "Float64", rematerializeable: true},
182 {name: "FMOVFconst", reg: fp01, asm: "MOVF", typ: "Float32", aux: "Float32", rematerializeable: true},
183
184
185 {name: "MOVBload", argLength: 2, reg: gpload, asm: "MOVB", aux: "SymOff", typ: "Int8", faultOnNilArg0: true, symEffect: "Read"},
186 {name: "MOVHload", argLength: 2, reg: gpload, asm: "MOVH", aux: "SymOff", typ: "Int16", faultOnNilArg0: true, symEffect: "Read"},
187 {name: "MOVWload", argLength: 2, reg: gpload, asm: "MOVW", aux: "SymOff", typ: "Int32", faultOnNilArg0: true, symEffect: "Read"},
188 {name: "MOVDload", argLength: 2, reg: gpload, asm: "MOV", aux: "SymOff", typ: "Int64", faultOnNilArg0: true, symEffect: "Read"},
189 {name: "MOVBUload", argLength: 2, reg: gpload, asm: "MOVBU", aux: "SymOff", typ: "UInt8", faultOnNilArg0: true, symEffect: "Read"},
190 {name: "MOVHUload", argLength: 2, reg: gpload, asm: "MOVHU", aux: "SymOff", typ: "UInt16", faultOnNilArg0: true, symEffect: "Read"},
191 {name: "MOVWUload", argLength: 2, reg: gpload, asm: "MOVWU", aux: "SymOff", typ: "UInt32", faultOnNilArg0: true, symEffect: "Read"},
192
193
194 {name: "MOVBstore", argLength: 3, reg: gpstore, asm: "MOVB", aux: "SymOff", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
195 {name: "MOVHstore", argLength: 3, reg: gpstore, asm: "MOVH", aux: "SymOff", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
196 {name: "MOVWstore", argLength: 3, reg: gpstore, asm: "MOVW", aux: "SymOff", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
197 {name: "MOVDstore", argLength: 3, reg: gpstore, asm: "MOV", aux: "SymOff", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
198
199
200 {name: "MOVBstorezero", argLength: 2, reg: gpstore0, aux: "SymOff", asm: "MOVB", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
201 {name: "MOVHstorezero", argLength: 2, reg: gpstore0, aux: "SymOff", asm: "MOVH", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
202 {name: "MOVWstorezero", argLength: 2, reg: gpstore0, aux: "SymOff", asm: "MOVW", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
203 {name: "MOVDstorezero", argLength: 2, reg: gpstore0, aux: "SymOff", asm: "MOV", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
204
205
206 {name: "MOVBreg", argLength: 1, reg: gp11, asm: "MOVB"},
207 {name: "MOVHreg", argLength: 1, reg: gp11, asm: "MOVH"},
208 {name: "MOVWreg", argLength: 1, reg: gp11, asm: "MOVW"},
209 {name: "MOVDreg", argLength: 1, reg: gp11, asm: "MOV"},
210 {name: "MOVBUreg", argLength: 1, reg: gp11, asm: "MOVBU"},
211 {name: "MOVHUreg", argLength: 1, reg: gp11, asm: "MOVHU"},
212 {name: "MOVWUreg", argLength: 1, reg: gp11, asm: "MOVWU"},
213
214 {name: "MOVDnop", argLength: 1, reg: regInfo{inputs: []regMask{gpMask}, outputs: []regMask{gpMask}}, resultInArg0: true},
215
216
217 {name: "SLL", argLength: 2, reg: gp21, asm: "SLL"},
218 {name: "SLLW", argLength: 2, reg: gp21, asm: "SLLW"},
219 {name: "SRA", argLength: 2, reg: gp21, asm: "SRA"},
220 {name: "SRAW", argLength: 2, reg: gp21, asm: "SRAW"},
221 {name: "SRL", argLength: 2, reg: gp21, asm: "SRL"},
222 {name: "SRLW", argLength: 2, reg: gp21, asm: "SRLW"},
223 {name: "SLLI", argLength: 1, reg: gp11, asm: "SLLI", aux: "Int64"},
224 {name: "SLLIW", argLength: 1, reg: gp11, asm: "SLLIW", aux: "Int64"},
225 {name: "SRAI", argLength: 1, reg: gp11, asm: "SRAI", aux: "Int64"},
226 {name: "SRAIW", argLength: 1, reg: gp11, asm: "SRAIW", aux: "Int64"},
227 {name: "SRLI", argLength: 1, reg: gp11, asm: "SRLI", aux: "Int64"},
228 {name: "SRLIW", argLength: 1, reg: gp11, asm: "SRLIW", aux: "Int64"},
229
230
231 {name: "SH1ADD", argLength: 2, reg: gp21, asm: "SH1ADD"},
232 {name: "SH2ADD", argLength: 2, reg: gp21, asm: "SH2ADD"},
233 {name: "SH3ADD", argLength: 2, reg: gp21, asm: "SH3ADD"},
234
235
236 {name: "AND", argLength: 2, reg: gp21, asm: "AND", commutative: true},
237 {name: "ANDN", argLength: 2, reg: gp21, asm: "ANDN"},
238 {name: "ANDI", argLength: 1, reg: gp11, asm: "ANDI", aux: "Int64"},
239 {name: "CLZ", argLength: 1, reg: gp11, asm: "CLZ"},
240 {name: "CLZW", argLength: 1, reg: gp11, asm: "CLZW"},
241 {name: "CPOP", argLength: 1, reg: gp11, asm: "CPOP"},
242 {name: "CPOPW", argLength: 1, reg: gp11, asm: "CPOPW"},
243 {name: "CTZ", argLength: 1, reg: gp11, asm: "CTZ"},
244 {name: "CTZW", argLength: 1, reg: gp11, asm: "CTZW"},
245 {name: "NOT", argLength: 1, reg: gp11, asm: "NOT"},
246 {name: "OR", argLength: 2, reg: gp21, asm: "OR", commutative: true},
247 {name: "ORN", argLength: 2, reg: gp21, asm: "ORN"},
248 {name: "ORI", argLength: 1, reg: gp11, asm: "ORI", aux: "Int64"},
249 {name: "REV8", argLength: 1, reg: gp11, asm: "REV8"},
250 {name: "ROL", argLength: 2, reg: gp21, asm: "ROL"},
251 {name: "ROLW", argLength: 2, reg: gp21, asm: "ROLW"},
252 {name: "ROR", argLength: 2, reg: gp21, asm: "ROR"},
253 {name: "RORI", argLength: 1, reg: gp11, asm: "RORI", aux: "Int64"},
254 {name: "RORIW", argLength: 1, reg: gp11, asm: "RORIW", aux: "Int64"},
255 {name: "RORW", argLength: 2, reg: gp21, asm: "RORW"},
256 {name: "XNOR", argLength: 2, reg: gp21, asm: "XNOR", commutative: true},
257 {name: "XOR", argLength: 2, reg: gp21, asm: "XOR", commutative: true},
258 {name: "XORI", argLength: 1, reg: gp11, asm: "XORI", aux: "Int64"},
259
260
261 {name: "MIN", argLength: 2, reg: gp21, asm: "MIN", commutative: true},
262 {name: "MAX", argLength: 2, reg: gp21, asm: "MAX", commutative: true},
263 {name: "MINU", argLength: 2, reg: gp21, asm: "MINU", commutative: true},
264 {name: "MAXU", argLength: 2, reg: gp21, asm: "MAXU", commutative: true},
265
266
267 {name: "SEQZ", argLength: 1, reg: gp11, asm: "SEQZ"},
268 {name: "SNEZ", argLength: 1, reg: gp11, asm: "SNEZ"},
269 {name: "SLT", argLength: 2, reg: gp21, asm: "SLT"},
270 {name: "SLTI", argLength: 1, reg: gp11, asm: "SLTI", aux: "Int64"},
271 {name: "SLTU", argLength: 2, reg: gp21, asm: "SLTU"},
272 {name: "SLTIU", argLength: 1, reg: gp11, asm: "SLTIU", aux: "Int64"},
273
274
275 {name: "LoweredRound32F", argLength: 1, reg: fp11, resultInArg0: true},
276 {name: "LoweredRound64F", argLength: 1, reg: fp11, resultInArg0: true},
277
278
279 {name: "CALLstatic", argLength: -1, reg: call, aux: "CallOff", call: true},
280 {name: "CALLtail", argLength: -1, reg: call, aux: "CallOff", call: true, tailCall: true},
281 {name: "CALLclosure", argLength: -1, reg: callClosure, aux: "CallOff", call: true},
282 {name: "CALLinter", argLength: -1, reg: callInter, aux: "CallOff", call: true},
283
284
285
286
287
288
289
290
291
292 {
293 name: "LoweredZero",
294 aux: "SymValAndOff",
295 typ: "Mem",
296 argLength: 2,
297 symEffect: "Write",
298 faultOnNilArg0: true,
299 reg: regInfo{
300 inputs: []regMask{gpMask},
301 },
302 },
303
304
305
306
307
308 {
309 name: "LoweredZeroLoop",
310 aux: "SymValAndOff",
311 typ: "Mem",
312 argLength: 2,
313 symEffect: "Write",
314 needIntTemp: true,
315 faultOnNilArg0: true,
316 reg: regInfo{
317 inputs: []regMask{gpMask},
318 clobbersArg0: true,
319 },
320 },
321
322
323
324
325
326
327
328
329
330 {
331 name: "LoweredMove",
332 aux: "SymValAndOff",
333 symEffect: "Write",
334 argLength: 3,
335 reg: regInfo{
336 inputs: []regMask{gpMask &^ regNamed["X5"], gpMask &^ regNamed["X5"]},
337 clobbers: regNamed["X5"],
338 },
339 faultOnNilArg0: true,
340 faultOnNilArg1: true,
341 },
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357 {
358 name: "LoweredMoveLoop",
359 aux: "SymValAndOff",
360 argLength: 3,
361 symEffect: "Write",
362 reg: regInfo{
363 inputs: []regMask{gpMask &^ r5toR6, gpMask &^ r5toR6},
364 clobbers: r5toR6,
365 clobbersArg0: true,
366 clobbersArg1: true,
367 },
368 faultOnNilArg0: true,
369 faultOnNilArg1: true,
370 },
371
372
373
374
375 {name: "LoweredAtomicLoad8", argLength: 2, reg: gpload, faultOnNilArg0: true},
376 {name: "LoweredAtomicLoad32", argLength: 2, reg: gpload, faultOnNilArg0: true},
377 {name: "LoweredAtomicLoad64", argLength: 2, reg: gpload, faultOnNilArg0: true},
378
379
380
381 {name: "LoweredAtomicStore8", argLength: 3, reg: gpstore, faultOnNilArg0: true, hasSideEffects: true},
382 {name: "LoweredAtomicStore32", argLength: 3, reg: gpstore, faultOnNilArg0: true, hasSideEffects: true},
383 {name: "LoweredAtomicStore64", argLength: 3, reg: gpstore, faultOnNilArg0: true, hasSideEffects: true},
384
385
386
387 {name: "LoweredAtomicExchange32", argLength: 3, reg: gpxchg, resultNotInArgs: true, faultOnNilArg0: true, hasSideEffects: true},
388 {name: "LoweredAtomicExchange64", argLength: 3, reg: gpxchg, resultNotInArgs: true, faultOnNilArg0: true, hasSideEffects: true},
389
390
391
392 {name: "LoweredAtomicAdd32", argLength: 3, reg: gpxchg, resultNotInArgs: true, faultOnNilArg0: true, hasSideEffects: true, unsafePoint: true},
393 {name: "LoweredAtomicAdd64", argLength: 3, reg: gpxchg, resultNotInArgs: true, faultOnNilArg0: true, hasSideEffects: true, unsafePoint: true},
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409 {name: "LoweredAtomicCas32", argLength: 4, reg: gpcas, resultNotInArgs: true, faultOnNilArg0: true, hasSideEffects: true, unsafePoint: true},
410 {name: "LoweredAtomicCas64", argLength: 4, reg: gpcas, resultNotInArgs: true, faultOnNilArg0: true, hasSideEffects: true, unsafePoint: true},
411
412
413
414 {name: "LoweredAtomicAnd32", argLength: 3, reg: gpatomic, asm: "AMOANDW", faultOnNilArg0: true, hasSideEffects: true},
415 {name: "LoweredAtomicOr32", argLength: 3, reg: gpatomic, asm: "AMOORW", faultOnNilArg0: true, hasSideEffects: true},
416
417
418 {name: "LoweredNilCheck", argLength: 2, faultOnNilArg0: true, nilCheck: true, reg: regInfo{inputs: []regMask{gpspMask}}},
419 {name: "LoweredGetClosurePtr", reg: regInfo{outputs: []regMask{regCtxt}}},
420
421
422 {name: "LoweredGetCallerSP", argLength: 1, reg: gp01, rematerializeable: true},
423
424
425
426
427
428 {name: "LoweredGetCallerPC", reg: gp01, rematerializeable: true},
429
430
431
432
433
434
435 {name: "LoweredWB", argLength: 1, reg: regInfo{clobbers: (callerSave &^ (gpMask | regNamed["g"])) | regNamed["X1"], outputs: []regMask{regNamed["X24"]}}, clobberFlags: true, aux: "Int64"},
436
437
438 {name: "LoweredPubBarrier", argLength: 1, asm: "FENCE", hasSideEffects: true},
439
440
441
442
443
444
445 {name: "LoweredPanicBoundsRR", argLength: 3, aux: "Int64", reg: regInfo{inputs: []regMask{first16Mask, first16Mask}}, typ: "Mem", call: true},
446 {name: "LoweredPanicBoundsRC", argLength: 2, aux: "PanicBoundsC", reg: regInfo{inputs: []regMask{first16Mask}}, typ: "Mem", call: true},
447 {name: "LoweredPanicBoundsCR", argLength: 2, aux: "PanicBoundsC", reg: regInfo{inputs: []regMask{first16Mask}}, typ: "Mem", call: true},
448 {name: "LoweredPanicBoundsCC", argLength: 1, aux: "PanicBoundsCC", reg: regInfo{}, typ: "Mem", call: true},
449
450
451 {name: "FADDS", argLength: 2, reg: fp21, asm: "FADDS", commutative: true, typ: "Float32"},
452 {name: "FSUBS", argLength: 2, reg: fp21, asm: "FSUBS", commutative: false, typ: "Float32"},
453 {name: "FMULS", argLength: 2, reg: fp21, asm: "FMULS", commutative: true, typ: "Float32"},
454 {name: "FDIVS", argLength: 2, reg: fp21, asm: "FDIVS", commutative: false, typ: "Float32"},
455 {name: "FMADDS", argLength: 3, reg: fp31, asm: "FMADDS", commutative: true, typ: "Float32"},
456 {name: "FMSUBS", argLength: 3, reg: fp31, asm: "FMSUBS", commutative: true, typ: "Float32"},
457 {name: "FNMADDS", argLength: 3, reg: fp31, asm: "FNMADDS", commutative: true, typ: "Float32"},
458 {name: "FNMSUBS", argLength: 3, reg: fp31, asm: "FNMSUBS", commutative: true, typ: "Float32"},
459 {name: "FSQRTS", argLength: 1, reg: fp11, asm: "FSQRTS", typ: "Float32"},
460 {name: "FNEGS", argLength: 1, reg: fp11, asm: "FNEGS", typ: "Float32"},
461 {name: "FMVSX", argLength: 1, reg: gpfp, asm: "FMVSX", typ: "Float32"},
462 {name: "FMVXS", argLength: 1, reg: fpgp, asm: "FMVXS", typ: "Int32"},
463 {name: "FCVTSW", argLength: 1, reg: gpfp, asm: "FCVTSW", typ: "Float32"},
464 {name: "FCVTSL", argLength: 1, reg: gpfp, asm: "FCVTSL", typ: "Float32"},
465 {name: "FCVTWS", argLength: 1, reg: fpgp, asm: "FCVTWS", typ: "Int32"},
466 {name: "FCVTLS", argLength: 1, reg: fpgp, asm: "FCVTLS", typ: "Int64"},
467 {name: "FMOVWload", argLength: 2, reg: fpload, asm: "MOVF", aux: "SymOff", typ: "Float32", faultOnNilArg0: true, symEffect: "Read"},
468 {name: "FMOVWstore", argLength: 3, reg: fpstore, asm: "MOVF", aux: "SymOff", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
469 {name: "FEQS", argLength: 2, reg: fp2gp, asm: "FEQS", commutative: true},
470 {name: "FNES", argLength: 2, reg: fp2gp, asm: "FNES", commutative: true},
471 {name: "FLTS", argLength: 2, reg: fp2gp, asm: "FLTS"},
472 {name: "FLES", argLength: 2, reg: fp2gp, asm: "FLES"},
473 {name: "LoweredFMAXS", argLength: 2, reg: fp21, resultNotInArgs: true, asm: "FMAXS", commutative: true, typ: "Float32"},
474 {name: "LoweredFMINS", argLength: 2, reg: fp21, resultNotInArgs: true, asm: "FMINS", commutative: true, typ: "Float32"},
475
476
477 {name: "FADDD", argLength: 2, reg: fp21, asm: "FADDD", commutative: true, typ: "Float64"},
478 {name: "FSUBD", argLength: 2, reg: fp21, asm: "FSUBD", commutative: false, typ: "Float64"},
479 {name: "FMULD", argLength: 2, reg: fp21, asm: "FMULD", commutative: true, typ: "Float64"},
480 {name: "FDIVD", argLength: 2, reg: fp21, asm: "FDIVD", commutative: false, typ: "Float64"},
481 {name: "FMADDD", argLength: 3, reg: fp31, asm: "FMADDD", commutative: true, typ: "Float64"},
482 {name: "FMSUBD", argLength: 3, reg: fp31, asm: "FMSUBD", commutative: true, typ: "Float64"},
483 {name: "FNMADDD", argLength: 3, reg: fp31, asm: "FNMADDD", commutative: true, typ: "Float64"},
484 {name: "FNMSUBD", argLength: 3, reg: fp31, asm: "FNMSUBD", commutative: true, typ: "Float64"},
485 {name: "FSQRTD", argLength: 1, reg: fp11, asm: "FSQRTD", typ: "Float64"},
486 {name: "FNEGD", argLength: 1, reg: fp11, asm: "FNEGD", typ: "Float64"},
487 {name: "FABSD", argLength: 1, reg: fp11, asm: "FABSD", typ: "Float64"},
488 {name: "FSGNJD", argLength: 2, reg: fp21, asm: "FSGNJD", typ: "Float64"},
489 {name: "FMVDX", argLength: 1, reg: gpfp, asm: "FMVDX", typ: "Float64"},
490 {name: "FMVXD", argLength: 1, reg: fpgp, asm: "FMVXD", typ: "Int64"},
491 {name: "FCVTDW", argLength: 1, reg: gpfp, asm: "FCVTDW", typ: "Float64"},
492 {name: "FCVTDL", argLength: 1, reg: gpfp, asm: "FCVTDL", typ: "Float64"},
493 {name: "FCVTWD", argLength: 1, reg: fpgp, asm: "FCVTWD", typ: "Int32"},
494 {name: "FCVTLD", argLength: 1, reg: fpgp, asm: "FCVTLD", typ: "Int64"},
495 {name: "FCVTDS", argLength: 1, reg: fp11, asm: "FCVTDS", typ: "Float64"},
496 {name: "FCVTSD", argLength: 1, reg: fp11, asm: "FCVTSD", typ: "Float32"},
497 {name: "FMOVDload", argLength: 2, reg: fpload, asm: "MOVD", aux: "SymOff", typ: "Float64", faultOnNilArg0: true, symEffect: "Read"},
498 {name: "FMOVDstore", argLength: 3, reg: fpstore, asm: "MOVD", aux: "SymOff", typ: "Mem", faultOnNilArg0: true, symEffect: "Write"},
499 {name: "FEQD", argLength: 2, reg: fp2gp, asm: "FEQD", commutative: true},
500 {name: "FNED", argLength: 2, reg: fp2gp, asm: "FNED", commutative: true},
501 {name: "FLTD", argLength: 2, reg: fp2gp, asm: "FLTD"},
502 {name: "FLED", argLength: 2, reg: fp2gp, asm: "FLED"},
503 {name: "LoweredFMIND", argLength: 2, reg: fp21, resultNotInArgs: true, asm: "FMIND", commutative: true, typ: "Float64"},
504 {name: "LoweredFMAXD", argLength: 2, reg: fp21, resultNotInArgs: true, asm: "FMAXD", commutative: true, typ: "Float64"},
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524 {name: "FCLASSS", argLength: 1, reg: fpgp, asm: "FCLASSS", typ: "Int64"},
525 {name: "FCLASSD", argLength: 1, reg: fpgp, asm: "FCLASSD", typ: "Int64"},
526 }
527
528 RISCV64blocks := []blockData{
529 {name: "BEQ", controls: 2},
530 {name: "BNE", controls: 2},
531 {name: "BLT", controls: 2},
532 {name: "BGE", controls: 2},
533 {name: "BLTU", controls: 2},
534 {name: "BGEU", controls: 2},
535
536 {name: "BEQZ", controls: 1},
537 {name: "BNEZ", controls: 1},
538 {name: "BLEZ", controls: 1},
539 {name: "BGEZ", controls: 1},
540 {name: "BLTZ", controls: 1},
541 {name: "BGTZ", controls: 1},
542 }
543
544 archs = append(archs, arch{
545 name: "RISCV64",
546 pkg: "cmd/internal/obj/riscv",
547 genfile: "../../riscv64/ssa.go",
548 ops: RISCV64ops,
549 blocks: RISCV64blocks,
550 regnames: regNamesRISCV64,
551 gpregmask: gpMask,
552 fpregmask: fpMask,
553 framepointerreg: -1,
554
555 ParamIntRegNames: "X10 X11 X12 X13 X14 X15 X16 X17 X8 X9 X18 X19 X20 X21 X22 X23",
556
557 ParamFloatRegNames: "F10 F11 F12 F13 F14 F15 F16 F17 F8 F9 F18 F19 F20 F21 F22 F23",
558 })
559 }
560
View as plain text