1
0
mirror of https://github.com/zellyn/go6502.git synced 2024-08-29 05:29:04 +00:00
go6502/asm/a2as/a2as.go

122 lines
2.6 KiB
Go
Raw Normal View History

2014-05-22 04:44:23 +00:00
package main
import (
"flag"
"fmt"
"os"
"strings"
"github.com/zellyn/go6502/asm"
"github.com/zellyn/go6502/asm/flavors"
2014-05-23 23:53:46 +00:00
"github.com/zellyn/go6502/asm/flavors/redbook"
2014-05-22 04:44:23 +00:00
"github.com/zellyn/go6502/asm/flavors/scma"
"github.com/zellyn/go6502/asm/ihex"
"github.com/zellyn/go6502/asm/lines"
)
var flavorsByName map[string]flavors.F
var flavor string
func init() {
flavorsByName = map[string]flavors.F{
2014-05-23 23:53:46 +00:00
"scma": scma.New(),
"redbook": redbook.New(),
2014-05-22 04:44:23 +00:00
}
var names []string
for name := range flavorsByName {
names = append(names, name)
}
usage := fmt.Sprintf("assembler flavor: %s", strings.Join(names, ","))
flag.StringVar(&flavor, "flavor", "", usage)
}
var infile = flag.String("in", "", "input file")
var outfile = flag.String("out", "", "output file")
var format = flag.String("format", "binary", "output format: binary/ihex")
var fill = flag.Uint("fillbyte", 0x00, "byte value to use when filling gaps between assmebler output regions")
func main() {
flag.Parse()
if *infile == "" {
fmt.Fprintln(os.Stderr, "no input file specified")
os.Exit(1)
}
if *outfile == "" {
fmt.Fprintln(os.Stderr, "no output file specified")
os.Exit(1)
}
if flavor == "" {
fmt.Fprintln(os.Stderr, "no flavor specified")
os.Exit(1)
}
f, ok := flavorsByName[flavor]
if !ok {
fmt.Fprintf(os.Stderr, "invalid flavor: '%s'\n", flavor)
os.Exit(1)
}
if *format != "binary" && *format != "ihex" {
fmt.Fprintf(os.Stderr, "format must be binary or ihex; got '%s'\n", *format)
os.Exit(1)
}
if *fill > 0xff {
fmt.Fprintf(os.Stderr, "fillbyte must be <= 255; got '%s'\n", *format)
os.Exit(1)
}
var o lines.OsOpener
a := asm.NewAssembler(f, o)
if err := a.Assemble(*infile); err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
out, err := os.Create(*outfile)
if err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
defer func() {
err := out.Close()
if err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
}()
m, err := a.Membuf()
if err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
switch *format {
case "binary":
p := m.Piece(byte(*fill))
n, err := out.Write(p.Data)
if err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
if n != len(p.Data) {
fmt.Fprintf(os.Stderr, "Error writing to '%s': wrote %d of %d bytes", *outfile, n, len(p.Data))
os.Exit(1)
}
case "ihex":
w := ihex.NewWriter(out)
for _, p := range m.Pieces() {
if err := w.Write(p.Addr, p.Data); err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
}
w.End()
default:
fmt.Fprintf(os.Stderr, "format must be binary or ihex; got '%s'\n", *format)
os.Exit(1)
}
}