|  | // Copyright 2018 The BoringSSL Authors | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //     https://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | //go:build ignore | 
|  |  | 
|  | // godeps prints out dependencies of a package in either CMake or Make depfile | 
|  | // format, for incremental rebuilds. | 
|  | // | 
|  | // The depfile format is preferred. It works correctly when new files are added. | 
|  | // However, CMake only supports depfiles for custom commands with Ninja and | 
|  | // starting CMake 3.7. For other configurations, we also support CMake's format, | 
|  | // but CMake must be rerun when file lists change. | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "flag" | 
|  | "fmt" | 
|  | "go/build" | 
|  | "os" | 
|  | "path/filepath" | 
|  | "sort" | 
|  | "strings" | 
|  | ) | 
|  |  | 
|  | var ( | 
|  | format  = flag.String("format", "cmake", "The format to output to, either 'cmake' or 'depfile'") | 
|  | mainPkg = flag.String("pkg", "", "The package to print dependencies for") | 
|  | target  = flag.String("target", "", "The name of the output file") | 
|  | out     = flag.String("out", "", "The path to write the output to. If unset, this is stdout") | 
|  | ) | 
|  |  | 
|  | func cMakeQuote(in string) string { | 
|  | // See https://cmake.org/cmake/help/v3.0/manual/cmake-language.7.html#quoted-argument | 
|  | var b strings.Builder | 
|  | b.Grow(len(in)) | 
|  | // Iterate over in as bytes. | 
|  | for i := 0; i < len(in); i++ { | 
|  | switch c := in[i]; c { | 
|  | case '\\', '"': | 
|  | b.WriteByte('\\') | 
|  | b.WriteByte(c) | 
|  | case '\t': | 
|  | b.WriteString("\\t") | 
|  | case '\r': | 
|  | b.WriteString("\\r") | 
|  | case '\n': | 
|  | b.WriteString("\\n") | 
|  | default: | 
|  | b.WriteByte(in[i]) | 
|  | } | 
|  | } | 
|  | return b.String() | 
|  | } | 
|  |  | 
|  | func writeCMake(outFile *os.File, files []string) error { | 
|  | for i, file := range files { | 
|  | if i != 0 { | 
|  | if _, err := outFile.WriteString(";"); err != nil { | 
|  | return err | 
|  | } | 
|  | } | 
|  | if _, err := outFile.WriteString(cMakeQuote(file)); err != nil { | 
|  | return err | 
|  | } | 
|  | } | 
|  | return nil | 
|  | } | 
|  |  | 
|  | func makeQuote(in string) string { | 
|  | // See https://www.gnu.org/software/make/manual/make.html#Rule-Syntax | 
|  | var b strings.Builder | 
|  | b.Grow(len(in)) | 
|  | // Iterate over in as bytes. | 
|  | for i := 0; i < len(in); i++ { | 
|  | switch c := in[i]; c { | 
|  | case '$': | 
|  | b.WriteString("$$") | 
|  | case '#', '\\', ' ': | 
|  | b.WriteByte('\\') | 
|  | b.WriteByte(c) | 
|  | default: | 
|  | b.WriteByte(c) | 
|  | } | 
|  | } | 
|  | return b.String() | 
|  | } | 
|  |  | 
|  | func writeDepfile(outFile *os.File, files []string) error { | 
|  | if _, err := fmt.Fprintf(outFile, "%s:", makeQuote(*target)); err != nil { | 
|  | return err | 
|  | } | 
|  | for _, file := range files { | 
|  | if _, err := fmt.Fprintf(outFile, " %s", makeQuote(file)); err != nil { | 
|  | return err | 
|  | } | 
|  | } | 
|  | _, err := outFile.WriteString("\n") | 
|  | return err | 
|  | } | 
|  |  | 
|  | func appendPrefixed(list, newFiles []string, prefix string) []string { | 
|  | for _, file := range newFiles { | 
|  | list = append(list, filepath.Join(prefix, file)) | 
|  | } | 
|  | return list | 
|  | } | 
|  |  | 
|  | func main() { | 
|  | flag.Parse() | 
|  |  | 
|  | if len(*mainPkg) == 0 { | 
|  | fmt.Fprintf(os.Stderr, "-pkg argument is required.\n") | 
|  | os.Exit(1) | 
|  | } | 
|  |  | 
|  | var isDepfile bool | 
|  | switch *format { | 
|  | case "depfile": | 
|  | isDepfile = true | 
|  | case "cmake": | 
|  | isDepfile = false | 
|  | default: | 
|  | fmt.Fprintf(os.Stderr, "Unknown format: %q\n", *format) | 
|  | os.Exit(1) | 
|  | } | 
|  |  | 
|  | if isDepfile && len(*target) == 0 { | 
|  | fmt.Fprintf(os.Stderr, "-target argument is required for depfile.\n") | 
|  | os.Exit(1) | 
|  | } | 
|  |  | 
|  | done := make(map[string]struct{}) | 
|  | var files []string | 
|  | var recurse func(pkgName string) error | 
|  | recurse = func(pkgName string) error { | 
|  | pkg, err := build.Default.Import(pkgName, ".", 0) | 
|  | if err != nil { | 
|  | return err | 
|  | } | 
|  |  | 
|  | // Skip standard packages. | 
|  | if pkg.Goroot { | 
|  | return nil | 
|  | } | 
|  |  | 
|  | // Skip already-visited packages. | 
|  | if _, ok := done[pkg.Dir]; ok { | 
|  | return nil | 
|  | } | 
|  | done[pkg.Dir] = struct{}{} | 
|  |  | 
|  | files = appendPrefixed(files, pkg.GoFiles, pkg.Dir) | 
|  | files = appendPrefixed(files, pkg.CgoFiles, pkg.Dir) | 
|  | // Include ignored Go files. A subsequent change may cause them | 
|  | // to no longer be ignored. | 
|  | files = appendPrefixed(files, pkg.IgnoredGoFiles, pkg.Dir) | 
|  |  | 
|  | // Recurse into imports. | 
|  | for _, importName := range pkg.Imports { | 
|  | if err := recurse(importName); err != nil { | 
|  | return err | 
|  | } | 
|  | } | 
|  | return nil | 
|  | } | 
|  | if err := recurse(*mainPkg); err != nil { | 
|  | fmt.Fprintf(os.Stderr, "Error getting dependencies: %s\n", err) | 
|  | os.Exit(1) | 
|  | } | 
|  |  | 
|  | sort.Strings(files) | 
|  |  | 
|  | outFile := os.Stdout | 
|  | if len(*out) != 0 { | 
|  | var err error | 
|  | outFile, err = os.Create(*out) | 
|  | if err != nil { | 
|  | fmt.Fprintf(os.Stderr, "Error writing output: %s\n", err) | 
|  | os.Exit(1) | 
|  | } | 
|  | defer outFile.Close() | 
|  | } | 
|  |  | 
|  | var err error | 
|  | if isDepfile { | 
|  | err = writeDepfile(outFile, files) | 
|  | } else { | 
|  | err = writeCMake(outFile, files) | 
|  | } | 
|  | if err != nil { | 
|  | fmt.Fprintf(os.Stderr, "Error writing output: %s\n", err) | 
|  | os.Exit(1) | 
|  | } | 
|  | } |