123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- // Build builds code as directed by json files.
- // We slurp in the JSON, and recursively process includes.
- // At the end, we issue a single gcc command for all the files.
- // Compilers are fast.
- package main
- import (
- // "fmt"
- "encoding/json"
- "io/ioutil"
- "log"
- "os"
- "os/exec"
- "path"
- )
- type build struct {
- // jsons is unexported so can not be set in a .json file
- jsons map[string]bool
- Name string
- // Projects name a whole subproject which is built independently of
- // this one. We'll need to be able to use environment variables at some point.
- Projects []string
- Pre []string
- Post []string
- Cflags []string
- Oflags []string
- Include []string
- SourceFiles []string
- ObjectFiles []string
- Libs []string
- Env []string
- // cmd's
- Programs []string
- SourceFilesCmd []string
- // Targets.
- Program string
- Library string
- }
- var (
- cwd string
- harvey string
- )
- func fail(err error) {
- if err != nil {
- log.Fatalf("%v", err)
- }
- }
- func adjust(s []string) (r []string) {
- for _, v := range s {
- if path.IsAbs(v) {
- v = path.Join(harvey, v)
- }
- r = append(r, v)
- }
- return
- }
- func process(f string, b *build) {
- if b.jsons[f] {
- return
- }
- log.Printf("Processing %v", f)
- d, err := ioutil.ReadFile(f)
- fail(err)
- var build build
- err = json.Unmarshal(d, &build)
- fail(err)
- b.jsons[f] = true
- b.SourceFiles = append(b.SourceFiles, build.SourceFiles...)
- b.Cflags = append(b.Cflags, build.Cflags...)
- b.Oflags = append(b.Oflags, build.Oflags...)
- b.Pre = append(b.Pre, build.Pre...)
- b.Post = append(b.Post, build.Post...)
- b.Libs = append(b.Libs, adjust(build.Libs)...)
- b.Projects = append(b.Projects, adjust(build.Projects)...)
- b.Env = append(b.Env, build.Env...)
- b.Programs = append(b.Programs, adjust(build.Programs)...)
- b.SourceFilesCmd = append(b.SourceFilesCmd, build.SourceFilesCmd...)
- b.Program += build.Program
- b.Library += build.Library
- // For each source file, assume we create an object file with the last char replaced
- // with 'o'. We can get smarter later.
- for _, v := range build.SourceFiles {
- f := path.Base(v)
- o := f[:len(f)-1] + "o"
- b.ObjectFiles = append(b.ObjectFiles, o)
- }
- b.ObjectFiles = append(b.ObjectFiles, adjust(build.ObjectFiles)...)
- for _, v := range build.Include {
- wd := path.Dir(f)
- f := path.Join(wd, v)
- process(f, b)
- }
- }
- func compile(b *build) {
- // N.B. Plan 9 has a very well defined include structure, just three things:
- // /amd64/include, /sys/include, .
- // TODO: replace amd64 with an arch variable. Later.
- args := []string{"-c"}
- args = append(args, adjust([]string{"-I", "/amd64/include", "-I", "/sys/include", "-I", "."})...)
- args = append(args, b.Cflags...)
- if len(b.SourceFilesCmd) > 0 {
- for _, i := range b.SourceFilesCmd {
- log.Printf("compiling program %v\n", i)
- argscmd := append(args, []string{i}...)
- cmd := exec.Command("gcc", argscmd...)
- cmd.Env = append(os.Environ(), b.Env...)
- cmd.Stdin = os.Stdin
- cmd.Stderr = os.Stderr
- cmd.Stdout = os.Stdout
- log.Printf("Run %v %v", cmd.Path, cmd.Args)
- err := cmd.Run()
- if err != nil {
- log.Fatalf("%v\n", err)
- }
- argscmd = args
- }
- } else {
- args = append(args, b.SourceFiles...)
- cmd := exec.Command("gcc", args...)
- cmd.Env = append(os.Environ(), b.Env...)
- cmd.Stdin = os.Stdin
- cmd.Stderr = os.Stderr
- cmd.Stdout = os.Stdout
- log.Printf("Run %v %v", cmd.Path, cmd.Args)
- err := cmd.Run()
- if err != nil {
- log.Fatalf("%v\n", err)
- }
- }
- }
- func link(b *build) {
- if len(b.Programs) > 0 {
- for _, n := range b.Programs {
- args := []string{"-o", n}
- args = append(args, b.Oflags...)
- f := path.Base(n)
- o := f[:len(f)] + ".o"
- args = append(args, []string{o}...)
- args = append(args, b.Libs...)
- cmd := exec.Command("ld", args...)
- cmd.Env = append(os.Environ(), b.Env...)
- cmd.Stdin = os.Stdin
- cmd.Stderr = os.Stderr
- cmd.Stdout = os.Stdout
- log.Printf("Run %v %v", cmd.Path, cmd.Args)
- err := cmd.Run()
- if err != nil {
- log.Fatalf("%v\n", err)
- }
- }
- } else {
- args := []string{"-o", b.Program}
- args = append(args, b.Oflags...)
- args = append(args, b.ObjectFiles...)
- args = append(args, b.Libs...)
- cmd := exec.Command("ld", args...)
- cmd.Env = append(os.Environ(), b.Env...)
- cmd.Stdin = os.Stdin
- cmd.Stderr = os.Stderr
- cmd.Stdout = os.Stdout
- log.Printf("Run %v %v", cmd.Path, cmd.Args)
- err := cmd.Run()
- if err != nil {
- log.Fatalf("%v\n", err)
- }
- }
- }
- func run(b *build, cmd []string) {
- for _, v := range cmd {
- cmd := exec.Command("bash", "-c", v)
- cmd.Env = append(os.Environ(), b.Env...)
- cmd.Stderr = os.Stderr
- cmd.Stdout = os.Stdout
- log.Printf("Run %v %v", cmd.Path, cmd.Args)
- err := cmd.Run()
- if err != nil {
- log.Fatalf("%v\n", err)
- }
- }
- }
- func projects(b *build) {
- for _, v := range b.Projects {
- wd := path.Dir(v)
- f := path.Base(v)
- cwd, err := os.Getwd()
- fail(err)
- os.Chdir(wd)
- project(f)
- os.Chdir(cwd)
- }
- }
- // assumes we are in the wd of the project.
- func project(root string) {
- b := &build{}
- b.jsons = map[string]bool{}
- process(root, b)
- projects(b)
- run(b, b.Pre)
- if len(b.SourceFiles) > 0 {
- compile(b)
- }
- if len(b.SourceFilesCmd) > 0 {
- compile(b)
- }
- log.Printf("root %v program %v\n", root, b.Program)
- if b.Program != "" {
- link(b)
- }
- if b.Library != "" {
- //library(b)
- log.Printf("\n\n*** Building %v ***\n\n", b.Library)
- }
- if len(b.Programs ) > 0 {
- link(b)
- }
- run(b, b.Post)
- }
- func main() {
- var err error
- cwd, err = os.Getwd()
- fail(err)
- harvey = os.Getenv("HARVEY")
- if harvey == "" {
- log.Fatalf("You need to set the HARVEY environment variable")
- }
- f := path.Join(cwd, os.Args[1])
- project(f)
- }
|