1
2
3
4
5
6
7
8
9
10
11
12 package buildcfg
13
14 import (
15 "fmt"
16 "os"
17 "path/filepath"
18 "strconv"
19 "strings"
20 )
21
22 var (
23 GOROOT = os.Getenv("GOROOT")
24 GOARCH = envOr("GOARCH", defaultGOARCH)
25 GOOS = envOr("GOOS", defaultGOOS)
26 GO386 = envOr("GO386", DefaultGO386)
27 GOAMD64 = goamd64()
28 GOARM = goarm()
29 GOARM64 = goarm64()
30 GOMIPS = gomips()
31 GOMIPS64 = gomips64()
32 GOPPC64 = goppc64()
33 GORISCV64 = goriscv64()
34 GOWASM = gowasm()
35 ToolTags = toolTags()
36 GO_LDSO = defaultGO_LDSO
37 GOFIPS140 = gofips140()
38 Version = version
39 )
40
41
42 var Error error
43
44
45 func Check() {
46 if Error != nil {
47 fmt.Fprintf(os.Stderr, "%s: %v\n", filepath.Base(os.Args[0]), Error)
48 os.Exit(2)
49 }
50 }
51
52 func envOr(key, value string) string {
53 if x := os.Getenv(key); x != "" {
54 return x
55 }
56 return value
57 }
58
59 func goamd64() int {
60 switch v := envOr("GOAMD64", DefaultGOAMD64); v {
61 case "v1":
62 return 1
63 case "v2":
64 return 2
65 case "v3":
66 return 3
67 case "v4":
68 return 4
69 }
70 Error = fmt.Errorf("invalid GOAMD64: must be v1, v2, v3, v4")
71 return int(DefaultGOAMD64[len("v")] - '0')
72 }
73
74 func gofips140() string {
75 v := envOr("GOFIPS140", DefaultGOFIPS140)
76 switch v {
77 case "off", "latest", "inprocess", "certified":
78 return v
79 }
80 if isFIPSVersion(v) {
81 return v
82 }
83 Error = fmt.Errorf("invalid GOFIPS140: must be off, latest, inprocess, certified, or vX.Y.Z")
84 return DefaultGOFIPS140
85 }
86
87
88
89 func isFIPSVersion(v string) bool {
90 if !strings.HasPrefix(v, "v") {
91 return false
92 }
93 v, ok := skipNum(v[len("v"):])
94 if !ok || !strings.HasPrefix(v, ".") {
95 return false
96 }
97 v, ok = skipNum(v[len("."):])
98 if !ok || !strings.HasPrefix(v, ".") {
99 return false
100 }
101 v, ok = skipNum(v[len("."):])
102 hasHash := strings.HasPrefix(v, "-") && len(v) == len("-")+8
103 return ok && (v == "" || hasHash)
104 }
105
106
107
108 func skipNum(s string) (rest string, ok bool) {
109 i := 0
110 for i < len(s) && '0' <= s[i] && s[i] <= '9' {
111 i++
112 }
113 return s[i:], i > 0
114 }
115
116 type GoarmFeatures struct {
117 Version int
118 SoftFloat bool
119 }
120
121 func (g GoarmFeatures) String() string {
122 armStr := strconv.Itoa(g.Version)
123 if g.SoftFloat {
124 armStr += ",softfloat"
125 } else {
126 armStr += ",hardfloat"
127 }
128 return armStr
129 }
130
131 func goarm() (g GoarmFeatures) {
132 const (
133 softFloatOpt = ",softfloat"
134 hardFloatOpt = ",hardfloat"
135 )
136 def := DefaultGOARM
137 if GOOS == "android" && GOARCH == "arm" {
138
139 def = "7"
140 }
141 v := envOr("GOARM", def)
142
143 floatSpecified := false
144 if strings.HasSuffix(v, softFloatOpt) {
145 g.SoftFloat = true
146 floatSpecified = true
147 v = v[:len(v)-len(softFloatOpt)]
148 }
149 if strings.HasSuffix(v, hardFloatOpt) {
150 floatSpecified = true
151 v = v[:len(v)-len(hardFloatOpt)]
152 }
153
154 switch v {
155 case "5":
156 g.Version = 5
157 case "6":
158 g.Version = 6
159 case "7":
160 g.Version = 7
161 default:
162 Error = fmt.Errorf("invalid GOARM: must start with 5, 6, or 7, and may optionally end in either %q or %q", hardFloatOpt, softFloatOpt)
163 g.Version = int(def[0] - '0')
164 }
165
166
167 if !floatSpecified && g.Version == 5 {
168 g.SoftFloat = true
169 }
170 return
171 }
172
173 type Goarm64Features struct {
174 Version string
175
176 LSE bool
177
178
179
180
181
182 Crypto bool
183 }
184
185 func (g Goarm64Features) String() string {
186 arm64Str := g.Version
187 if g.LSE {
188 arm64Str += ",lse"
189 }
190 if g.Crypto {
191 arm64Str += ",crypto"
192 }
193 return arm64Str
194 }
195
196 func ParseGoarm64(v string) (g Goarm64Features, e error) {
197 const (
198 lseOpt = ",lse"
199 cryptoOpt = ",crypto"
200 )
201
202 g.LSE = false
203 g.Crypto = false
204
205 for {
206 if strings.HasSuffix(v, lseOpt) {
207 g.LSE = true
208 v = v[:len(v)-len(lseOpt)]
209 continue
210 }
211
212 if strings.HasSuffix(v, cryptoOpt) {
213 g.Crypto = true
214 v = v[:len(v)-len(cryptoOpt)]
215 continue
216 }
217
218 break
219 }
220
221 switch v {
222 case "v8.0":
223 g.Version = v
224 case "v8.1", "v8.2", "v8.3", "v8.4", "v8.5", "v8.6", "v8.7", "v8.8", "v8.9",
225 "v9.0", "v9.1", "v9.2", "v9.3", "v9.4", "v9.5":
226 g.Version = v
227
228 g.LSE = true
229 default:
230 e = fmt.Errorf("invalid GOARM64: must start with v8.{0-9} or v9.{0-5} and may optionally end in %q and/or %q",
231 lseOpt, cryptoOpt)
232 g.Version = DefaultGOARM64
233 }
234
235 return
236 }
237
238 func goarm64() (g Goarm64Features) {
239 g, Error = ParseGoarm64(envOr("GOARM64", DefaultGOARM64))
240 return
241 }
242
243
244
245 func (g Goarm64Features) Supports(s string) bool {
246
247 if len(s) != 4 {
248 return false
249 }
250
251 major := s[1]
252 minor := s[3]
253
254
255 if major < '8' || major > '9' ||
256 minor < '0' || minor > '9' ||
257 s[0] != 'v' || s[2] != '.' {
258 return false
259 }
260
261 g_major := g.Version[1]
262 g_minor := g.Version[3]
263
264 if major == g_major {
265 return minor <= g_minor
266 } else if g_major == '9' {
267
268 return minor <= g_minor+5
269 } else {
270 return false
271 }
272 }
273
274 func gomips() string {
275 switch v := envOr("GOMIPS", DefaultGOMIPS); v {
276 case "hardfloat", "softfloat":
277 return v
278 }
279 Error = fmt.Errorf("invalid GOMIPS: must be hardfloat, softfloat")
280 return DefaultGOMIPS
281 }
282
283 func gomips64() string {
284 switch v := envOr("GOMIPS64", DefaultGOMIPS64); v {
285 case "hardfloat", "softfloat":
286 return v
287 }
288 Error = fmt.Errorf("invalid GOMIPS64: must be hardfloat, softfloat")
289 return DefaultGOMIPS64
290 }
291
292 func goppc64() int {
293 switch v := envOr("GOPPC64", DefaultGOPPC64); v {
294 case "power8":
295 return 8
296 case "power9":
297 return 9
298 case "power10":
299 return 10
300 }
301 Error = fmt.Errorf("invalid GOPPC64: must be power8, power9, power10")
302 return int(DefaultGOPPC64[len("power")] - '0')
303 }
304
305 func goriscv64() int {
306 switch v := envOr("GORISCV64", DefaultGORISCV64); v {
307 case "rva20u64":
308 return 20
309 case "rva22u64":
310 return 22
311 case "rva23u64":
312 return 23
313 }
314 Error = fmt.Errorf("invalid GORISCV64: must be rva20u64, rva22u64, rva23u64")
315 v := DefaultGORISCV64[len("rva"):]
316 i := strings.IndexFunc(v, func(r rune) bool {
317 return r < '0' || r > '9'
318 })
319 year, _ := strconv.Atoi(v[:i])
320 return year
321 }
322
323 type gowasmFeatures struct {
324 SatConv bool
325 SignExt bool
326 }
327
328 func (f gowasmFeatures) String() string {
329 var flags []string
330 if f.SatConv {
331 flags = append(flags, "satconv")
332 }
333 if f.SignExt {
334 flags = append(flags, "signext")
335 }
336 return strings.Join(flags, ",")
337 }
338
339 func gowasm() (f gowasmFeatures) {
340 for _, opt := range strings.Split(envOr("GOWASM", ""), ",") {
341 switch opt {
342 case "satconv":
343 f.SatConv = true
344 case "signext":
345 f.SignExt = true
346 case "":
347
348 default:
349 Error = fmt.Errorf("invalid GOWASM: no such feature %q", opt)
350 }
351 }
352 return
353 }
354
355 func Getgoextlinkenabled() string {
356 return envOr("GO_EXTLINK_ENABLED", defaultGO_EXTLINK_ENABLED)
357 }
358
359 func toolTags() []string {
360 tags := experimentTags()
361 tags = append(tags, gogoarchTags()...)
362 return tags
363 }
364
365 func experimentTags() []string {
366 var list []string
367
368
369
370
371
372 for _, exp := range Experiment.Enabled() {
373 list = append(list, "goexperiment."+exp)
374 }
375 return list
376 }
377
378
379
380 func GOGOARCH() (name, value string) {
381 switch GOARCH {
382 case "386":
383 return "GO386", GO386
384 case "amd64":
385 return "GOAMD64", fmt.Sprintf("v%d", GOAMD64)
386 case "arm":
387 return "GOARM", GOARM.String()
388 case "arm64":
389 return "GOARM64", GOARM64.String()
390 case "mips", "mipsle":
391 return "GOMIPS", GOMIPS
392 case "mips64", "mips64le":
393 return "GOMIPS64", GOMIPS64
394 case "ppc64", "ppc64le":
395 return "GOPPC64", fmt.Sprintf("power%d", GOPPC64)
396 case "wasm":
397 return "GOWASM", GOWASM.String()
398 }
399 return "", ""
400 }
401
402 func gogoarchTags() []string {
403 switch GOARCH {
404 case "386":
405 return []string{GOARCH + "." + GO386}
406 case "amd64":
407 var list []string
408 for i := 1; i <= GOAMD64; i++ {
409 list = append(list, fmt.Sprintf("%s.v%d", GOARCH, i))
410 }
411 return list
412 case "arm":
413 var list []string
414 for i := 5; i <= GOARM.Version; i++ {
415 list = append(list, fmt.Sprintf("%s.%d", GOARCH, i))
416 }
417 return list
418 case "arm64":
419 var list []string
420 major := int(GOARM64.Version[1] - '0')
421 minor := int(GOARM64.Version[3] - '0')
422 for i := 0; i <= minor; i++ {
423 list = append(list, fmt.Sprintf("%s.v%d.%d", GOARCH, major, i))
424 }
425
426 if major == 9 {
427 for i := 0; i <= minor+5 && i <= 9; i++ {
428 list = append(list, fmt.Sprintf("%s.v%d.%d", GOARCH, 8, i))
429 }
430 }
431 return list
432 case "mips", "mipsle":
433 return []string{GOARCH + "." + GOMIPS}
434 case "mips64", "mips64le":
435 return []string{GOARCH + "." + GOMIPS64}
436 case "ppc64", "ppc64le":
437 var list []string
438 for i := 8; i <= GOPPC64; i++ {
439 list = append(list, fmt.Sprintf("%s.power%d", GOARCH, i))
440 }
441 return list
442 case "riscv64":
443 list := []string{GOARCH + "." + "rva20u64"}
444 if GORISCV64 >= 22 {
445 list = append(list, GOARCH+"."+"rva22u64")
446 }
447 if GORISCV64 >= 23 {
448 list = append(list, GOARCH+"."+"rva23u64")
449 }
450 return list
451 case "wasm":
452 var list []string
453 if GOWASM.SatConv {
454 list = append(list, GOARCH+".satconv")
455 }
456 if GOWASM.SignExt {
457 list = append(list, GOARCH+".signext")
458 }
459 return list
460 }
461 return nil
462 }
463
View as plain text