| // Copyright 2019 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| //go:build ignore |
| // +build ignore |
| |
| // mkstdlib generates the zstdlib.go file, containing the Go standard |
| // library API symbols. It's baked into the binary to avoid scanning |
| // GOPATH in the common case. |
| package main |
| |
| import ( |
| "bufio" |
| "bytes" |
| "fmt" |
| "go/format" |
| "io" |
| "io/ioutil" |
| "log" |
| "os" |
| "path/filepath" |
| "regexp" |
| "runtime" |
| "sort" |
| "strings" |
| |
| exec "golang.org/x/sys/execabs" |
| ) |
| |
| func mustOpen(name string) io.Reader { |
| f, err := os.Open(name) |
| if err != nil { |
| log.Fatal(err) |
| } |
| return f |
| } |
| |
| func api(base string) string { |
| return filepath.Join(runtime.GOROOT(), "api", base) |
| } |
| |
| var sym = regexp.MustCompile(`^pkg (\S+).*?, (?:var|func|type|const) ([A-Z]\w*)`) |
| |
| var unsafeSyms = map[string]bool{"Alignof": true, "ArbitraryType": true, "Offsetof": true, "Pointer": true, "Sizeof": true} |
| |
| func main() { |
| var buf bytes.Buffer |
| outf := func(format string, args ...interface{}) { |
| fmt.Fprintf(&buf, format, args...) |
| } |
| outf(`// Copyright 2022 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| `) |
| outf("// Code generated by mkstdlib.go. DO NOT EDIT.\n\n") |
| outf("package imports\n") |
| outf("var stdlib = map[string][]string{\n") |
| f := readAPI() |
| sc := bufio.NewScanner(f) |
| |
| pkgs := map[string]map[string]bool{ |
| "unsafe": unsafeSyms, |
| } |
| paths := []string{"unsafe"} |
| |
| for sc.Scan() { |
| l := sc.Text() |
| if m := sym.FindStringSubmatch(l); m != nil { |
| path, sym := m[1], m[2] |
| |
| if _, ok := pkgs[path]; !ok { |
| pkgs[path] = map[string]bool{} |
| paths = append(paths, path) |
| } |
| pkgs[path][sym] = true |
| } |
| } |
| if err := sc.Err(); err != nil { |
| log.Fatal(err) |
| } |
| sort.Strings(paths) |
| for _, path := range paths { |
| outf("\t%q: {\n", path) |
| pkg := pkgs[path] |
| var syms []string |
| for sym := range pkg { |
| syms = append(syms, sym) |
| } |
| sort.Strings(syms) |
| for _, sym := range syms { |
| outf("\t\t%q,\n", sym) |
| } |
| outf("},\n") |
| } |
| outf("}\n") |
| fmtbuf, err := format.Source(buf.Bytes()) |
| if err != nil { |
| log.Fatal(err) |
| } |
| err = ioutil.WriteFile("zstdlib.go", fmtbuf, 0666) |
| if err != nil { |
| log.Fatal(err) |
| } |
| } |
| |
| // readAPI opens an io.Reader that reads all stdlib API content. |
| func readAPI() io.Reader { |
| entries, err := os.ReadDir(filepath.Join(runtime.GOROOT(), "api")) |
| if err != nil { |
| log.Fatal(err) |
| } |
| var readers []io.Reader |
| for _, entry := range entries { |
| name := entry.Name() |
| if strings.HasPrefix(name, "go") && strings.HasSuffix(name, ".txt") { |
| readers = append(readers, mustOpen(api(name))) |
| } |
| } |
| // The API of the syscall/js package needs to be computed explicitly, |
| // because it's not included in the GOROOT/api/go1.*.txt files at this time. |
| readers = append(readers, syscallJSAPI()) |
| return io.MultiReader(readers...) |
| } |
| |
| // syscallJSAPI returns the API of the syscall/js package. |
| // It's computed from the contents of $(go env GOROOT)/src/syscall/js. |
| func syscallJSAPI() io.Reader { |
| var exeSuffix string |
| if runtime.GOOS == "windows" { |
| exeSuffix = ".exe" |
| } |
| cmd := exec.Command("go"+exeSuffix, "run", "cmd/api", "-contexts", "js-wasm", "syscall/js") |
| out, err := cmd.Output() |
| if err != nil { |
| log.Fatalln(err) |
| } |
| return bytes.NewReader(out) |
| } |