aboutsummaryrefslogtreecommitdiffstats
path: root/pkg/ifuzz/arm64/pseudo.go
blob: b906fefad16c02def8848a8f8affbd0b82815dd3 (plain)
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...)
}