generate the instruction table
This commit is contained in:
@@ -11,9 +11,9 @@ import (
|
|||||||
"github.com/mmcloughlin/avo/internal/load"
|
"github.com/mmcloughlin/avo/internal/load"
|
||||||
)
|
)
|
||||||
|
|
||||||
var generators = map[string]gen.Interface{
|
var generators = map[string]gen.Builder{
|
||||||
"asmtest": gen.NewAsmTest(),
|
"asmtest": gen.NewAsmTest,
|
||||||
"godata": gen.NewGoData(),
|
"godata": gen.NewGoData,
|
||||||
}
|
}
|
||||||
|
|
||||||
var datadir = flag.String(
|
var datadir = flag.String(
|
||||||
@@ -29,18 +29,23 @@ func main() {
|
|||||||
|
|
||||||
// Build generator.
|
// Build generator.
|
||||||
t := flag.Arg(0)
|
t := flag.Arg(0)
|
||||||
g := generators[t]
|
builder := generators[t]
|
||||||
if g == nil {
|
if builder == nil {
|
||||||
log.Fatalf("unknown generator type '%s'", t)
|
log.Fatalf("unknown generator type '%s'", t)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
g := builder(gen.Config{
|
||||||
|
Argv: os.Args,
|
||||||
|
})
|
||||||
|
|
||||||
// Determine output writer.
|
// Determine output writer.
|
||||||
w := os.Stdout
|
w := os.Stdout
|
||||||
if *output != "" {
|
if *output != "" {
|
||||||
f, err := os.Open(*output)
|
f, err := os.Create(*output)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
defer f.Close()
|
||||||
w = f
|
w = f
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,18 +10,21 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type asmtest struct {
|
type asmtest struct {
|
||||||
|
cfg Config
|
||||||
sym string // reference to the test function symbol
|
sym string // reference to the test function symbol
|
||||||
rel8 string // label to be used for near jumps
|
rel8 string // label to be used for near jumps
|
||||||
rel32 string // label for far jumps
|
rel32 string // label for far jumps
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewAsmTest() Interface {
|
func NewAsmTest(cfg Config) Interface {
|
||||||
return &asmtest{}
|
return &asmtest{cfg: cfg}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *asmtest) Generate(is []*inst.Instruction) ([]byte, error) {
|
func (a *asmtest) Generate(is []*inst.Instruction) ([]byte, error) {
|
||||||
p := &printer{}
|
p := &printer{}
|
||||||
|
|
||||||
|
p.Printf("# %s\n\n", a.cfg.GeneratedWarning())
|
||||||
|
|
||||||
a.sym = "\u00b7loadertest(SB)"
|
a.sym = "\u00b7loadertest(SB)"
|
||||||
p.Printf("TEXT %s, 0, $0\n", a.sym)
|
p.Printf("TEXT %s, 0, $0\n", a.sym)
|
||||||
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"go/format"
|
"go/format"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/mmcloughlin/avo/internal/inst"
|
"github.com/mmcloughlin/avo/internal/inst"
|
||||||
)
|
)
|
||||||
@@ -18,6 +19,24 @@ func (f Func) Generate(is []*inst.Instruction) ([]byte, error) {
|
|||||||
return f(is)
|
return f(is)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
Name string
|
||||||
|
Argv []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Config) GeneratedBy() string {
|
||||||
|
if c.Argv == nil {
|
||||||
|
return c.Name
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("command: %s", strings.Join(c.Argv, " "))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Config) GeneratedWarning() string {
|
||||||
|
return fmt.Sprintf("Code generated by %s. DO NOT EDIT.", c.GeneratedBy())
|
||||||
|
}
|
||||||
|
|
||||||
|
type Builder func(Config) Interface
|
||||||
|
|
||||||
// GoFmt formats Go code produced from the given generator.
|
// GoFmt formats Go code produced from the given generator.
|
||||||
func GoFmt(i Interface) Interface {
|
func GoFmt(i Interface) Interface {
|
||||||
return Func(func(is []*inst.Instruction) ([]byte, error) {
|
return Func(func(is []*inst.Instruction) ([]byte, error) {
|
||||||
|
|||||||
8
internal/gen/gen_test.go
Normal file
8
internal/gen/gen_test.go
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
package gen
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestBuilderInterfaces(t *testing.T) {
|
||||||
|
var _ Builder = NewAsmTest
|
||||||
|
var _ Builder = NewGoData
|
||||||
|
}
|
||||||
@@ -5,15 +5,17 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type godata struct {
|
type godata struct {
|
||||||
|
cfg Config
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewGoData() Interface {
|
func NewGoData(cfg Config) Interface {
|
||||||
return GoFmt(godata{})
|
return GoFmt(godata{cfg: cfg})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g godata) Generate(is []*inst.Instruction) ([]byte, error) {
|
func (g godata) Generate(is []*inst.Instruction) ([]byte, error) {
|
||||||
p := &printer{}
|
p := &printer{}
|
||||||
|
|
||||||
|
p.Printf("// %s\n\n", g.cfg.GeneratedWarning())
|
||||||
p.Printf("package inst\n\n")
|
p.Printf("package inst\n\n")
|
||||||
|
|
||||||
p.Printf("var Instructions = []Instruction{\n")
|
p.Printf("var Instructions = []Instruction{\n")
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package inst
|
package inst
|
||||||
|
|
||||||
|
//go:generate avogen -data ../data -output ztable.go godata
|
||||||
|
|
||||||
type Instruction struct {
|
type Instruction struct {
|
||||||
Opcode string
|
Opcode string
|
||||||
Summary string
|
Summary string
|
||||||
|
|||||||
25487
internal/inst/ztable.go
Normal file
25487
internal/inst/ztable.go
Normal file
File diff suppressed because it is too large
Load Diff
@@ -2,6 +2,7 @@ package load
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@@ -61,12 +62,16 @@ func (l *Loader) Load() ([]*inst.Instruction, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert to a slice to return.
|
// Convert to a slice, sorted by opcode.
|
||||||
is := make([]*inst.Instruction, 0, len(im))
|
is := make([]*inst.Instruction, 0, len(im))
|
||||||
for _, i := range im {
|
for _, i := range im {
|
||||||
is = append(is, i)
|
is = append(is, i)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sort.Slice(is, func(i, j int) bool {
|
||||||
|
return is[i].Opcode < is[j].Opcode
|
||||||
|
})
|
||||||
|
|
||||||
return is, nil
|
return is, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user