| // Copyright 2013 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" |
| "os" |
| "strings" |
| "unicode" |
| ) |
| |
| var ( |
| nl = []byte("\n") |
| slashSlash = []byte("//") |
| plusBuild = []byte("+build") |
| ) |
| |
| func badfLine(f *File, line int, format string, args ...interface{}) { |
| msg := fmt.Sprintf(format, args...) |
| fmt.Fprintf(os.Stderr, "%s:%d: %s\n", f.name, line, msg) |
| setExit(1) |
| } |
| |
| // checkBuildTag checks that build tags are in the correct location and well-formed. |
| func checkBuildTag(f *File) { |
| if !vet("buildtags") { |
| return |
| } |
| |
| // we must look at the raw lines, as build tags may appear in non-Go |
| // files such as assembly files. |
| lines := bytes.SplitAfter(f.content, nl) |
| |
| // lineWithComment reports whether a line corresponds to a comment in |
| // the source file. If the source file wasn't Go, the function always |
| // returns true. |
| lineWithComment := func(line int) bool { |
| if f.file == nil { |
| // Current source file is not Go, so be conservative. |
| return true |
| } |
| for _, group := range f.file.Comments { |
| startLine := f.fset.Position(group.Pos()).Line |
| endLine := f.fset.Position(group.End()).Line |
| if startLine <= line && line <= endLine { |
| return true |
| } |
| } |
| return false |
| } |
| |
| // Determine cutpoint where +build comments are no longer valid. |
| // They are valid in leading // comments in the file followed by |
| // a blank line. |
| var cutoff int |
| for i, line := range lines { |
| line = bytes.TrimSpace(line) |
| if len(line) == 0 { |
| cutoff = i |
| continue |
| } |
| if bytes.HasPrefix(line, slashSlash) { |
| continue |
| } |
| break |
| } |
| |
| for i, line := range lines { |
| line = bytes.TrimSpace(line) |
| if !bytes.HasPrefix(line, slashSlash) { |
| continue |
| } |
| if !bytes.Contains(line, plusBuild) { |
| // Check that the comment contains "+build" early, to |
| // avoid unnecessary lineWithComment calls that may |
| // incur linear searches. |
| continue |
| } |
| if !lineWithComment(i + 1) { |
| // This is a line in a Go source file that looks like a |
| // comment, but actually isn't - such as part of a raw |
| // string. |
| continue |
| } |
| |
| text := bytes.TrimSpace(line[2:]) |
| if bytes.HasPrefix(text, plusBuild) { |
| fields := bytes.Fields(text) |
| if !bytes.Equal(fields[0], plusBuild) { |
| // Comment is something like +buildasdf not +build. |
| badfLine(f, i+1, "possible malformed +build comment") |
| continue |
| } |
| if i >= cutoff { |
| badfLine(f, i+1, "+build comment must appear before package clause and be followed by a blank line") |
| continue |
| } |
| // Check arguments. |
| Args: |
| for _, arg := range fields[1:] { |
| for _, elem := range strings.Split(string(arg), ",") { |
| if strings.HasPrefix(elem, "!!") { |
| badfLine(f, i+1, "invalid double negative in build constraint: %s", arg) |
| break Args |
| } |
| elem = strings.TrimPrefix(elem, "!") |
| for _, c := range elem { |
| if !unicode.IsLetter(c) && !unicode.IsDigit(c) && c != '_' && c != '.' { |
| badfLine(f, i+1, "invalid non-alphanumeric build constraint: %s", arg) |
| break Args |
| } |
| } |
| } |
| } |
| continue |
| } |
| // Comment with +build but not at beginning. |
| if i < cutoff { |
| badfLine(f, i+1, "possible malformed +build comment") |
| continue |
| } |
| } |
| } |