1
2
3
4
5 package work
6
7 import (
8 "bufio"
9 "bytes"
10 "fmt"
11 "internal/buildcfg"
12 "internal/platform"
13 "io"
14 "log"
15 "os"
16 "path/filepath"
17 "runtime"
18 "strings"
19
20 "cmd/go/internal/base"
21 "cmd/go/internal/cfg"
22 "cmd/go/internal/fips140"
23 "cmd/go/internal/fsys"
24 "cmd/go/internal/gover"
25 "cmd/go/internal/load"
26 "cmd/go/internal/str"
27 "cmd/internal/quoted"
28 "crypto/sha1"
29 )
30
31
32 var ToolchainVersion = runtime.Version()
33
34
35
36 type gcToolchain struct{}
37
38 func (gcToolchain) compiler() string {
39 return base.Tool("compile")
40 }
41
42 func (gcToolchain) linker() string {
43 return base.Tool("link")
44 }
45
46 func pkgPath(a *Action) string {
47 p := a.Package
48 ppath := p.ImportPath
49 if cfg.BuildBuildmode == "plugin" {
50 ppath = pluginPath(a)
51 } else if p.Name == "main" && !p.Internal.ForceLibrary {
52 ppath = "main"
53 }
54 return ppath
55 }
56
57 func (gcToolchain) gc(b *Builder, a *Action, archive string, importcfg, embedcfg []byte, symabis string, asmhdr bool, pgoProfile string, gofiles []string) (ofile string, output []byte, err error) {
58 p := a.Package
59 sh := b.Shell(a)
60 objdir := a.Objdir
61 if archive != "" {
62 ofile = archive
63 } else {
64 out := "_go_.o"
65 ofile = objdir + out
66 }
67
68 pkgpath := pkgPath(a)
69 defaultGcFlags := []string{"-p", pkgpath}
70 vers := gover.Local()
71 if p.Module != nil {
72 v := p.Module.GoVersion
73 if v == "" {
74 v = gover.DefaultGoModVersion
75 }
76
77 if allowedVersion(v) {
78 vers = v
79 }
80 }
81 defaultGcFlags = append(defaultGcFlags, "-lang=go"+gover.Lang(vers))
82 if p.Standard {
83 defaultGcFlags = append(defaultGcFlags, "-std")
84 }
85
86
87
88
89
90 extFiles := len(p.CgoFiles) + len(p.CFiles) + len(p.CXXFiles) + len(p.MFiles) + len(p.FFiles) + len(p.SFiles) + len(p.SysoFiles) + len(p.SwigFiles) + len(p.SwigCXXFiles)
91 if p.Standard {
92 switch p.ImportPath {
93 case "bytes", "internal/poll", "net", "os":
94 fallthrough
95 case "runtime/metrics", "runtime/pprof", "runtime/trace":
96 fallthrough
97 case "sync", "syscall", "time":
98 extFiles++
99 }
100 }
101 if extFiles == 0 {
102 defaultGcFlags = append(defaultGcFlags, "-complete")
103 }
104 if cfg.BuildContext.InstallSuffix != "" {
105 defaultGcFlags = append(defaultGcFlags, "-installsuffix", cfg.BuildContext.InstallSuffix)
106 }
107 if a.buildID != "" {
108 defaultGcFlags = append(defaultGcFlags, "-buildid", a.buildID)
109 }
110 if p.Internal.OmitDebug || cfg.Goos == "plan9" || cfg.Goarch == "wasm" {
111 defaultGcFlags = append(defaultGcFlags, "-dwarf=false")
112 }
113 if strings.HasPrefix(ToolchainVersion, "go1") && !strings.Contains(os.Args[0], "go_bootstrap") {
114 defaultGcFlags = append(defaultGcFlags, "-goversion", ToolchainVersion)
115 }
116 if p.Internal.Cover.Cfg != "" {
117 defaultGcFlags = append(defaultGcFlags, "-coveragecfg="+p.Internal.Cover.Cfg)
118 }
119 if pgoProfile != "" {
120 defaultGcFlags = append(defaultGcFlags, "-pgoprofile="+pgoProfile)
121 }
122 if symabis != "" {
123 defaultGcFlags = append(defaultGcFlags, "-symabis", symabis)
124 }
125
126 gcflags := str.StringList(forcedGcflags, p.Internal.Gcflags)
127 if p.Internal.FuzzInstrument {
128 gcflags = append(gcflags, fuzzInstrumentFlags()...)
129 }
130
131 if c := gcBackendConcurrency(gcflags); c > 1 {
132 defaultGcFlags = append(defaultGcFlags, fmt.Sprintf("-c=%d", c))
133 }
134
135 args := []any{cfg.BuildToolexec, base.Tool("compile"), "-o", ofile, "-trimpath", a.trimpath(), defaultGcFlags, gcflags}
136 if p.Internal.LocalPrefix == "" {
137 args = append(args, "-nolocalimports")
138 } else {
139 args = append(args, "-D", p.Internal.LocalPrefix)
140 }
141 if importcfg != nil {
142 if err := sh.writeFile(objdir+"importcfg", importcfg); err != nil {
143 return "", nil, err
144 }
145 args = append(args, "-importcfg", objdir+"importcfg")
146 }
147 if embedcfg != nil {
148 if err := sh.writeFile(objdir+"embedcfg", embedcfg); err != nil {
149 return "", nil, err
150 }
151 args = append(args, "-embedcfg", objdir+"embedcfg")
152 }
153 if ofile == archive {
154 args = append(args, "-pack")
155 }
156 if asmhdr {
157 args = append(args, "-asmhdr", objdir+"go_asm.h")
158 }
159
160 for _, f := range gofiles {
161 f := mkAbs(p.Dir, f)
162
163
164
165
166
167
168
169
170
171
172
173
174
175 args = append(args, fsys.Actual(f))
176 }
177
178 output, err = sh.runOut(base.Cwd(), nil, args...)
179 return ofile, output, err
180 }
181
182
183 func gcBackendConcurrency(gcflags []string) int {
184
185 canDashC := concurrentGCBackendCompilationEnabledByDefault
186
187 switch e := os.Getenv("GO19CONCURRENTCOMPILATION"); e {
188 case "0":
189 canDashC = false
190 case "1":
191 canDashC = true
192 case "":
193
194 default:
195 log.Fatalf("GO19CONCURRENTCOMPILATION must be 0, 1, or unset, got %q", e)
196 }
197
198
199 if cfg.ExperimentErr != nil || cfg.Experiment.FieldTrack || cfg.Experiment.PreemptibleLoops {
200 canDashC = false
201 }
202
203 if !canDashC {
204 return 1
205 }
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230 c := runtime.GOMAXPROCS(0)
231 if cfg.BuildP == 1 {
232
233 return c
234 }
235
236 if c > 4 {
237 c = 4
238 }
239 return c
240 }
241
242
243
244 func (a *Action) trimpath() string {
245
246
247
248
249
250 objdir := strings.TrimSuffix(a.Objdir, string(filepath.Separator))
251 rewrite := ""
252
253 rewriteDir := a.Package.Dir
254 if cfg.BuildTrimpath {
255 importPath := a.Package.Internal.OrigImportPath
256 if m := a.Package.Module; m != nil && m.Version != "" {
257 rewriteDir = m.Path + "@" + m.Version + strings.TrimPrefix(importPath, m.Path)
258 } else {
259 rewriteDir = importPath
260 }
261 rewrite += a.Package.Dir + "=>" + rewriteDir + ";"
262 }
263
264
265
266
267
268 cgoFiles := make(map[string]bool)
269 for _, f := range a.Package.CgoFiles {
270 cgoFiles[f] = true
271 }
272
273
274
275
276
277 var overlayNonGoRewrites string
278 hasCgoOverlay := false
279 if fsys.OverlayFile != "" {
280 for _, filename := range a.Package.AllFiles() {
281 path := filename
282 if !filepath.IsAbs(path) {
283 path = filepath.Join(a.Package.Dir, path)
284 }
285 base := filepath.Base(path)
286 isGo := strings.HasSuffix(filename, ".go") || strings.HasSuffix(filename, ".s")
287 isCgo := cgoFiles[filename] || !isGo
288 if fsys.Replaced(path) {
289 if isCgo {
290 hasCgoOverlay = true
291 } else {
292 rewrite += fsys.Actual(path) + "=>" + filepath.Join(rewriteDir, base) + ";"
293 }
294 } else if isCgo {
295
296 if filepath.Dir(path) == a.Package.Dir {
297
298 overlayNonGoRewrites += filepath.Join(objdir, base) + "=>" + filepath.Join(rewriteDir, base) + ";"
299 }
300 } else {
301
302 }
303 }
304 }
305 if hasCgoOverlay {
306 rewrite += overlayNonGoRewrites
307 }
308 rewrite += objdir + "=>"
309
310 return rewrite
311 }
312
313 func asmArgs(a *Action, p *load.Package) []any {
314
315 inc := filepath.Join(cfg.GOROOT, "pkg", "include")
316 pkgpath := pkgPath(a)
317 args := []any{cfg.BuildToolexec, base.Tool("asm"), "-p", pkgpath, "-trimpath", a.trimpath(), "-I", a.Objdir, "-I", inc, "-D", "GOOS_" + cfg.Goos, "-D", "GOARCH_" + cfg.Goarch, forcedAsmflags, p.Internal.Asmflags}
318 if p.ImportPath == "runtime" && cfg.Goarch == "386" {
319 for _, arg := range forcedAsmflags {
320 if arg == "-dynlink" {
321 args = append(args, "-D=GOBUILDMODE_shared=1")
322 }
323 }
324 }
325
326 if cfg.Goarch == "386" {
327
328 args = append(args, "-D", "GO386_"+cfg.GO386)
329 }
330
331 if cfg.Goarch == "amd64" {
332
333 args = append(args, "-D", "GOAMD64_"+cfg.GOAMD64)
334 }
335
336 if cfg.Goarch == "mips" || cfg.Goarch == "mipsle" {
337
338 args = append(args, "-D", "GOMIPS_"+cfg.GOMIPS)
339 }
340
341 if cfg.Goarch == "mips64" || cfg.Goarch == "mips64le" {
342
343 args = append(args, "-D", "GOMIPS64_"+cfg.GOMIPS64)
344 }
345
346 if cfg.Goarch == "ppc64" || cfg.Goarch == "ppc64le" {
347
348
349 switch cfg.GOPPC64 {
350 case "power10":
351 args = append(args, "-D", "GOPPC64_power10")
352 fallthrough
353 case "power9":
354 args = append(args, "-D", "GOPPC64_power9")
355 fallthrough
356 default:
357 args = append(args, "-D", "GOPPC64_power8")
358 }
359 }
360
361 if cfg.Goarch == "riscv64" {
362
363 args = append(args, "-D", "GORISCV64_"+cfg.GORISCV64)
364 }
365
366 if cfg.Goarch == "arm" {
367
368
369 switch {
370 case strings.Contains(cfg.GOARM, "7"):
371 args = append(args, "-D", "GOARM_7")
372 fallthrough
373 case strings.Contains(cfg.GOARM, "6"):
374 args = append(args, "-D", "GOARM_6")
375 fallthrough
376 default:
377 args = append(args, "-D", "GOARM_5")
378 }
379 }
380
381 if cfg.Goarch == "arm64" {
382 g, err := buildcfg.ParseGoarm64(cfg.GOARM64)
383 if err == nil && g.LSE {
384 args = append(args, "-D", "GOARM64_LSE")
385 }
386 }
387
388 return args
389 }
390
391 func (gcToolchain) asm(b *Builder, a *Action, sfiles []string) ([]string, error) {
392 p := a.Package
393 args := asmArgs(a, p)
394
395 var ofiles []string
396 for _, sfile := range sfiles {
397 ofile := a.Objdir + sfile[:len(sfile)-len(".s")] + ".o"
398 ofiles = append(ofiles, ofile)
399 args1 := append(args, "-o", ofile, fsys.Actual(mkAbs(p.Dir, sfile)))
400 if err := b.Shell(a).run(p.Dir, p.ImportPath, nil, args1...); err != nil {
401 return nil, err
402 }
403 }
404 return ofiles, nil
405 }
406
407 func (gcToolchain) symabis(b *Builder, a *Action, sfiles []string) (string, error) {
408 sh := b.Shell(a)
409
410 mkSymabis := func(p *load.Package, sfiles []string, path string) error {
411 args := asmArgs(a, p)
412 args = append(args, "-gensymabis", "-o", path)
413 for _, sfile := range sfiles {
414 if p.ImportPath == "runtime/cgo" && strings.HasPrefix(sfile, "gcc_") {
415 continue
416 }
417 args = append(args, fsys.Actual(mkAbs(p.Dir, sfile)))
418 }
419
420
421
422
423 if err := sh.writeFile(a.Objdir+"go_asm.h", nil); err != nil {
424 return err
425 }
426
427 return sh.run(p.Dir, p.ImportPath, nil, args...)
428 }
429
430 var symabis string
431 p := a.Package
432 if len(sfiles) != 0 {
433 symabis = a.Objdir + "symabis"
434 if err := mkSymabis(p, sfiles, symabis); err != nil {
435 return "", err
436 }
437 }
438
439 return symabis, nil
440 }
441
442
443
444
445 func toolVerify(a *Action, b *Builder, p *load.Package, newTool string, ofile string, args []any) error {
446 newArgs := make([]any, len(args))
447 copy(newArgs, args)
448 newArgs[1] = base.Tool(newTool)
449 newArgs[3] = ofile + ".new"
450 if err := b.Shell(a).run(p.Dir, p.ImportPath, nil, newArgs...); err != nil {
451 return err
452 }
453 data1, err := os.ReadFile(ofile)
454 if err != nil {
455 return err
456 }
457 data2, err := os.ReadFile(ofile + ".new")
458 if err != nil {
459 return err
460 }
461 if !bytes.Equal(data1, data2) {
462 return fmt.Errorf("%s and %s produced different output files:\n%s\n%s", filepath.Base(args[1].(string)), newTool, strings.Join(str.StringList(args...), " "), strings.Join(str.StringList(newArgs...), " "))
463 }
464 os.Remove(ofile + ".new")
465 return nil
466 }
467
468 func (gcToolchain) pack(b *Builder, a *Action, afile string, ofiles []string) error {
469 absOfiles := make([]string, 0, len(ofiles))
470 for _, f := range ofiles {
471 absOfiles = append(absOfiles, mkAbs(a.Objdir, f))
472 }
473 absAfile := mkAbs(a.Objdir, afile)
474
475
476
477 if !cfg.BuildN {
478 if _, err := os.Stat(absAfile); err != nil {
479 base.Fatalf("os.Stat of archive file failed: %v", err)
480 }
481 }
482
483 p := a.Package
484 sh := b.Shell(a)
485 if cfg.BuildN || cfg.BuildX {
486 cmdline := str.StringList(base.Tool("pack"), "r", absAfile, absOfiles)
487 sh.ShowCmd(p.Dir, "%s # internal", joinUnambiguously(cmdline))
488 }
489 if cfg.BuildN {
490 return nil
491 }
492 if err := packInternal(absAfile, absOfiles); err != nil {
493 return sh.reportCmd("", "", nil, err)
494 }
495 return nil
496 }
497
498 func packInternal(afile string, ofiles []string) error {
499 dst, err := os.OpenFile(afile, os.O_WRONLY|os.O_APPEND, 0)
500 if err != nil {
501 return err
502 }
503 defer dst.Close()
504 w := bufio.NewWriter(dst)
505
506 for _, ofile := range ofiles {
507 src, err := os.Open(ofile)
508 if err != nil {
509 return err
510 }
511 fi, err := src.Stat()
512 if err != nil {
513 src.Close()
514 return err
515 }
516
517
518 name := fi.Name()
519 if len(name) > 16 {
520 name = name[:16]
521 } else {
522 name += strings.Repeat(" ", 16-len(name))
523 }
524 size := fi.Size()
525 fmt.Fprintf(w, "%s%-12d%-6d%-6d%-8o%-10d`\n",
526 name, 0, 0, 0, 0644, size)
527 n, err := io.Copy(w, src)
528 src.Close()
529 if err == nil && n < size {
530 err = io.ErrUnexpectedEOF
531 } else if err == nil && n > size {
532 err = fmt.Errorf("file larger than size reported by stat")
533 }
534 if err != nil {
535 return fmt.Errorf("copying %s to %s: %v", ofile, afile, err)
536 }
537 if size&1 != 0 {
538 w.WriteByte(0)
539 }
540 }
541
542 if err := w.Flush(); err != nil {
543 return err
544 }
545 return dst.Close()
546 }
547
548
549 func setextld(ldflags []string, compiler []string) ([]string, error) {
550 for _, f := range ldflags {
551 if f == "-extld" || strings.HasPrefix(f, "-extld=") {
552
553 return ldflags, nil
554 }
555 }
556 joined, err := quoted.Join(compiler)
557 if err != nil {
558 return nil, err
559 }
560 return append(ldflags, "-extld="+joined), nil
561 }
562
563
564
565
566
567
568
569
570 func pluginPath(a *Action) string {
571 p := a.Package
572 if p.ImportPath != "command-line-arguments" {
573 return p.ImportPath
574 }
575 h := sha1.New()
576 buildID := a.buildID
577 if a.Mode == "link" {
578
579
580
581
582
583
584
585
586
587 id := strings.Split(buildID, buildIDSeparator)
588 buildID = id[1] + buildIDSeparator + id[1]
589 }
590 fmt.Fprintf(h, "build ID: %s\n", buildID)
591 for _, file := range str.StringList(p.GoFiles, p.CgoFiles, p.SFiles) {
592 data, err := os.ReadFile(filepath.Join(p.Dir, file))
593 if err != nil {
594 base.Fatalf("go: %s", err)
595 }
596 h.Write(data)
597 }
598 return fmt.Sprintf("plugin/unnamed-%x", h.Sum(nil))
599 }
600
601 func (gcToolchain) ld(b *Builder, root *Action, targetPath, importcfg, mainpkg string) error {
602 cxx := len(root.Package.CXXFiles) > 0 || len(root.Package.SwigCXXFiles) > 0
603 for _, a := range root.Deps {
604 if a.Package != nil && (len(a.Package.CXXFiles) > 0 || len(a.Package.SwigCXXFiles) > 0) {
605 cxx = true
606 }
607 }
608 var ldflags []string
609 if cfg.BuildContext.InstallSuffix != "" {
610 ldflags = append(ldflags, "-installsuffix", cfg.BuildContext.InstallSuffix)
611 }
612 if root.Package.Internal.OmitDebug {
613 ldflags = append(ldflags, "-s", "-w")
614 }
615 if cfg.BuildBuildmode == "plugin" {
616 ldflags = append(ldflags, "-pluginpath", pluginPath(root))
617 }
618 if fips140.Enabled() {
619 ldflags = append(ldflags, "-fipso", filepath.Join(root.Objdir, "fips.o"))
620 }
621
622
623
624 if root.Package.Goroot && strings.HasPrefix(root.Package.ImportPath, "cmd/") {
625
626
627
628
629 if !platform.MustLinkExternal(cfg.Goos, cfg.Goarch, false) {
630 ldflags = append(ldflags, "-X=cmd/internal/objabi.buildID="+root.buildID)
631 }
632 }
633
634
635 if root.Package.DefaultGODEBUG != "" {
636 ldflags = append(ldflags, "-X=runtime.godebugDefault="+root.Package.DefaultGODEBUG)
637 }
638
639
640
641
642
643 var compiler []string
644 if cxx {
645 compiler = envList("CXX", cfg.DefaultCXX(cfg.Goos, cfg.Goarch))
646 } else {
647 compiler = envList("CC", cfg.DefaultCC(cfg.Goos, cfg.Goarch))
648 }
649 ldflags = append(ldflags, "-buildmode="+ldBuildmode)
650 if root.buildID != "" {
651 ldflags = append(ldflags, "-buildid="+root.buildID)
652 }
653 ldflags = append(ldflags, forcedLdflags...)
654 ldflags = append(ldflags, root.Package.Internal.Ldflags...)
655 ldflags, err := setextld(ldflags, compiler)
656 if err != nil {
657 return err
658 }
659
660
661
662
663
664
665
666
667
668
669
670
671 dir := "."
672 if cfg.BuildBuildmode == "c-shared" || cfg.BuildBuildmode == "plugin" {
673 dir, targetPath = filepath.Split(targetPath)
674 }
675
676 env := []string{}
677
678 if cfg.BuildTrimpath {
679 env = append(env, "GOROOT=")
680 } else {
681 env = append(env, "GOROOT="+cfg.GOROOT)
682 }
683 return b.Shell(root).run(dir, root.Package.ImportPath, env, cfg.BuildToolexec, base.Tool("link"), "-o", targetPath, "-importcfg", importcfg, ldflags, mainpkg)
684 }
685
686 func (gcToolchain) ldShared(b *Builder, root *Action, toplevelactions []*Action, targetPath, importcfg string, allactions []*Action) error {
687 ldflags := []string{"-installsuffix", cfg.BuildContext.InstallSuffix}
688 ldflags = append(ldflags, "-buildmode=shared")
689 ldflags = append(ldflags, forcedLdflags...)
690 ldflags = append(ldflags, root.Package.Internal.Ldflags...)
691 cxx := false
692 for _, a := range allactions {
693 if a.Package != nil && (len(a.Package.CXXFiles) > 0 || len(a.Package.SwigCXXFiles) > 0) {
694 cxx = true
695 }
696 }
697
698
699
700
701 var compiler []string
702 if cxx {
703 compiler = envList("CXX", cfg.DefaultCXX(cfg.Goos, cfg.Goarch))
704 } else {
705 compiler = envList("CC", cfg.DefaultCC(cfg.Goos, cfg.Goarch))
706 }
707 ldflags, err := setextld(ldflags, compiler)
708 if err != nil {
709 return err
710 }
711 for _, d := range toplevelactions {
712 if !strings.HasSuffix(d.Target, ".a") {
713 continue
714 }
715 ldflags = append(ldflags, d.Package.ImportPath+"="+d.Target)
716 }
717
718
719
720
721
722
723
724
725
726
727
728
729 dir, targetPath := filepath.Split(targetPath)
730
731 return b.Shell(root).run(dir, targetPath, nil, cfg.BuildToolexec, base.Tool("link"), "-o", targetPath, "-importcfg", importcfg, ldflags)
732 }
733
734 func (gcToolchain) cc(b *Builder, a *Action, ofile, cfile string) error {
735 return fmt.Errorf("%s: C source files not supported without cgo", mkAbs(a.Package.Dir, cfile))
736 }
737
View as plain text