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
|
// Copyright 2024 syzkaller project authors. All rights reserved.
// Use of this source code is governed by Apache 2 LICENSE that can be found in the LICENSE file.
// Pseudo instructions for arm64 architecture.
package arm64
import (
"math/rand"
"github.com/google/syzkaller/pkg/ifuzz/iset"
)
var pseudo = []*Insn{
{
Name: "PSEUDO_HCALL",
Pseudo: true,
Priv: true,
Generator: func(cfg *iset.Config, r *rand.Rand) []byte {
gen := makeGen(cfg, r)
gen.smcccHvc()
return gen.text
},
},
}
type generator struct {
r *rand.Rand
text []byte
}
func makeGen(cfg *iset.Config, r *rand.Rand) *generator {
return &generator{
r: r,
}
}
func (gen *generator) smcccHvc() {
cmd := (uint32(1) << 31) | (uint32(gen.r.Intn(2)) << 30) |
(uint32(gen.r.Intn(8)&0x3F) << 24) | (uint32(gen.r.Intn(0x10000)) & 0xFFFF)
gen.movRegImm32(0, cmd)
gen.movRegImm16(1, uint32(gen.r.Intn(16)))
gen.movRegImm16(2, uint32(gen.r.Intn(16)))
gen.movRegImm16(3, uint32(gen.r.Intn(16)))
gen.movRegImm16(4, uint32(gen.r.Intn(16)))
gen.byte(0x02, 0x00, 0x00, 0xd4)
}
func (gen *generator) movRegImm32(reg, imm uint32) {
gen.movRegImm16(reg, imm)
// Encoding `movk reg, imm16, LSL #16`.
upper := (imm >> 16) & 0xffff
opcode := uint32(0xf2a00000)
opcode |= upper << 5
opcode |= reg & 0xf
gen.imm32(opcode)
}
func (gen *generator) movRegImm16(reg, imm uint32) {
// Encoding `mov reg, imm16`.
imm = imm & 0xffff
opcode := uint32(0xd2800000)
opcode |= imm << 5
opcode |= reg & 0xf
gen.imm32(opcode)
}
func (gen *generator) imm32(v uint32) {
gen.byte(byte(v>>0), byte(v>>8), byte(v>>16), byte(v>>24))
}
func (gen *generator) byte(v ...uint8) {
gen.text = append(gen.text, v...)
}
|