|  | // Copyright 2009 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. | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "bytes" | 
|  | "fmt" | 
|  | "go/token" | 
|  | "io/ioutil" | 
|  | "os" | 
|  | "os/exec" | 
|  | ) | 
|  |  | 
|  | // run runs the command argv, feeding in stdin on standard input. | 
|  | // It returns the output to standard output and standard error. | 
|  | // ok indicates whether the command exited successfully. | 
|  | func run(stdin []byte, argv []string) (stdout, stderr []byte, ok bool) { | 
|  | if i := find(argv, "-xc"); i >= 0 && argv[len(argv)-1] == "-" { | 
|  | // Some compilers have trouble with standard input. | 
|  | // Others have trouble with -xc. | 
|  | // Avoid both problems by writing a file with a .c extension. | 
|  | f, err := ioutil.TempFile("", "cgo-gcc-input-") | 
|  | if err != nil { | 
|  | fatalf("%s", err) | 
|  | } | 
|  | name := f.Name() | 
|  | f.Close() | 
|  | if err := ioutil.WriteFile(name+".c", stdin, 0666); err != nil { | 
|  | os.Remove(name) | 
|  | fatalf("%s", err) | 
|  | } | 
|  | defer os.Remove(name) | 
|  | defer os.Remove(name + ".c") | 
|  |  | 
|  | // Build new argument list without -xc and trailing -. | 
|  | new := append(argv[:i:i], argv[i+1:len(argv)-1]...) | 
|  |  | 
|  | // Since we are going to write the file to a temporary directory, | 
|  | // we will need to add -I . explicitly to the command line: | 
|  | // any #include "foo" before would have looked in the current | 
|  | // directory as the directory "holding" standard input, but now | 
|  | // the temporary directory holds the input. | 
|  | // We've also run into compilers that reject "-I." but allow "-I", ".", | 
|  | // so be sure to use two arguments. | 
|  | // This matters mainly for people invoking cgo -godefs by hand. | 
|  | new = append(new, "-I", ".") | 
|  |  | 
|  | // Finish argument list with path to C file. | 
|  | new = append(new, name+".c") | 
|  |  | 
|  | argv = new | 
|  | stdin = nil | 
|  | } | 
|  |  | 
|  | p := exec.Command(argv[0], argv[1:]...) | 
|  | p.Stdin = bytes.NewReader(stdin) | 
|  | var bout, berr bytes.Buffer | 
|  | p.Stdout = &bout | 
|  | p.Stderr = &berr | 
|  | // Disable escape codes in clang error messages. | 
|  | p.Env = append(os.Environ(), "TERM=dumb") | 
|  | err := p.Run() | 
|  | if _, ok := err.(*exec.ExitError); err != nil && !ok { | 
|  | fatalf("%s", err) | 
|  | } | 
|  | ok = p.ProcessState.Success() | 
|  | stdout, stderr = bout.Bytes(), berr.Bytes() | 
|  | return | 
|  | } | 
|  |  | 
|  | func find(argv []string, target string) int { | 
|  | for i, arg := range argv { | 
|  | if arg == target { | 
|  | return i | 
|  | } | 
|  | } | 
|  | return -1 | 
|  | } | 
|  |  | 
|  | func lineno(pos token.Pos) string { | 
|  | return fset.Position(pos).String() | 
|  | } | 
|  |  | 
|  | // Die with an error message. | 
|  | func fatalf(msg string, args ...interface{}) { | 
|  | // If we've already printed other errors, they might have | 
|  | // caused the fatal condition. Assume they're enough. | 
|  | if nerrors == 0 { | 
|  | fmt.Fprintf(os.Stderr, msg+"\n", args...) | 
|  | } | 
|  | os.Exit(2) | 
|  | } | 
|  |  | 
|  | var nerrors int | 
|  |  | 
|  | func error_(pos token.Pos, msg string, args ...interface{}) { | 
|  | nerrors++ | 
|  | if pos.IsValid() { | 
|  | fmt.Fprintf(os.Stderr, "%s: ", fset.Position(pos).String()) | 
|  | } else { | 
|  | fmt.Fprintf(os.Stderr, "cgo: ") | 
|  | } | 
|  | fmt.Fprintf(os.Stderr, msg, args...) | 
|  | fmt.Fprintf(os.Stderr, "\n") | 
|  | } | 
|  |  | 
|  | // isName reports whether s is a valid C identifier | 
|  | func isName(s string) bool { | 
|  | for i, v := range s { | 
|  | if v != '_' && (v < 'A' || v > 'Z') && (v < 'a' || v > 'z') && (v < '0' || v > '9') { | 
|  | return false | 
|  | } | 
|  | if i == 0 && '0' <= v && v <= '9' { | 
|  | return false | 
|  | } | 
|  | } | 
|  | return s != "" | 
|  | } | 
|  |  | 
|  | func creat(name string) *os.File { | 
|  | f, err := os.Create(name) | 
|  | if err != nil { | 
|  | fatalf("%s", err) | 
|  | } | 
|  | return f | 
|  | } | 
|  |  | 
|  | func slashToUnderscore(c rune) rune { | 
|  | if c == '/' || c == '\\' || c == ':' { | 
|  | c = '_' | 
|  | } | 
|  | return c | 
|  | } |