package avo import ( "errors" "github.com/mmcloughlin/avo/gotypes" "github.com/mmcloughlin/avo/operand" "github.com/mmcloughlin/avo/reg" ) type Asm interface { Asm() string } type Node interface { node() } type Label string func (l Label) node() {} // Instruction is a single instruction in a function. type Instruction struct { Opcode string Operands []operand.Op Inputs []operand.Op Outputs []operand.Op IsTerminal bool IsBranch bool IsConditional bool // CFG. Pred []*Instruction Succ []*Instruction // LiveIn/LiveOut are sets of live register IDs pre/post execution. LiveIn reg.Set LiveOut reg.Set } func (i *Instruction) node() {} func (i Instruction) TargetLabel() *Label { if !i.IsBranch { return nil } if len(i.Operands) == 0 { return nil } if ref, ok := i.Operands[0].(operand.LabelRef); ok { lbl := Label(ref) return &lbl } return nil } func (i Instruction) Registers() []reg.Register { var rs []reg.Register for _, op := range i.Operands { rs = append(rs, operand.Registers(op)...) } return rs } func (i Instruction) InputRegisters() []reg.Register { var rs []reg.Register for _, op := range i.Inputs { rs = append(rs, operand.Registers(op)...) } for _, op := range i.Outputs { if operand.IsMem(op) { rs = append(rs, operand.Registers(op)...) } } return rs } func (i Instruction) OutputRegisters() []reg.Register { var rs []reg.Register for _, op := range i.Outputs { if r, ok := op.(reg.Register); ok { rs = append(rs, r) } } return rs } type Section interface { section() } // File represents an assembly file. type File struct { Includes []string Sections []Section } func NewFile() *File { return &File{} } func (f *File) AddSection(s Section) { f.Sections = append(f.Sections, s) } func (f *File) Functions() []*Function { var fns []*Function for _, s := range f.Sections { if fn, ok := s.(*Function); ok { fns = append(fns, fn) } } return fns } // Function represents an assembly function. type Function struct { Name string Attributes Attribute Doc []string Signature *gotypes.Signature LocalSize int Nodes []Node // LabelTarget maps from label name to the following instruction. LabelTarget map[Label]*Instruction // Register allocation. Allocation reg.Allocation } func (f *Function) section() {} func NewFunction(name string) *Function { return &Function{ Name: name, Signature: gotypes.NewSignatureVoid(), } } func (f *Function) SetSignature(s *gotypes.Signature) { f.Signature = s } func (f *Function) AllocLocal(size int) operand.Mem { ptr := operand.NewStackAddr(f.LocalSize) f.LocalSize += size return ptr } func (f *Function) AddInstruction(i *Instruction) { f.AddNode(i) } func (f *Function) AddLabel(l Label) { f.AddNode(l) } func (f *Function) AddNode(n Node) { f.Nodes = append(f.Nodes, n) } // Instructions returns just the list of instruction nodes. func (f *Function) Instructions() []*Instruction { var is []*Instruction for _, n := range f.Nodes { i, ok := n.(*Instruction) if ok { is = append(is, i) } } return is } // Stub returns the Go function declaration. func (f *Function) Stub() string { return "func " + f.Name + f.Signature.String() } // FrameBytes returns the size of the stack frame in bytes. func (f *Function) FrameBytes() int { return f.LocalSize } // ArgumentBytes returns the size of the arguments in bytes. func (f *Function) ArgumentBytes() int { return f.Signature.Bytes() } type Datum struct { Offset int Value operand.Constant } func NewDatum(offset int, v operand.Constant) Datum { return Datum{ Offset: offset, Value: v, } } // Interval returns the range of bytes this datum will occupy within its section. func (d Datum) Interval() (int, int) { return d.Offset, d.Offset + d.Value.Bytes() } func (d Datum) Overlaps(other Datum) bool { s, e := d.Interval() so, eo := other.Interval() return !(eo <= s || e <= so) } type Global struct { Symbol operand.Symbol Attributes Attribute Data []Datum Size int } func NewGlobal(sym operand.Symbol) *Global { return &Global{ Symbol: sym, } } func NewStaticGlobal(name string) *Global { return NewGlobal(operand.NewStaticSymbol(name)) } func (g *Global) section() {} func (g *Global) Base() operand.Mem { return operand.NewDataAddr(g.Symbol, 0) } func (g *Global) Grow(size int) { if g.Size < size { g.Size = size } } func (g *Global) AddDatum(d Datum) error { for _, other := range g.Data { if d.Overlaps(other) { return errors.New("overlaps existing datum") } } g.add(d) return nil } func (g *Global) Append(v operand.Constant) { g.add(Datum{ Offset: g.Size, Value: v, }) } func (g *Global) add(d Datum) { _, end := d.Interval() g.Grow(end) g.Data = append(g.Data, d) }