-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontext.go
116 lines (98 loc) · 3.11 KB
/
context.go
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
package bpf
import (
"go/ast"
"log"
"strconv"
"github.com/cilium/ebpf"
)
// Context is an BPF context, it holds (global) data that each bpf Go program needs to access while running. It is used
// when parsing a Go program and then generating the new ebpf program.
type Context struct {
ROdata *ebpf.Map // RO is a map that holds the .rodata for a eBPF program, used when generating.
Constants []*ast.BasicLit // Constants holds all the constants.
Insns []string // Insns are the assembly instruction in Go syntax.
}
// New returns a new context.
func New() *Context {
c := &Context{}
c.Insns = []string{
// set the error code for the ebpf to zero, in reverse order because we slices.Reverse this (TODO(xxx))
`asm.Return()`,
`asm.Mov.Imm(asm.R0, 0)`,
}
return c
}
func (ctx *Context) Map() {
rodata, err := ebpf.NewMap(&ebpf.MapSpec{
Type: ebpf.Array,
KeySize: 4,
ValueSize: 8,
Name: "rodata",
MaxEntries: 10,
})
if err != nil {
log.Fatal(err)
}
ctx.ROdata = rodata
}
// FD returns the fd of the rodata map.
func (ctx *Context) FD() int { return ctx.ROdata.FD() }
func (ctx *Context) AddIns(ins string) { ctx.Insns = append(ctx.Insns, ins) }
// AddConstant adds a constant to the context, it returns the index of the element so the caller can use that as a reference.
func (ctx *Context) AddConstant(basicLit *ast.BasicLit) string {
ctx.Constants = append(ctx.Constants, basicLit)
return strconv.FormatUint(uint64(len(ctx.Constants)-1), 10) // zero based, hence -1
}
// Visitor function to traverse AST and generate code
func (ctx *Context) Visit(node ast.Node) ast.Visitor {
if node == nil {
return nil
}
switch n := node.(type) {
case *ast.ImportSpec: // skip any imports
return nil
case *ast.IfStmt:
ctx.genIfStmt(n)
case *ast.AssignStmt:
ctx.genAssignStmt(n)
case *ast.FuncDecl:
ctx.genFuncDecl(n)
case *ast.CallExpr:
ctx.genCallExpr(n)
case *ast.BasicLit:
ctx.genBasicLit(n)
}
return ctx
}
func (ctx *Context) genCallExpr(callExpr *ast.CallExpr) {
if fun, ok := callExpr.Fun.(*ast.SelectorExpr); ok {
switch fun.Sel.Name {
case "TracePrintk":
ctx.AddIns(`asm.FnTracePrintk.Call()`)
}
}
}
func (ctx *Context) genFuncDecl(funcDecl *ast.FuncDecl) {
// check for builtins, or there others??
// check package??
// println(funcDecl.Name.Name)
/*
asm.LoadMapPtr(asm.R2, events.FD()), // file descriptor of the perf event array
asm.LoadImm(asm.R3, 0xffffffff, asm.DWord),
asm.Mov.Reg(asm.R4, asm.RFP),
asm.Add.Imm(asm.R4, -8),
asm.Mov.Imm(asm.R5, 4),
// call FnPerfEventOutput, an eBPF kernel helper
asm.FnPerfEventOutput.Call()
*/
}
func (ctx *Context) genIfStmt(ifStmt *ast.IfStmt) {}
func (ctx *Context) genAssignStmt(assignStmt *ast.AssignStmt) {}
func (ctx *Context) genBasicLit(basicLit *ast.BasicLit) {
index := ctx.AddConstant(basicLit) // order in generated program will be the same
// use helper to lookup element, will be stored in R0, move to R1
ctx.AddIns(`asm.LoadMapPtr(asm.R1, ctx.FD())`)
ctx.AddIns(`asm.Mov.Imm(asm.R2, ` + index + `)`)
ctx.AddIns(`asm.FnMapLookupElem.Call()`)
ctx.AddIns(`asm.Mov.Reg(asm.R0, asm.R1)`)
}