]> Cypherpunks.ru repositories - gostls13.git/blobdiff - src/cmd/compile/internal/types2/version.go
go/types, types2: remove local version processing in favor of go/version
[gostls13.git] / src / cmd / compile / internal / types2 / version.go
index 37e86a2bb4c4e4dd78c7cfd4177133e0420638e5..12c86ef9fe6219716fcd649fa146aaceee6f6b48 100644 (file)
@@ -6,20 +6,58 @@ package types2
 
 import (
        "cmd/compile/internal/syntax"
-       "errors"
+       "fmt"
+       "go/version"
+       "internal/goversion"
        "strings"
 )
 
+// A goVersion is a Go language version string of the form "go1.%d"
+// where d is the minor version number. goVersion strings don't
+// contain release numbers ("go1.20.1" is not a valid goVersion).
+type goVersion string
+
+// asGoVersion returns v as a goVersion (e.g., "go1.20.1" becomes "go1.20").
+// If v is not a valid Go version, the result is the empty string.
+func asGoVersion(v string) goVersion {
+       return goVersion(version.Lang(v))
+}
+
+// isValid reports whether v is a valid Go version.
+func (v goVersion) isValid() bool {
+       return v != ""
+}
+
+// cmp returns -1, 0, or +1 depending on whether x < y, x == y, or x > y,
+// interpreted as Go versions.
+func (x goVersion) cmp(y goVersion) int {
+       return version.Compare(string(x), string(y))
+}
+
+var (
+       // Go versions that introduced language changes
+       go1_9  = asGoVersion("go1.9")
+       go1_13 = asGoVersion("go1.13")
+       go1_14 = asGoVersion("go1.14")
+       go1_17 = asGoVersion("go1.17")
+       go1_18 = asGoVersion("go1.18")
+       go1_20 = asGoVersion("go1.20")
+       go1_21 = asGoVersion("go1.21")
+
+       // current (deployed) Go version
+       go_current = asGoVersion(fmt.Sprintf("go1.%d", goversion.Version))
+)
+
 // langCompat reports an error if the representation of a numeric
 // literal is not compatible with the current language version.
 func (check *Checker) langCompat(lit *syntax.BasicLit) {
        s := lit.Value
-       if len(s) <= 2 || check.allowVersion(check.pkg, lit.Pos(), 1, 13) {
+       if len(s) <= 2 || check.allowVersion(check.pkg, lit, go1_13) {
                return
        }
        // len(s) > 2
        if strings.Contains(s, "_") {
-               check.versionErrorf(lit, "go1.13", "underscores in numeric literals")
+               check.versionErrorf(lit, go1_13, "underscores in numeric literals")
                return
        }
        if s[0] != '0' {
@@ -27,42 +65,54 @@ func (check *Checker) langCompat(lit *syntax.BasicLit) {
        }
        radix := s[1]
        if radix == 'b' || radix == 'B' {
-               check.versionErrorf(lit, "go1.13", "binary literals")
+               check.versionErrorf(lit, go1_13, "binary literals")
                return
        }
        if radix == 'o' || radix == 'O' {
-               check.versionErrorf(lit, "go1.13", "0o/0O-style octal literals")
+               check.versionErrorf(lit, go1_13, "0o/0O-style octal literals")
                return
        }
        if lit.Kind != syntax.IntLit && (radix == 'x' || radix == 'X') {
-               check.versionErrorf(lit, "go1.13", "hexadecimal floating-point literals")
+               check.versionErrorf(lit, go1_13, "hexadecimal floating-point literals")
        }
 }
 
-// allowVersion reports whether the given package
-// is allowed to use version major.minor.
-func (check *Checker) allowVersion(pkg *Package, pos syntax.Pos, major, minor int) bool {
+// allowVersion reports whether the given package is allowed to use version v.
+func (check *Checker) allowVersion(pkg *Package, at poser, v goVersion) bool {
        // We assume that imported packages have all been checked,
        // so we only have to check for the local package.
        if pkg != check.pkg {
                return true
        }
 
-       // If the source file declares its Go version, use that to decide.
-       if check.posVers != nil {
-               if v, ok := check.posVers[base(pos)]; ok && v.major >= 1 {
-                       return v.major > major || v.major == major && v.minor >= minor
-               }
-       }
+       // If no explicit file version is specified,
+       // fileVersion corresponds to the module version.
+       var fileVersion goVersion
+       if pos := at.Pos(); pos.IsKnown() {
+               // We need version.Lang below because file versions
+               // can be (unaltered) Config.GoVersion strings that
+               // may contain dot-release information.
+               fileVersion = asGoVersion(check.versions[base(pos)])
+       }
+       return !fileVersion.isValid() || fileVersion.cmp(v) >= 0
+}
 
-       // Otherwise fall back to the version in the checker.
-       ma, mi := check.version.major, check.version.minor
-       return ma == 0 && mi == 0 || ma > major || ma == major && mi >= minor
+// verifyVersionf is like allowVersion but also accepts a format string and arguments
+// which are used to report a version error if allowVersion returns false. It uses the
+// current package.
+func (check *Checker) verifyVersionf(at poser, v goVersion, format string, args ...interface{}) bool {
+       if !check.allowVersion(check.pkg, at, v) {
+               check.versionErrorf(at, v, format, args...)
+               return false
+       }
+       return true
 }
 
 // base finds the underlying PosBase of the source file containing pos,
 // skipping over intermediate PosBase layers created by //line directives.
+// The positions must be known.
 func base(pos syntax.Pos) *syntax.PosBase {
+       assert(pos.IsKnown())
        b := pos.Base()
        for {
                bb := b.Pos().Base()
@@ -73,66 +123,3 @@ func base(pos syntax.Pos) *syntax.PosBase {
        }
        return b
 }
-
-type version struct {
-       major, minor int
-}
-
-var errVersionSyntax = errors.New("invalid Go version syntax")
-
-// parseGoVersion parses a Go version string (such as "go1.12")
-// and returns the version, or an error. If s is the empty
-// string, the version is 0.0.
-func parseGoVersion(s string) (v version, err error) {
-       if s == "" {
-               return
-       }
-       if !strings.HasPrefix(s, "go") {
-               return version{}, errVersionSyntax
-       }
-       s = s[len("go"):]
-       i := 0
-       for ; i < len(s) && '0' <= s[i] && s[i] <= '9'; i++ {
-               if i >= 10 || i == 0 && s[i] == '0' {
-                       return version{}, errVersionSyntax
-               }
-               v.major = 10*v.major + int(s[i]) - '0'
-       }
-       if i > 0 && i == len(s) {
-               return
-       }
-       if i == 0 || s[i] != '.' {
-               return version{}, errVersionSyntax
-       }
-       s = s[i+1:]
-       if s == "0" {
-               // We really should not accept "go1.0",
-               // but we didn't reject it from the start
-               // and there are now programs that use it.
-               // So accept it.
-               return
-       }
-       i = 0
-       for ; i < len(s) && '0' <= s[i] && s[i] <= '9'; i++ {
-               if i >= 10 || i == 0 && s[i] == '0' {
-                       return version{}, errVersionSyntax
-               }
-               v.minor = 10*v.minor + int(s[i]) - '0'
-       }
-       if i > 0 && i == len(s) {
-               return
-       }
-       return version{}, errVersionSyntax
-}
-
-func (v version) equal(u version) bool {
-       return v.major == u.major && v.minor == u.minor
-}
-
-func (v version) before(u version) bool {
-       return v.major < u.major || v.major == u.major && v.minor < u.minor
-}
-
-func (v version) after(u version) bool {
-       return v.major > u.major || v.major == u.major && v.minor > u.minor
-}