1
2
3
4
5
6
7 package work
8
9 import (
10 "bytes"
11 "cmd/internal/cov/covcmd"
12 "cmd/internal/pathcache"
13 "context"
14 "crypto/sha256"
15 "encoding/json"
16 "errors"
17 "fmt"
18 "go/token"
19 "internal/lazyregexp"
20 "io"
21 "io/fs"
22 "log"
23 "math/rand"
24 "os"
25 "os/exec"
26 "path/filepath"
27 "regexp"
28 "runtime"
29 "slices"
30 "sort"
31 "strconv"
32 "strings"
33 "sync"
34 "time"
35
36 "cmd/go/internal/base"
37 "cmd/go/internal/cache"
38 "cmd/go/internal/cfg"
39 "cmd/go/internal/fsys"
40 "cmd/go/internal/gover"
41 "cmd/go/internal/load"
42 "cmd/go/internal/modload"
43 "cmd/go/internal/str"
44 "cmd/go/internal/trace"
45 "cmd/internal/buildid"
46 "cmd/internal/quoted"
47 "cmd/internal/sys"
48 )
49
50 const DefaultCFlags = "-O2 -g"
51
52
53
54 func actionList(root *Action) []*Action {
55 seen := map[*Action]bool{}
56 all := []*Action{}
57 var walk func(*Action)
58 walk = func(a *Action) {
59 if seen[a] {
60 return
61 }
62 seen[a] = true
63 for _, a1 := range a.Deps {
64 walk(a1)
65 }
66 all = append(all, a)
67 }
68 walk(root)
69 return all
70 }
71
72
73 func (b *Builder) Do(ctx context.Context, root *Action) {
74 ctx, span := trace.StartSpan(ctx, "exec.Builder.Do ("+root.Mode+" "+root.Target+")")
75 defer span.Done()
76
77 if !b.IsCmdList {
78
79 c := cache.Default()
80 defer func() {
81 if err := c.Close(); err != nil {
82 base.Fatalf("go: failed to trim cache: %v", err)
83 }
84 }()
85 }
86
87
88
89
90
91
92
93
94
95
96
97
98 all := actionList(root)
99 for i, a := range all {
100 a.priority = i
101 }
102
103
104 writeActionGraph := func() {
105 if file := cfg.DebugActiongraph; file != "" {
106 if strings.HasSuffix(file, ".go") {
107
108
109 base.Fatalf("go: refusing to write action graph to %v\n", file)
110 }
111 js := actionGraphJSON(root)
112 if err := os.WriteFile(file, []byte(js), 0666); err != nil {
113 fmt.Fprintf(os.Stderr, "go: writing action graph: %v\n", err)
114 base.SetExitStatus(1)
115 }
116 }
117 }
118 writeActionGraph()
119
120 b.readySema = make(chan bool, len(all))
121
122
123 for _, a := range all {
124 for _, a1 := range a.Deps {
125 a1.triggers = append(a1.triggers, a)
126 }
127 a.pending = len(a.Deps)
128 if a.pending == 0 {
129 b.ready.push(a)
130 b.readySema <- true
131 }
132 }
133
134
135
136 handle := func(ctx context.Context, a *Action) {
137 if a.json != nil {
138 a.json.TimeStart = time.Now()
139 }
140 var err error
141 if a.Actor != nil && (a.Failed == nil || a.IgnoreFail) {
142
143 desc := "Executing action (" + a.Mode
144 if a.Package != nil {
145 desc += " " + a.Package.Desc()
146 }
147 desc += ")"
148 ctx, span := trace.StartSpan(ctx, desc)
149 a.traceSpan = span
150 for _, d := range a.Deps {
151 trace.Flow(ctx, d.traceSpan, a.traceSpan)
152 }
153 err = a.Actor.Act(b, ctx, a)
154 span.Done()
155 }
156 if a.json != nil {
157 a.json.TimeDone = time.Now()
158 }
159
160
161
162 b.exec.Lock()
163 defer b.exec.Unlock()
164
165 if err != nil {
166 if b.AllowErrors && a.Package != nil {
167 if a.Package.Error == nil {
168 a.Package.Error = &load.PackageError{Err: err}
169 a.Package.Incomplete = true
170 }
171 } else {
172 var ipe load.ImportPathError
173 if a.Package != nil && (!errors.As(err, &ipe) || ipe.ImportPath() != a.Package.ImportPath) {
174 err = fmt.Errorf("%s: %v", a.Package.ImportPath, err)
175 }
176 sh := b.Shell(a)
177 sh.Errorf("%s", err)
178 }
179 if a.Failed == nil {
180 a.Failed = a
181 }
182 }
183
184 for _, a0 := range a.triggers {
185 if a.Failed != nil {
186 a0.Failed = a.Failed
187 }
188 if a0.pending--; a0.pending == 0 {
189 b.ready.push(a0)
190 b.readySema <- true
191 }
192 }
193
194 if a == root {
195 close(b.readySema)
196 }
197 }
198
199 var wg sync.WaitGroup
200
201
202
203
204
205 par := cfg.BuildP
206 if cfg.BuildN {
207 par = 1
208 }
209 for i := 0; i < par; i++ {
210 wg.Add(1)
211 go func() {
212 ctx := trace.StartGoroutine(ctx)
213 defer wg.Done()
214 for {
215 select {
216 case _, ok := <-b.readySema:
217 if !ok {
218 return
219 }
220
221
222 b.exec.Lock()
223 a := b.ready.pop()
224 b.exec.Unlock()
225 handle(ctx, a)
226 case <-base.Interrupted:
227 base.SetExitStatus(1)
228 return
229 }
230 }
231 }()
232 }
233
234 wg.Wait()
235
236
237 writeActionGraph()
238 }
239
240
241 func (b *Builder) buildActionID(a *Action) cache.ActionID {
242 p := a.Package
243 h := cache.NewHash("build " + p.ImportPath)
244
245
246
247
248
249
250 fmt.Fprintf(h, "compile\n")
251
252
253
254 if cfg.BuildTrimpath {
255
256
257
258 if p.Module != nil {
259 fmt.Fprintf(h, "module %s@%s\n", p.Module.Path, p.Module.Version)
260 }
261 } else if p.Goroot {
262
263
264
265
266
267
268
269
270
271
272
273
274
275 } else if !strings.HasPrefix(p.Dir, b.WorkDir) {
276
277
278
279 fmt.Fprintf(h, "dir %s\n", p.Dir)
280 }
281
282 if p.Module != nil {
283 fmt.Fprintf(h, "go %s\n", p.Module.GoVersion)
284 }
285 fmt.Fprintf(h, "goos %s goarch %s\n", cfg.Goos, cfg.Goarch)
286 fmt.Fprintf(h, "import %q\n", p.ImportPath)
287 fmt.Fprintf(h, "omitdebug %v standard %v local %v prefix %q\n", p.Internal.OmitDebug, p.Standard, p.Internal.Local, p.Internal.LocalPrefix)
288 if cfg.BuildTrimpath {
289 fmt.Fprintln(h, "trimpath")
290 }
291 if p.Internal.ForceLibrary {
292 fmt.Fprintf(h, "forcelibrary\n")
293 }
294 if len(p.CgoFiles)+len(p.SwigFiles)+len(p.SwigCXXFiles) > 0 {
295 fmt.Fprintf(h, "cgo %q\n", b.toolID("cgo"))
296 cppflags, cflags, cxxflags, fflags, ldflags, _ := b.CFlags(p)
297
298 ccExe := b.ccExe()
299 fmt.Fprintf(h, "CC=%q %q %q %q\n", ccExe, cppflags, cflags, ldflags)
300
301
302 if ccID, _, err := b.gccToolID(ccExe[0], "c"); err == nil {
303 fmt.Fprintf(h, "CC ID=%q\n", ccID)
304 }
305 if len(p.CXXFiles)+len(p.SwigCXXFiles) > 0 {
306 cxxExe := b.cxxExe()
307 fmt.Fprintf(h, "CXX=%q %q\n", cxxExe, cxxflags)
308 if cxxID, _, err := b.gccToolID(cxxExe[0], "c++"); err == nil {
309 fmt.Fprintf(h, "CXX ID=%q\n", cxxID)
310 }
311 }
312 if len(p.FFiles) > 0 {
313 fcExe := b.fcExe()
314 fmt.Fprintf(h, "FC=%q %q\n", fcExe, fflags)
315 if fcID, _, err := b.gccToolID(fcExe[0], "f95"); err == nil {
316 fmt.Fprintf(h, "FC ID=%q\n", fcID)
317 }
318 }
319
320 }
321 if p.Internal.Cover.Mode != "" {
322 fmt.Fprintf(h, "cover %q %q\n", p.Internal.Cover.Mode, b.toolID("cover"))
323 }
324 if p.Internal.FuzzInstrument {
325 if fuzzFlags := fuzzInstrumentFlags(); fuzzFlags != nil {
326 fmt.Fprintf(h, "fuzz %q\n", fuzzFlags)
327 }
328 }
329 if p.Internal.BuildInfo != nil {
330 fmt.Fprintf(h, "modinfo %q\n", p.Internal.BuildInfo.String())
331 }
332
333
334 switch cfg.BuildToolchainName {
335 default:
336 base.Fatalf("buildActionID: unknown build toolchain %q", cfg.BuildToolchainName)
337 case "gc":
338 fmt.Fprintf(h, "compile %s %q %q\n", b.toolID("compile"), forcedGcflags, p.Internal.Gcflags)
339 if len(p.SFiles) > 0 {
340 fmt.Fprintf(h, "asm %q %q %q\n", b.toolID("asm"), forcedAsmflags, p.Internal.Asmflags)
341 }
342
343
344 key, val, _ := cfg.GetArchEnv()
345 fmt.Fprintf(h, "%s=%s\n", key, val)
346
347 if cfg.CleanGOEXPERIMENT != "" {
348 fmt.Fprintf(h, "GOEXPERIMENT=%q\n", cfg.CleanGOEXPERIMENT)
349 }
350
351
352
353
354
355
356 magic := []string{
357 "GOCLOBBERDEADHASH",
358 "GOSSAFUNC",
359 "GOSSADIR",
360 "GOCOMPILEDEBUG",
361 }
362 for _, env := range magic {
363 if x := os.Getenv(env); x != "" {
364 fmt.Fprintf(h, "magic %s=%s\n", env, x)
365 }
366 }
367
368 case "gccgo":
369 id, _, err := b.gccToolID(BuildToolchain.compiler(), "go")
370 if err != nil {
371 base.Fatalf("%v", err)
372 }
373 fmt.Fprintf(h, "compile %s %q %q\n", id, forcedGccgoflags, p.Internal.Gccgoflags)
374 fmt.Fprintf(h, "pkgpath %s\n", gccgoPkgpath(p))
375 fmt.Fprintf(h, "ar %q\n", BuildToolchain.(gccgoToolchain).ar())
376 if len(p.SFiles) > 0 {
377 id, _, _ = b.gccToolID(BuildToolchain.compiler(), "assembler-with-cpp")
378
379
380 fmt.Fprintf(h, "asm %q\n", id)
381 }
382 }
383
384
385 inputFiles := str.StringList(
386 p.GoFiles,
387 p.CgoFiles,
388 p.CFiles,
389 p.CXXFiles,
390 p.FFiles,
391 p.MFiles,
392 p.HFiles,
393 p.SFiles,
394 p.SysoFiles,
395 p.SwigFiles,
396 p.SwigCXXFiles,
397 p.EmbedFiles,
398 )
399 for _, file := range inputFiles {
400 fmt.Fprintf(h, "file %s %s\n", file, b.fileHash(filepath.Join(p.Dir, file)))
401 }
402 for _, a1 := range a.Deps {
403 p1 := a1.Package
404 if p1 != nil {
405 fmt.Fprintf(h, "import %s %s\n", p1.ImportPath, contentID(a1.buildID))
406 }
407 if a1.Mode == "preprocess PGO profile" {
408 fmt.Fprintf(h, "pgofile %s\n", b.fileHash(a1.built))
409 }
410 }
411
412 return h.Sum()
413 }
414
415
416
417 func (b *Builder) needCgoHdr(a *Action) bool {
418
419 if !b.IsCmdList && (a.Package.UsesCgo() || a.Package.UsesSwig()) && (cfg.BuildBuildmode == "c-archive" || cfg.BuildBuildmode == "c-shared") {
420 for _, t1 := range a.triggers {
421 if t1.Mode == "install header" {
422 return true
423 }
424 }
425 for _, t1 := range a.triggers {
426 for _, t2 := range t1.triggers {
427 if t2.Mode == "install header" {
428 return true
429 }
430 }
431 }
432 }
433 return false
434 }
435
436
437
438
439 func allowedVersion(v string) bool {
440
441 if v == "" {
442 return true
443 }
444 return gover.Compare(gover.Local(), v) >= 0
445 }
446
447 const (
448 needBuild uint32 = 1 << iota
449 needCgoHdr
450 needVet
451 needCompiledGoFiles
452 needCovMetaFile
453 needStale
454 )
455
456
457
458 func (b *Builder) build(ctx context.Context, a *Action) (err error) {
459 p := a.Package
460 sh := b.Shell(a)
461
462 bit := func(x uint32, b bool) uint32 {
463 if b {
464 return x
465 }
466 return 0
467 }
468
469 cachedBuild := false
470 needCovMeta := p.Internal.Cover.GenMeta
471 need := bit(needBuild, !b.IsCmdList && a.needBuild || b.NeedExport) |
472 bit(needCgoHdr, b.needCgoHdr(a)) |
473 bit(needVet, a.needVet) |
474 bit(needCovMetaFile, needCovMeta) |
475 bit(needCompiledGoFiles, b.NeedCompiledGoFiles)
476
477 if !p.BinaryOnly {
478 if b.useCache(a, b.buildActionID(a), p.Target, need&needBuild != 0) {
479
480
481
482
483
484 cachedBuild = true
485 a.output = []byte{}
486 need &^= needBuild
487 if b.NeedExport {
488 p.Export = a.built
489 p.BuildID = a.buildID
490 }
491 if need&needCompiledGoFiles != 0 {
492 if err := b.loadCachedCompiledGoFiles(a); err == nil {
493 need &^= needCompiledGoFiles
494 }
495 }
496 }
497
498
499
500 if !cachedBuild && need&needCompiledGoFiles != 0 {
501 if err := b.loadCachedCompiledGoFiles(a); err == nil {
502 need &^= needCompiledGoFiles
503 }
504 }
505
506 if need == 0 {
507 return nil
508 }
509 defer b.flushOutput(a)
510 }
511
512 defer func() {
513 if err != nil && b.IsCmdList && b.NeedError && p.Error == nil {
514 p.Error = &load.PackageError{Err: err}
515 }
516 }()
517 if cfg.BuildN {
518
519
520
521
522
523 sh.Printf("\n#\n# %s\n#\n\n", p.ImportPath)
524 }
525
526 if cfg.BuildV {
527 sh.Printf("%s\n", p.ImportPath)
528 }
529
530 if p.Error != nil {
531
532
533 return p.Error
534 }
535
536 if p.BinaryOnly {
537 p.Stale = true
538 p.StaleReason = "binary-only packages are no longer supported"
539 if b.IsCmdList {
540 return nil
541 }
542 return errors.New("binary-only packages are no longer supported")
543 }
544
545 if p.Module != nil && !allowedVersion(p.Module.GoVersion) {
546 return errors.New("module requires Go " + p.Module.GoVersion + " or later")
547 }
548
549 if err := b.checkDirectives(a); err != nil {
550 return err
551 }
552
553 if err := sh.Mkdir(a.Objdir); err != nil {
554 return err
555 }
556 objdir := a.Objdir
557
558
559 if cachedBuild && need&needCgoHdr != 0 {
560 if err := b.loadCachedCgoHdr(a); err == nil {
561 need &^= needCgoHdr
562 }
563 }
564
565
566
567 if cachedBuild && need&needCovMetaFile != 0 {
568 bact := a.Actor.(*buildActor)
569 if err := b.loadCachedObjdirFile(a, cache.Default(), bact.covMetaFileName); err == nil {
570 need &^= needCovMetaFile
571 }
572 }
573
574
575
576
577
578 if need == needVet {
579 if err := b.loadCachedVet(a); err == nil {
580 need &^= needVet
581 }
582 }
583 if need == 0 {
584 return nil
585 }
586
587 if err := AllowInstall(a); err != nil {
588 return err
589 }
590
591
592 dir, _ := filepath.Split(a.Target)
593 if dir != "" {
594 if err := sh.Mkdir(dir); err != nil {
595 return err
596 }
597 }
598
599 gofiles := str.StringList(p.GoFiles)
600 cgofiles := str.StringList(p.CgoFiles)
601 cfiles := str.StringList(p.CFiles)
602 sfiles := str.StringList(p.SFiles)
603 cxxfiles := str.StringList(p.CXXFiles)
604 var objects, cgoObjects, pcCFLAGS, pcLDFLAGS []string
605
606 if p.UsesCgo() || p.UsesSwig() {
607 if pcCFLAGS, pcLDFLAGS, err = b.getPkgConfigFlags(a); err != nil {
608 return
609 }
610 }
611
612
613
614
615
616 nonGoFileLists := [][]string{p.CFiles, p.SFiles, p.CXXFiles, p.HFiles, p.FFiles}
617 OverlayLoop:
618 for _, fs := range nonGoFileLists {
619 for _, f := range fs {
620 if fsys.Replaced(mkAbs(p.Dir, f)) {
621 a.nonGoOverlay = make(map[string]string)
622 break OverlayLoop
623 }
624 }
625 }
626 if a.nonGoOverlay != nil {
627 for _, fs := range nonGoFileLists {
628 for i := range fs {
629 from := mkAbs(p.Dir, fs[i])
630 dst := objdir + filepath.Base(fs[i])
631 if err := sh.CopyFile(dst, fsys.Actual(from), 0666, false); err != nil {
632 return err
633 }
634 a.nonGoOverlay[from] = dst
635 }
636 }
637 }
638
639
640 if p.Internal.Cover.Mode != "" {
641 outfiles := []string{}
642 infiles := []string{}
643 for i, file := range str.StringList(gofiles, cgofiles) {
644 if base.IsTestFile(file) {
645 continue
646 }
647
648 var sourceFile string
649 var coverFile string
650 var key string
651 if base, found := strings.CutSuffix(file, ".cgo1.go"); found {
652
653 base = filepath.Base(base)
654 sourceFile = file
655 coverFile = objdir + base + ".cgo1.go"
656 key = base + ".go"
657 } else {
658 sourceFile = filepath.Join(p.Dir, file)
659 coverFile = objdir + file
660 key = file
661 }
662 coverFile = strings.TrimSuffix(coverFile, ".go") + ".cover.go"
663 if cfg.Experiment.CoverageRedesign {
664 infiles = append(infiles, sourceFile)
665 outfiles = append(outfiles, coverFile)
666 } else {
667 cover := p.Internal.CoverVars[key]
668 if cover == nil {
669 continue
670 }
671 if err := b.cover(a, coverFile, sourceFile, cover.Var); err != nil {
672 return err
673 }
674 }
675 if i < len(gofiles) {
676 gofiles[i] = coverFile
677 } else {
678 cgofiles[i-len(gofiles)] = coverFile
679 }
680 }
681
682 if cfg.Experiment.CoverageRedesign {
683 if len(infiles) != 0 {
684
685
686
687
688
689
690
691
692
693 sum := sha256.Sum256([]byte(a.Package.ImportPath))
694 coverVar := fmt.Sprintf("goCover_%x_", sum[:6])
695 mode := a.Package.Internal.Cover.Mode
696 if mode == "" {
697 panic("covermode should be set at this point")
698 }
699 if newoutfiles, err := b.cover2(a, infiles, outfiles, coverVar, mode); err != nil {
700 return err
701 } else {
702 outfiles = newoutfiles
703 gofiles = append([]string{newoutfiles[0]}, gofiles...)
704 }
705 } else {
706
707
708
709
710
711 p.Internal.Cover.Mode = ""
712 }
713 if ba, ok := a.Actor.(*buildActor); ok && ba.covMetaFileName != "" {
714 b.cacheObjdirFile(a, cache.Default(), ba.covMetaFileName)
715 }
716 }
717 }
718
719
720
721
722
723
724
725 if p.UsesSwig() {
726 outGo, outC, outCXX, err := b.swig(a, objdir, pcCFLAGS)
727 if err != nil {
728 return err
729 }
730 cgofiles = append(cgofiles, outGo...)
731 cfiles = append(cfiles, outC...)
732 cxxfiles = append(cxxfiles, outCXX...)
733 }
734
735
736 if p.UsesCgo() || p.UsesSwig() {
737
738
739
740
741 var gccfiles []string
742 gccfiles = append(gccfiles, cfiles...)
743 cfiles = nil
744 if p.Standard && p.ImportPath == "runtime/cgo" {
745 filter := func(files, nongcc, gcc []string) ([]string, []string) {
746 for _, f := range files {
747 if strings.HasPrefix(f, "gcc_") {
748 gcc = append(gcc, f)
749 } else {
750 nongcc = append(nongcc, f)
751 }
752 }
753 return nongcc, gcc
754 }
755 sfiles, gccfiles = filter(sfiles, sfiles[:0], gccfiles)
756 } else {
757 for _, sfile := range sfiles {
758 data, err := os.ReadFile(filepath.Join(p.Dir, sfile))
759 if err == nil {
760 if bytes.HasPrefix(data, []byte("TEXT")) || bytes.Contains(data, []byte("\nTEXT")) ||
761 bytes.HasPrefix(data, []byte("DATA")) || bytes.Contains(data, []byte("\nDATA")) ||
762 bytes.HasPrefix(data, []byte("GLOBL")) || bytes.Contains(data, []byte("\nGLOBL")) {
763 return fmt.Errorf("package using cgo has Go assembly file %s", sfile)
764 }
765 }
766 }
767 gccfiles = append(gccfiles, sfiles...)
768 sfiles = nil
769 }
770
771 outGo, outObj, err := b.cgo(a, base.Tool("cgo"), objdir, pcCFLAGS, pcLDFLAGS, mkAbsFiles(p.Dir, cgofiles), gccfiles, cxxfiles, p.MFiles, p.FFiles)
772
773
774 cxxfiles = nil
775
776 if err != nil {
777 return err
778 }
779 if cfg.BuildToolchainName == "gccgo" {
780 cgoObjects = append(cgoObjects, a.Objdir+"_cgo_flags")
781 }
782 cgoObjects = append(cgoObjects, outObj...)
783 gofiles = append(gofiles, outGo...)
784
785 switch cfg.BuildBuildmode {
786 case "c-archive", "c-shared":
787 b.cacheCgoHdr(a)
788 }
789 }
790
791 var srcfiles []string
792 srcfiles = append(srcfiles, gofiles...)
793 srcfiles = append(srcfiles, sfiles...)
794 srcfiles = append(srcfiles, cfiles...)
795 srcfiles = append(srcfiles, cxxfiles...)
796 b.cacheSrcFiles(a, srcfiles)
797
798
799 need &^= needCgoHdr
800
801
802 if len(gofiles) == 0 {
803 return &load.NoGoError{Package: p}
804 }
805
806
807 if need&needVet != 0 {
808 buildVetConfig(a, srcfiles)
809 need &^= needVet
810 }
811 if need&needCompiledGoFiles != 0 {
812 if err := b.loadCachedCompiledGoFiles(a); err != nil {
813 return fmt.Errorf("loading compiled Go files from cache: %w", err)
814 }
815 need &^= needCompiledGoFiles
816 }
817 if need == 0 {
818
819 return nil
820 }
821
822
823 symabis, err := BuildToolchain.symabis(b, a, sfiles)
824 if err != nil {
825 return err
826 }
827
828
829
830
831
832
833
834 var icfg bytes.Buffer
835 fmt.Fprintf(&icfg, "# import config\n")
836 for i, raw := range p.Internal.RawImports {
837 final := p.Imports[i]
838 if final != raw {
839 fmt.Fprintf(&icfg, "importmap %s=%s\n", raw, final)
840 }
841 }
842 for _, a1 := range a.Deps {
843 p1 := a1.Package
844 if p1 == nil || p1.ImportPath == "" || a1.built == "" {
845 continue
846 }
847 fmt.Fprintf(&icfg, "packagefile %s=%s\n", p1.ImportPath, a1.built)
848 }
849
850
851
852 var embedcfg []byte
853 if len(p.Internal.Embed) > 0 {
854 var embed struct {
855 Patterns map[string][]string
856 Files map[string]string
857 }
858 embed.Patterns = p.Internal.Embed
859 embed.Files = make(map[string]string)
860 for _, file := range p.EmbedFiles {
861 embed.Files[file] = fsys.Actual(filepath.Join(p.Dir, file))
862 }
863 js, err := json.MarshalIndent(&embed, "", "\t")
864 if err != nil {
865 return fmt.Errorf("marshal embedcfg: %v", err)
866 }
867 embedcfg = js
868 }
869
870
871 var pgoProfile string
872 for _, a1 := range a.Deps {
873 if a1.Mode != "preprocess PGO profile" {
874 continue
875 }
876 if pgoProfile != "" {
877 return fmt.Errorf("action contains multiple PGO profile dependencies")
878 }
879 pgoProfile = a1.built
880 }
881
882 if p.Internal.BuildInfo != nil && cfg.ModulesEnabled {
883 prog := modload.ModInfoProg(p.Internal.BuildInfo.String(), cfg.BuildToolchainName == "gccgo")
884 if len(prog) > 0 {
885 if err := sh.writeFile(objdir+"_gomod_.go", prog); err != nil {
886 return err
887 }
888 gofiles = append(gofiles, objdir+"_gomod_.go")
889 }
890 }
891
892
893 objpkg := objdir + "_pkg_.a"
894 ofile, out, err := BuildToolchain.gc(b, a, objpkg, icfg.Bytes(), embedcfg, symabis, len(sfiles) > 0, pgoProfile, gofiles)
895 if err := sh.reportCmd("", "", out, err); err != nil {
896 return err
897 }
898 if ofile != objpkg {
899 objects = append(objects, ofile)
900 }
901
902
903
904
905 _goos_goarch := "_" + cfg.Goos + "_" + cfg.Goarch
906 _goos := "_" + cfg.Goos
907 _goarch := "_" + cfg.Goarch
908 for _, file := range p.HFiles {
909 name, ext := fileExtSplit(file)
910 switch {
911 case strings.HasSuffix(name, _goos_goarch):
912 targ := file[:len(name)-len(_goos_goarch)] + "_GOOS_GOARCH." + ext
913 if err := sh.CopyFile(objdir+targ, filepath.Join(p.Dir, file), 0666, true); err != nil {
914 return err
915 }
916 case strings.HasSuffix(name, _goarch):
917 targ := file[:len(name)-len(_goarch)] + "_GOARCH." + ext
918 if err := sh.CopyFile(objdir+targ, filepath.Join(p.Dir, file), 0666, true); err != nil {
919 return err
920 }
921 case strings.HasSuffix(name, _goos):
922 targ := file[:len(name)-len(_goos)] + "_GOOS." + ext
923 if err := sh.CopyFile(objdir+targ, filepath.Join(p.Dir, file), 0666, true); err != nil {
924 return err
925 }
926 }
927 }
928
929 for _, file := range cfiles {
930 out := file[:len(file)-len(".c")] + ".o"
931 if err := BuildToolchain.cc(b, a, objdir+out, file); err != nil {
932 return err
933 }
934 objects = append(objects, out)
935 }
936
937
938 if len(sfiles) > 0 {
939 ofiles, err := BuildToolchain.asm(b, a, sfiles)
940 if err != nil {
941 return err
942 }
943 objects = append(objects, ofiles...)
944 }
945
946
947
948
949 if a.buildID != "" && cfg.BuildToolchainName == "gccgo" {
950 switch cfg.Goos {
951 case "aix", "android", "dragonfly", "freebsd", "illumos", "linux", "netbsd", "openbsd", "solaris":
952 asmfile, err := b.gccgoBuildIDFile(a)
953 if err != nil {
954 return err
955 }
956 ofiles, err := BuildToolchain.asm(b, a, []string{asmfile})
957 if err != nil {
958 return err
959 }
960 objects = append(objects, ofiles...)
961 }
962 }
963
964
965
966
967
968 objects = append(objects, cgoObjects...)
969
970
971 for _, syso := range p.SysoFiles {
972 objects = append(objects, filepath.Join(p.Dir, syso))
973 }
974
975
976
977
978
979
980 if len(objects) > 0 {
981 if err := BuildToolchain.pack(b, a, objpkg, objects); err != nil {
982 return err
983 }
984 }
985
986 if err := b.updateBuildID(a, objpkg); err != nil {
987 return err
988 }
989
990 a.built = objpkg
991 return nil
992 }
993
994 func (b *Builder) checkDirectives(a *Action) error {
995 var msg []byte
996 p := a.Package
997 var seen map[string]token.Position
998 for _, d := range p.Internal.Build.Directives {
999 if strings.HasPrefix(d.Text, "//go:debug") {
1000 key, _, err := load.ParseGoDebug(d.Text)
1001 if err != nil && err != load.ErrNotGoDebug {
1002 msg = fmt.Appendf(msg, "%s: invalid //go:debug: %v\n", d.Pos, err)
1003 continue
1004 }
1005 if pos, ok := seen[key]; ok {
1006 msg = fmt.Appendf(msg, "%s: repeated //go:debug for %v\n\t%s: previous //go:debug\n", d.Pos, key, pos)
1007 continue
1008 }
1009 if seen == nil {
1010 seen = make(map[string]token.Position)
1011 }
1012 seen[key] = d.Pos
1013 }
1014 }
1015 if len(msg) > 0 {
1016
1017
1018
1019 err := errors.New("invalid directive")
1020 return b.Shell(a).reportCmd("", "", msg, err)
1021 }
1022 return nil
1023 }
1024
1025 func (b *Builder) cacheObjdirFile(a *Action, c cache.Cache, name string) error {
1026 f, err := os.Open(a.Objdir + name)
1027 if err != nil {
1028 return err
1029 }
1030 defer f.Close()
1031 _, _, err = c.Put(cache.Subkey(a.actionID, name), f)
1032 return err
1033 }
1034
1035 func (b *Builder) findCachedObjdirFile(a *Action, c cache.Cache, name string) (string, error) {
1036 file, _, err := cache.GetFile(c, cache.Subkey(a.actionID, name))
1037 if err != nil {
1038 return "", fmt.Errorf("loading cached file %s: %w", name, err)
1039 }
1040 return file, nil
1041 }
1042
1043 func (b *Builder) loadCachedObjdirFile(a *Action, c cache.Cache, name string) error {
1044 cached, err := b.findCachedObjdirFile(a, c, name)
1045 if err != nil {
1046 return err
1047 }
1048 return b.Shell(a).CopyFile(a.Objdir+name, cached, 0666, true)
1049 }
1050
1051 func (b *Builder) cacheCgoHdr(a *Action) {
1052 c := cache.Default()
1053 b.cacheObjdirFile(a, c, "_cgo_install.h")
1054 }
1055
1056 func (b *Builder) loadCachedCgoHdr(a *Action) error {
1057 c := cache.Default()
1058 return b.loadCachedObjdirFile(a, c, "_cgo_install.h")
1059 }
1060
1061 func (b *Builder) cacheSrcFiles(a *Action, srcfiles []string) {
1062 c := cache.Default()
1063 var buf bytes.Buffer
1064 for _, file := range srcfiles {
1065 if !strings.HasPrefix(file, a.Objdir) {
1066
1067 buf.WriteString("./")
1068 buf.WriteString(file)
1069 buf.WriteString("\n")
1070 continue
1071 }
1072 name := file[len(a.Objdir):]
1073 buf.WriteString(name)
1074 buf.WriteString("\n")
1075 if err := b.cacheObjdirFile(a, c, name); err != nil {
1076 return
1077 }
1078 }
1079 cache.PutBytes(c, cache.Subkey(a.actionID, "srcfiles"), buf.Bytes())
1080 }
1081
1082 func (b *Builder) loadCachedVet(a *Action) error {
1083 c := cache.Default()
1084 list, _, err := cache.GetBytes(c, cache.Subkey(a.actionID, "srcfiles"))
1085 if err != nil {
1086 return fmt.Errorf("reading srcfiles list: %w", err)
1087 }
1088 var srcfiles []string
1089 for _, name := range strings.Split(string(list), "\n") {
1090 if name == "" {
1091 continue
1092 }
1093 if strings.HasPrefix(name, "./") {
1094 srcfiles = append(srcfiles, name[2:])
1095 continue
1096 }
1097 if err := b.loadCachedObjdirFile(a, c, name); err != nil {
1098 return err
1099 }
1100 srcfiles = append(srcfiles, a.Objdir+name)
1101 }
1102 buildVetConfig(a, srcfiles)
1103 return nil
1104 }
1105
1106 func (b *Builder) loadCachedCompiledGoFiles(a *Action) error {
1107 c := cache.Default()
1108 list, _, err := cache.GetBytes(c, cache.Subkey(a.actionID, "srcfiles"))
1109 if err != nil {
1110 return fmt.Errorf("reading srcfiles list: %w", err)
1111 }
1112 var gofiles []string
1113 for _, name := range strings.Split(string(list), "\n") {
1114 if name == "" {
1115 continue
1116 } else if !strings.HasSuffix(name, ".go") {
1117 continue
1118 }
1119 if strings.HasPrefix(name, "./") {
1120 gofiles = append(gofiles, name[len("./"):])
1121 continue
1122 }
1123 file, err := b.findCachedObjdirFile(a, c, name)
1124 if err != nil {
1125 return fmt.Errorf("finding %s: %w", name, err)
1126 }
1127 gofiles = append(gofiles, file)
1128 }
1129 a.Package.CompiledGoFiles = gofiles
1130 return nil
1131 }
1132
1133
1134 type vetConfig struct {
1135 ID string
1136 Compiler string
1137 Dir string
1138 ImportPath string
1139 GoFiles []string
1140 NonGoFiles []string
1141 IgnoredFiles []string
1142
1143 ModulePath string
1144 ModuleVersion string
1145 ImportMap map[string]string
1146 PackageFile map[string]string
1147 Standard map[string]bool
1148 PackageVetx map[string]string
1149 VetxOnly bool
1150 VetxOutput string
1151 GoVersion string
1152
1153 SucceedOnTypecheckFailure bool
1154 }
1155
1156 func buildVetConfig(a *Action, srcfiles []string) {
1157
1158
1159 var gofiles, nongofiles []string
1160 for _, name := range srcfiles {
1161 if strings.HasSuffix(name, ".go") {
1162 gofiles = append(gofiles, name)
1163 } else {
1164 nongofiles = append(nongofiles, name)
1165 }
1166 }
1167
1168 ignored := str.StringList(a.Package.IgnoredGoFiles, a.Package.IgnoredOtherFiles)
1169
1170
1171
1172
1173
1174 vcfg := &vetConfig{
1175 ID: a.Package.ImportPath,
1176 Compiler: cfg.BuildToolchainName,
1177 Dir: a.Package.Dir,
1178 GoFiles: actualFiles(mkAbsFiles(a.Package.Dir, gofiles)),
1179 NonGoFiles: actualFiles(mkAbsFiles(a.Package.Dir, nongofiles)),
1180 IgnoredFiles: actualFiles(mkAbsFiles(a.Package.Dir, ignored)),
1181 ImportPath: a.Package.ImportPath,
1182 ImportMap: make(map[string]string),
1183 PackageFile: make(map[string]string),
1184 Standard: make(map[string]bool),
1185 }
1186 vcfg.GoVersion = "go" + gover.Local()
1187 if a.Package.Module != nil {
1188 v := a.Package.Module.GoVersion
1189 if v == "" {
1190 v = gover.DefaultGoModVersion
1191 }
1192 vcfg.GoVersion = "go" + v
1193
1194 if a.Package.Module.Error == nil {
1195 vcfg.ModulePath = a.Package.Module.Path
1196 vcfg.ModuleVersion = a.Package.Module.Version
1197 }
1198 }
1199 a.vetCfg = vcfg
1200 for i, raw := range a.Package.Internal.RawImports {
1201 final := a.Package.Imports[i]
1202 vcfg.ImportMap[raw] = final
1203 }
1204
1205
1206
1207 vcfgMapped := make(map[string]bool)
1208 for _, p := range vcfg.ImportMap {
1209 vcfgMapped[p] = true
1210 }
1211
1212 for _, a1 := range a.Deps {
1213 p1 := a1.Package
1214 if p1 == nil || p1.ImportPath == "" {
1215 continue
1216 }
1217
1218
1219 if !vcfgMapped[p1.ImportPath] {
1220 vcfg.ImportMap[p1.ImportPath] = p1.ImportPath
1221 }
1222 if a1.built != "" {
1223 vcfg.PackageFile[p1.ImportPath] = a1.built
1224 }
1225 if p1.Standard {
1226 vcfg.Standard[p1.ImportPath] = true
1227 }
1228 }
1229 }
1230
1231
1232
1233 var VetTool string
1234
1235
1236
1237 var VetFlags []string
1238
1239
1240 var VetExplicit bool
1241
1242 func (b *Builder) vet(ctx context.Context, a *Action) error {
1243
1244
1245
1246 a.Failed = nil
1247
1248 if a.Deps[0].Failed != nil {
1249
1250
1251
1252 return nil
1253 }
1254
1255 vcfg := a.Deps[0].vetCfg
1256 if vcfg == nil {
1257
1258 return fmt.Errorf("vet config not found")
1259 }
1260
1261 sh := b.Shell(a)
1262
1263 vcfg.VetxOnly = a.VetxOnly
1264 vcfg.VetxOutput = a.Objdir + "vet.out"
1265 vcfg.PackageVetx = make(map[string]string)
1266
1267 h := cache.NewHash("vet " + a.Package.ImportPath)
1268 fmt.Fprintf(h, "vet %q\n", b.toolID("vet"))
1269
1270 vetFlags := VetFlags
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288 if a.Package.Goroot && !VetExplicit && VetTool == "" {
1289
1290
1291
1292
1293
1294
1295
1296
1297 vetFlags = []string{"-unsafeptr=false"}
1298
1299
1300
1301
1302
1303
1304
1305
1306 if cfg.CmdName == "test" {
1307 vetFlags = append(vetFlags, "-unreachable=false")
1308 }
1309 }
1310
1311
1312
1313
1314
1315
1316 fmt.Fprintf(h, "vetflags %q\n", vetFlags)
1317
1318 fmt.Fprintf(h, "pkg %q\n", a.Deps[0].actionID)
1319 for _, a1 := range a.Deps {
1320 if a1.Mode == "vet" && a1.built != "" {
1321 fmt.Fprintf(h, "vetout %q %s\n", a1.Package.ImportPath, b.fileHash(a1.built))
1322 vcfg.PackageVetx[a1.Package.ImportPath] = a1.built
1323 }
1324 }
1325 key := cache.ActionID(h.Sum())
1326
1327 if vcfg.VetxOnly && !cfg.BuildA {
1328 c := cache.Default()
1329 if file, _, err := cache.GetFile(c, key); err == nil {
1330 a.built = file
1331 return nil
1332 }
1333 }
1334
1335 js, err := json.MarshalIndent(vcfg, "", "\t")
1336 if err != nil {
1337 return fmt.Errorf("internal error marshaling vet config: %v", err)
1338 }
1339 js = append(js, '\n')
1340 if err := sh.writeFile(a.Objdir+"vet.cfg", js); err != nil {
1341 return err
1342 }
1343
1344
1345 env := b.cCompilerEnv()
1346 if cfg.BuildToolchainName == "gccgo" {
1347 env = append(env, "GCCGO="+BuildToolchain.compiler())
1348 }
1349
1350 p := a.Package
1351 tool := VetTool
1352 if tool == "" {
1353 tool = base.Tool("vet")
1354 }
1355 runErr := sh.run(p.Dir, p.ImportPath, env, cfg.BuildToolexec, tool, vetFlags, a.Objdir+"vet.cfg")
1356
1357
1358 if f, err := os.Open(vcfg.VetxOutput); err == nil {
1359 a.built = vcfg.VetxOutput
1360 cache.Default().Put(key, f)
1361 f.Close()
1362 }
1363
1364 return runErr
1365 }
1366
1367
1368 func (b *Builder) linkActionID(a *Action) cache.ActionID {
1369 p := a.Package
1370 h := cache.NewHash("link " + p.ImportPath)
1371
1372
1373 fmt.Fprintf(h, "link\n")
1374 fmt.Fprintf(h, "buildmode %s goos %s goarch %s\n", cfg.BuildBuildmode, cfg.Goos, cfg.Goarch)
1375 fmt.Fprintf(h, "import %q\n", p.ImportPath)
1376 fmt.Fprintf(h, "omitdebug %v standard %v local %v prefix %q\n", p.Internal.OmitDebug, p.Standard, p.Internal.Local, p.Internal.LocalPrefix)
1377 fmt.Fprintf(h, "defaultgodebug %q\n", p.DefaultGODEBUG)
1378 if cfg.BuildTrimpath {
1379 fmt.Fprintln(h, "trimpath")
1380 }
1381
1382
1383 b.printLinkerConfig(h, p)
1384
1385
1386 for _, a1 := range a.Deps {
1387 p1 := a1.Package
1388 if p1 != nil {
1389 if a1.built != "" || a1.buildID != "" {
1390 buildID := a1.buildID
1391 if buildID == "" {
1392 buildID = b.buildID(a1.built)
1393 }
1394 fmt.Fprintf(h, "packagefile %s=%s\n", p1.ImportPath, contentID(buildID))
1395 }
1396
1397
1398 if p1.Name == "main" {
1399 fmt.Fprintf(h, "packagemain %s\n", a1.buildID)
1400 }
1401 if p1.Shlib != "" {
1402 fmt.Fprintf(h, "packageshlib %s=%s\n", p1.ImportPath, contentID(b.buildID(p1.Shlib)))
1403 }
1404 }
1405 }
1406
1407 return h.Sum()
1408 }
1409
1410
1411
1412 func (b *Builder) printLinkerConfig(h io.Writer, p *load.Package) {
1413 switch cfg.BuildToolchainName {
1414 default:
1415 base.Fatalf("linkActionID: unknown toolchain %q", cfg.BuildToolchainName)
1416
1417 case "gc":
1418 fmt.Fprintf(h, "link %s %q %s\n", b.toolID("link"), forcedLdflags, ldBuildmode)
1419 if p != nil {
1420 fmt.Fprintf(h, "linkflags %q\n", p.Internal.Ldflags)
1421 }
1422
1423
1424 key, val, _ := cfg.GetArchEnv()
1425 fmt.Fprintf(h, "%s=%s\n", key, val)
1426
1427 if cfg.CleanGOEXPERIMENT != "" {
1428 fmt.Fprintf(h, "GOEXPERIMENT=%q\n", cfg.CleanGOEXPERIMENT)
1429 }
1430
1431
1432
1433 gorootFinal := cfg.GOROOT
1434 if cfg.BuildTrimpath {
1435 gorootFinal = ""
1436 }
1437 fmt.Fprintf(h, "GOROOT=%s\n", gorootFinal)
1438
1439
1440 fmt.Fprintf(h, "GO_EXTLINK_ENABLED=%s\n", cfg.Getenv("GO_EXTLINK_ENABLED"))
1441
1442
1443
1444
1445 case "gccgo":
1446 id, _, err := b.gccToolID(BuildToolchain.linker(), "go")
1447 if err != nil {
1448 base.Fatalf("%v", err)
1449 }
1450 fmt.Fprintf(h, "link %s %s\n", id, ldBuildmode)
1451
1452 }
1453 }
1454
1455
1456
1457 func (b *Builder) link(ctx context.Context, a *Action) (err error) {
1458 if b.useCache(a, b.linkActionID(a), a.Package.Target, !b.IsCmdList) || b.IsCmdList {
1459 return nil
1460 }
1461 defer b.flushOutput(a)
1462
1463 sh := b.Shell(a)
1464 if err := sh.Mkdir(a.Objdir); err != nil {
1465 return err
1466 }
1467
1468 importcfg := a.Objdir + "importcfg.link"
1469 if err := b.writeLinkImportcfg(a, importcfg); err != nil {
1470 return err
1471 }
1472
1473 if err := AllowInstall(a); err != nil {
1474 return err
1475 }
1476
1477
1478 dir, _ := filepath.Split(a.Target)
1479 if dir != "" {
1480 if err := sh.Mkdir(dir); err != nil {
1481 return err
1482 }
1483 }
1484
1485 if err := BuildToolchain.ld(b, a, a.Target, importcfg, a.Deps[0].built); err != nil {
1486 return err
1487 }
1488
1489
1490 if err := b.updateBuildID(a, a.Target); err != nil {
1491 return err
1492 }
1493
1494 a.built = a.Target
1495 return nil
1496 }
1497
1498 func (b *Builder) writeLinkImportcfg(a *Action, file string) error {
1499
1500 var icfg bytes.Buffer
1501 for _, a1 := range a.Deps {
1502 p1 := a1.Package
1503 if p1 == nil {
1504 continue
1505 }
1506 fmt.Fprintf(&icfg, "packagefile %s=%s\n", p1.ImportPath, a1.built)
1507 if p1.Shlib != "" {
1508 fmt.Fprintf(&icfg, "packageshlib %s=%s\n", p1.ImportPath, p1.Shlib)
1509 }
1510 }
1511 info := ""
1512 if a.Package.Internal.BuildInfo != nil {
1513 info = a.Package.Internal.BuildInfo.String()
1514 }
1515 fmt.Fprintf(&icfg, "modinfo %q\n", modload.ModInfoData(info))
1516 return b.Shell(a).writeFile(file, icfg.Bytes())
1517 }
1518
1519
1520
1521 func (b *Builder) PkgconfigCmd() string {
1522 return envList("PKG_CONFIG", cfg.DefaultPkgConfig)[0]
1523 }
1524
1525
1526
1527
1528
1529
1530
1531 func splitPkgConfigOutput(out []byte) ([]string, error) {
1532 if len(out) == 0 {
1533 return nil, nil
1534 }
1535 var flags []string
1536 flag := make([]byte, 0, len(out))
1537 didQuote := false
1538 escaped := false
1539 quote := byte(0)
1540
1541 for _, c := range out {
1542 if escaped {
1543 if quote == '"' {
1544
1545
1546
1547 switch c {
1548 case '$', '`', '"', '\\', '\n':
1549
1550 default:
1551
1552 flag = append(flag, '\\', c)
1553 escaped = false
1554 continue
1555 }
1556 }
1557
1558 if c == '\n' {
1559
1560
1561 } else {
1562 flag = append(flag, c)
1563 }
1564 escaped = false
1565 continue
1566 }
1567
1568 if quote != 0 && c == quote {
1569 quote = 0
1570 continue
1571 }
1572 switch quote {
1573 case '\'':
1574
1575 flag = append(flag, c)
1576 continue
1577 case '"':
1578
1579
1580 switch c {
1581 case '`', '$', '\\':
1582 default:
1583 flag = append(flag, c)
1584 continue
1585 }
1586 }
1587
1588
1589
1590 switch c {
1591 case '|', '&', ';', '<', '>', '(', ')', '$', '`':
1592 return nil, fmt.Errorf("unexpected shell character %q in pkgconf output", c)
1593
1594 case '\\':
1595
1596
1597 escaped = true
1598 continue
1599
1600 case '"', '\'':
1601 quote = c
1602 didQuote = true
1603 continue
1604
1605 case ' ', '\t', '\n':
1606 if len(flag) > 0 || didQuote {
1607 flags = append(flags, string(flag))
1608 }
1609 flag, didQuote = flag[:0], false
1610 continue
1611 }
1612
1613 flag = append(flag, c)
1614 }
1615
1616
1617
1618
1619 if quote != 0 {
1620 return nil, errors.New("unterminated quoted string in pkgconf output")
1621 }
1622 if escaped {
1623 return nil, errors.New("broken character escaping in pkgconf output")
1624 }
1625
1626 if len(flag) > 0 || didQuote {
1627 flags = append(flags, string(flag))
1628 }
1629 return flags, nil
1630 }
1631
1632
1633 func (b *Builder) getPkgConfigFlags(a *Action) (cflags, ldflags []string, err error) {
1634 p := a.Package
1635 sh := b.Shell(a)
1636 if pcargs := p.CgoPkgConfig; len(pcargs) > 0 {
1637
1638
1639 var pcflags []string
1640 var pkgs []string
1641 for _, pcarg := range pcargs {
1642 if pcarg == "--" {
1643
1644 } else if strings.HasPrefix(pcarg, "--") {
1645 pcflags = append(pcflags, pcarg)
1646 } else {
1647 pkgs = append(pkgs, pcarg)
1648 }
1649 }
1650 for _, pkg := range pkgs {
1651 if !load.SafeArg(pkg) {
1652 return nil, nil, fmt.Errorf("invalid pkg-config package name: %s", pkg)
1653 }
1654 }
1655 var out []byte
1656 out, err = sh.runOut(p.Dir, nil, b.PkgconfigCmd(), "--cflags", pcflags, "--", pkgs)
1657 if err != nil {
1658 desc := b.PkgconfigCmd() + " --cflags " + strings.Join(pcflags, " ") + " -- " + strings.Join(pkgs, " ")
1659 return nil, nil, sh.reportCmd(desc, "", out, err)
1660 }
1661 if len(out) > 0 {
1662 cflags, err = splitPkgConfigOutput(bytes.TrimSpace(out))
1663 if err != nil {
1664 return nil, nil, err
1665 }
1666 if err := checkCompilerFlags("CFLAGS", "pkg-config --cflags", cflags); err != nil {
1667 return nil, nil, err
1668 }
1669 }
1670 out, err = sh.runOut(p.Dir, nil, b.PkgconfigCmd(), "--libs", pcflags, "--", pkgs)
1671 if err != nil {
1672 desc := b.PkgconfigCmd() + " --libs " + strings.Join(pcflags, " ") + " -- " + strings.Join(pkgs, " ")
1673 return nil, nil, sh.reportCmd(desc, "", out, err)
1674 }
1675 if len(out) > 0 {
1676
1677
1678 ldflags, err = splitPkgConfigOutput(bytes.TrimSpace(out))
1679 if err != nil {
1680 return nil, nil, err
1681 }
1682 if err := checkLinkerFlags("LDFLAGS", "pkg-config --libs", ldflags); err != nil {
1683 return nil, nil, err
1684 }
1685 }
1686 }
1687
1688 return
1689 }
1690
1691 func (b *Builder) installShlibname(ctx context.Context, a *Action) error {
1692 if err := AllowInstall(a); err != nil {
1693 return err
1694 }
1695
1696 sh := b.Shell(a)
1697 a1 := a.Deps[0]
1698 if !cfg.BuildN {
1699 if err := sh.Mkdir(filepath.Dir(a.Target)); err != nil {
1700 return err
1701 }
1702 }
1703 return sh.writeFile(a.Target, []byte(filepath.Base(a1.Target)+"\n"))
1704 }
1705
1706 func (b *Builder) linkSharedActionID(a *Action) cache.ActionID {
1707 h := cache.NewHash("linkShared")
1708
1709
1710 fmt.Fprintf(h, "linkShared\n")
1711 fmt.Fprintf(h, "goos %s goarch %s\n", cfg.Goos, cfg.Goarch)
1712
1713
1714 b.printLinkerConfig(h, nil)
1715
1716
1717 for _, a1 := range a.Deps {
1718 p1 := a1.Package
1719 if a1.built == "" {
1720 continue
1721 }
1722 if p1 != nil {
1723 fmt.Fprintf(h, "packagefile %s=%s\n", p1.ImportPath, contentID(b.buildID(a1.built)))
1724 if p1.Shlib != "" {
1725 fmt.Fprintf(h, "packageshlib %s=%s\n", p1.ImportPath, contentID(b.buildID(p1.Shlib)))
1726 }
1727 }
1728 }
1729
1730 for _, a1 := range a.Deps[0].Deps {
1731 p1 := a1.Package
1732 fmt.Fprintf(h, "top %s=%s\n", p1.ImportPath, contentID(b.buildID(a1.built)))
1733 }
1734
1735 return h.Sum()
1736 }
1737
1738 func (b *Builder) linkShared(ctx context.Context, a *Action) (err error) {
1739 if b.useCache(a, b.linkSharedActionID(a), a.Target, !b.IsCmdList) || b.IsCmdList {
1740 return nil
1741 }
1742 defer b.flushOutput(a)
1743
1744 if err := AllowInstall(a); err != nil {
1745 return err
1746 }
1747
1748 if err := b.Shell(a).Mkdir(a.Objdir); err != nil {
1749 return err
1750 }
1751
1752 importcfg := a.Objdir + "importcfg.link"
1753 if err := b.writeLinkImportcfg(a, importcfg); err != nil {
1754 return err
1755 }
1756
1757
1758
1759 a.built = a.Target
1760 return BuildToolchain.ldShared(b, a, a.Deps[0].Deps, a.Target, importcfg, a.Deps)
1761 }
1762
1763
1764 func BuildInstallFunc(b *Builder, ctx context.Context, a *Action) (err error) {
1765 defer func() {
1766 if err != nil {
1767
1768
1769
1770 sep, path := "", ""
1771 if a.Package != nil {
1772 sep, path = " ", a.Package.ImportPath
1773 }
1774 err = fmt.Errorf("go %s%s%s: %v", cfg.CmdName, sep, path, err)
1775 }
1776 }()
1777 sh := b.Shell(a)
1778
1779 a1 := a.Deps[0]
1780 a.buildID = a1.buildID
1781 if a.json != nil {
1782 a.json.BuildID = a.buildID
1783 }
1784
1785
1786
1787
1788
1789
1790 if a1.built == a.Target {
1791 a.built = a.Target
1792 if !a.buggyInstall {
1793 b.cleanup(a1)
1794 }
1795
1796
1797
1798
1799
1800
1801
1802
1803
1804
1805
1806
1807
1808
1809
1810
1811
1812
1813 if !a.buggyInstall && !b.IsCmdList {
1814 if cfg.BuildN {
1815 sh.ShowCmd("", "touch %s", a.Target)
1816 } else if err := AllowInstall(a); err == nil {
1817 now := time.Now()
1818 os.Chtimes(a.Target, now, now)
1819 }
1820 }
1821 return nil
1822 }
1823
1824
1825
1826 if b.IsCmdList {
1827 a.built = a1.built
1828 return nil
1829 }
1830 if err := AllowInstall(a); err != nil {
1831 return err
1832 }
1833
1834 if err := sh.Mkdir(a.Objdir); err != nil {
1835 return err
1836 }
1837
1838 perm := fs.FileMode(0666)
1839 if a1.Mode == "link" {
1840 switch cfg.BuildBuildmode {
1841 case "c-archive", "c-shared", "plugin":
1842 default:
1843 perm = 0777
1844 }
1845 }
1846
1847
1848 dir, _ := filepath.Split(a.Target)
1849 if dir != "" {
1850 if err := sh.Mkdir(dir); err != nil {
1851 return err
1852 }
1853 }
1854
1855 if !a.buggyInstall {
1856 defer b.cleanup(a1)
1857 }
1858
1859 return sh.moveOrCopyFile(a.Target, a1.built, perm, false)
1860 }
1861
1862
1863
1864
1865
1866
1867 var AllowInstall = func(*Action) error { return nil }
1868
1869
1870
1871
1872
1873 func (b *Builder) cleanup(a *Action) {
1874 if !cfg.BuildWork {
1875 b.Shell(a).RemoveAll(a.Objdir)
1876 }
1877 }
1878
1879
1880 func (b *Builder) installHeader(ctx context.Context, a *Action) error {
1881 sh := b.Shell(a)
1882
1883 src := a.Objdir + "_cgo_install.h"
1884 if _, err := os.Stat(src); os.IsNotExist(err) {
1885
1886
1887
1888
1889
1890 if cfg.BuildX {
1891 sh.ShowCmd("", "# %s not created", src)
1892 }
1893 return nil
1894 }
1895
1896 if err := AllowInstall(a); err != nil {
1897 return err
1898 }
1899
1900 dir, _ := filepath.Split(a.Target)
1901 if dir != "" {
1902 if err := sh.Mkdir(dir); err != nil {
1903 return err
1904 }
1905 }
1906
1907 return sh.moveOrCopyFile(a.Target, src, 0666, true)
1908 }
1909
1910
1911
1912
1913 func (b *Builder) cover(a *Action, dst, src string, varName string) error {
1914 return b.Shell(a).run(a.Objdir, "", nil,
1915 cfg.BuildToolexec,
1916 base.Tool("cover"),
1917 "-mode", a.Package.Internal.Cover.Mode,
1918 "-var", varName,
1919 "-o", dst,
1920 src)
1921 }
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931 func (b *Builder) cover2(a *Action, infiles, outfiles []string, varName string, mode string) ([]string, error) {
1932 pkgcfg := a.Objdir + "pkgcfg.txt"
1933 covoutputs := a.Objdir + "coveroutfiles.txt"
1934 odir := filepath.Dir(outfiles[0])
1935 cv := filepath.Join(odir, "covervars.go")
1936 outfiles = append([]string{cv}, outfiles...)
1937 if err := b.writeCoverPkgInputs(a, pkgcfg, covoutputs, outfiles); err != nil {
1938 return nil, err
1939 }
1940 args := []string{base.Tool("cover"),
1941 "-pkgcfg", pkgcfg,
1942 "-mode", mode,
1943 "-var", varName,
1944 "-outfilelist", covoutputs,
1945 }
1946 args = append(args, infiles...)
1947 if err := b.Shell(a).run(a.Objdir, "", nil,
1948 cfg.BuildToolexec, args); err != nil {
1949 return nil, err
1950 }
1951 return outfiles, nil
1952 }
1953
1954 func (b *Builder) writeCoverPkgInputs(a *Action, pconfigfile string, covoutputsfile string, outfiles []string) error {
1955 sh := b.Shell(a)
1956 p := a.Package
1957 p.Internal.Cover.Cfg = a.Objdir + "coveragecfg"
1958 pcfg := covcmd.CoverPkgConfig{
1959 PkgPath: p.ImportPath,
1960 PkgName: p.Name,
1961
1962
1963
1964
1965 Granularity: "perblock",
1966 OutConfig: p.Internal.Cover.Cfg,
1967 Local: p.Internal.Local,
1968 }
1969 if ba, ok := a.Actor.(*buildActor); ok && ba.covMetaFileName != "" {
1970 pcfg.EmitMetaFile = a.Objdir + ba.covMetaFileName
1971 }
1972 if a.Package.Module != nil {
1973 pcfg.ModulePath = a.Package.Module.Path
1974 }
1975 data, err := json.Marshal(pcfg)
1976 if err != nil {
1977 return err
1978 }
1979 data = append(data, '\n')
1980 if err := sh.writeFile(pconfigfile, data); err != nil {
1981 return err
1982 }
1983 var sb strings.Builder
1984 for i := range outfiles {
1985 fmt.Fprintf(&sb, "%s\n", outfiles[i])
1986 }
1987 return sh.writeFile(covoutputsfile, []byte(sb.String()))
1988 }
1989
1990 var objectMagic = [][]byte{
1991 {'!', '<', 'a', 'r', 'c', 'h', '>', '\n'},
1992 {'<', 'b', 'i', 'g', 'a', 'f', '>', '\n'},
1993 {'\x7F', 'E', 'L', 'F'},
1994 {0xFE, 0xED, 0xFA, 0xCE},
1995 {0xFE, 0xED, 0xFA, 0xCF},
1996 {0xCE, 0xFA, 0xED, 0xFE},
1997 {0xCF, 0xFA, 0xED, 0xFE},
1998 {0x4d, 0x5a, 0x90, 0x00, 0x03, 0x00},
1999 {0x4d, 0x5a, 0x78, 0x00, 0x01, 0x00},
2000 {0x00, 0x00, 0x01, 0xEB},
2001 {0x00, 0x00, 0x8a, 0x97},
2002 {0x00, 0x00, 0x06, 0x47},
2003 {0x00, 0x61, 0x73, 0x6D},
2004 {0x01, 0xDF},
2005 {0x01, 0xF7},
2006 }
2007
2008 func isObject(s string) bool {
2009 f, err := os.Open(s)
2010 if err != nil {
2011 return false
2012 }
2013 defer f.Close()
2014 buf := make([]byte, 64)
2015 io.ReadFull(f, buf)
2016 for _, magic := range objectMagic {
2017 if bytes.HasPrefix(buf, magic) {
2018 return true
2019 }
2020 }
2021 return false
2022 }
2023
2024
2025
2026
2027 func (b *Builder) cCompilerEnv() []string {
2028 return []string{"TERM=dumb"}
2029 }
2030
2031
2032
2033
2034
2035
2036 func mkAbs(dir, f string) string {
2037
2038
2039
2040
2041 if filepath.IsAbs(f) || strings.HasPrefix(f, "$WORK") {
2042 return f
2043 }
2044 return filepath.Join(dir, f)
2045 }
2046
2047 type toolchain interface {
2048
2049
2050 gc(b *Builder, a *Action, archive string, importcfg, embedcfg []byte, symabis string, asmhdr bool, pgoProfile string, gofiles []string) (ofile string, out []byte, err error)
2051
2052
2053 cc(b *Builder, a *Action, ofile, cfile string) error
2054
2055
2056 asm(b *Builder, a *Action, sfiles []string) ([]string, error)
2057
2058
2059 symabis(b *Builder, a *Action, sfiles []string) (string, error)
2060
2061
2062
2063 pack(b *Builder, a *Action, afile string, ofiles []string) error
2064
2065 ld(b *Builder, root *Action, targetPath, importcfg, mainpkg string) error
2066
2067 ldShared(b *Builder, root *Action, toplevelactions []*Action, targetPath, importcfg string, allactions []*Action) error
2068
2069 compiler() string
2070 linker() string
2071 }
2072
2073 type noToolchain struct{}
2074
2075 func noCompiler() error {
2076 log.Fatalf("unknown compiler %q", cfg.BuildContext.Compiler)
2077 return nil
2078 }
2079
2080 func (noToolchain) compiler() string {
2081 noCompiler()
2082 return ""
2083 }
2084
2085 func (noToolchain) linker() string {
2086 noCompiler()
2087 return ""
2088 }
2089
2090 func (noToolchain) gc(b *Builder, a *Action, archive string, importcfg, embedcfg []byte, symabis string, asmhdr bool, pgoProfile string, gofiles []string) (ofile string, out []byte, err error) {
2091 return "", nil, noCompiler()
2092 }
2093
2094 func (noToolchain) asm(b *Builder, a *Action, sfiles []string) ([]string, error) {
2095 return nil, noCompiler()
2096 }
2097
2098 func (noToolchain) symabis(b *Builder, a *Action, sfiles []string) (string, error) {
2099 return "", noCompiler()
2100 }
2101
2102 func (noToolchain) pack(b *Builder, a *Action, afile string, ofiles []string) error {
2103 return noCompiler()
2104 }
2105
2106 func (noToolchain) ld(b *Builder, root *Action, targetPath, importcfg, mainpkg string) error {
2107 return noCompiler()
2108 }
2109
2110 func (noToolchain) ldShared(b *Builder, root *Action, toplevelactions []*Action, targetPath, importcfg string, allactions []*Action) error {
2111 return noCompiler()
2112 }
2113
2114 func (noToolchain) cc(b *Builder, a *Action, ofile, cfile string) error {
2115 return noCompiler()
2116 }
2117
2118
2119 func (b *Builder) gcc(a *Action, workdir, out string, flags []string, cfile string) error {
2120 p := a.Package
2121 return b.ccompile(a, out, flags, cfile, b.GccCmd(p.Dir, workdir))
2122 }
2123
2124
2125 func (b *Builder) gxx(a *Action, workdir, out string, flags []string, cxxfile string) error {
2126 p := a.Package
2127 return b.ccompile(a, out, flags, cxxfile, b.GxxCmd(p.Dir, workdir))
2128 }
2129
2130
2131 func (b *Builder) gfortran(a *Action, workdir, out string, flags []string, ffile string) error {
2132 p := a.Package
2133 return b.ccompile(a, out, flags, ffile, b.gfortranCmd(p.Dir, workdir))
2134 }
2135
2136
2137 func (b *Builder) ccompile(a *Action, outfile string, flags []string, file string, compiler []string) error {
2138 p := a.Package
2139 sh := b.Shell(a)
2140 file = mkAbs(p.Dir, file)
2141 outfile = mkAbs(p.Dir, outfile)
2142
2143
2144
2145
2146
2147
2148 if b.gccSupportsFlag(compiler, "-fdebug-prefix-map=a=b") {
2149 if cfg.BuildTrimpath || p.Goroot {
2150 prefixMapFlag := "-fdebug-prefix-map"
2151 if b.gccSupportsFlag(compiler, "-ffile-prefix-map=a=b") {
2152 prefixMapFlag = "-ffile-prefix-map"
2153 }
2154
2155
2156
2157 var from, toPath string
2158 if m := p.Module; m == nil {
2159 if p.Root == "" {
2160 from = p.Dir
2161 toPath = p.ImportPath
2162 } else if p.Goroot {
2163 from = p.Root
2164 toPath = "GOROOT"
2165 } else {
2166 from = p.Root
2167 toPath = "GOPATH"
2168 }
2169 } else if m.Dir == "" {
2170
2171
2172 from = modload.VendorDir()
2173 toPath = "vendor"
2174 } else {
2175 from = m.Dir
2176 toPath = m.Path
2177 if m.Version != "" {
2178 toPath += "@" + m.Version
2179 }
2180 }
2181
2182
2183
2184 var to string
2185 if cfg.BuildContext.GOOS == "windows" {
2186 to = filepath.Join(`\\_\_`, toPath)
2187 } else {
2188 to = filepath.Join("/_", toPath)
2189 }
2190 flags = append(slices.Clip(flags), prefixMapFlag+"="+from+"="+to)
2191 }
2192 }
2193
2194
2195
2196 if b.gccSupportsFlag(compiler, "-frandom-seed=1") {
2197 flags = append(flags, "-frandom-seed="+buildid.HashToString(a.actionID))
2198 }
2199
2200 overlayPath := file
2201 if p, ok := a.nonGoOverlay[overlayPath]; ok {
2202 overlayPath = p
2203 }
2204 output, err := sh.runOut(filepath.Dir(overlayPath), b.cCompilerEnv(), compiler, flags, "-o", outfile, "-c", filepath.Base(overlayPath))
2205
2206
2207
2208
2209
2210
2211
2212
2213
2214 if bytes.Contains(output, []byte("DWARF2 only supports one section per compilation unit")) {
2215 newFlags := make([]string, 0, len(flags))
2216 for _, f := range flags {
2217 if !strings.HasPrefix(f, "-g") {
2218 newFlags = append(newFlags, f)
2219 }
2220 }
2221 if len(newFlags) < len(flags) {
2222 return b.ccompile(a, outfile, newFlags, file, compiler)
2223 }
2224 }
2225
2226 if len(output) > 0 && err == nil && os.Getenv("GO_BUILDER_NAME") != "" {
2227 output = append(output, "C compiler warning promoted to error on Go builders\n"...)
2228 err = errors.New("warning promoted to error")
2229 }
2230
2231 return sh.reportCmd("", "", output, err)
2232 }
2233
2234
2235 func (b *Builder) gccld(a *Action, objdir, outfile string, flags []string, objs []string) error {
2236 p := a.Package
2237 sh := b.Shell(a)
2238 var cmd []string
2239 if len(p.CXXFiles) > 0 || len(p.SwigCXXFiles) > 0 {
2240 cmd = b.GxxCmd(p.Dir, objdir)
2241 } else {
2242 cmd = b.GccCmd(p.Dir, objdir)
2243 }
2244
2245 cmdargs := []any{cmd, "-o", outfile, objs, flags}
2246 _, err := sh.runOut(base.Cwd(), b.cCompilerEnv(), cmdargs...)
2247
2248
2249
2250 if cfg.BuildN || cfg.BuildX {
2251 saw := "succeeded"
2252 if err != nil {
2253 saw = "failed"
2254 }
2255 sh.ShowCmd("", "%s # test for internal linking errors (%s)", joinUnambiguously(str.StringList(cmdargs...)), saw)
2256 }
2257
2258 return err
2259 }
2260
2261
2262
2263 func (b *Builder) GccCmd(incdir, workdir string) []string {
2264 return b.compilerCmd(b.ccExe(), incdir, workdir)
2265 }
2266
2267
2268
2269 func (b *Builder) GxxCmd(incdir, workdir string) []string {
2270 return b.compilerCmd(b.cxxExe(), incdir, workdir)
2271 }
2272
2273
2274 func (b *Builder) gfortranCmd(incdir, workdir string) []string {
2275 return b.compilerCmd(b.fcExe(), incdir, workdir)
2276 }
2277
2278
2279 func (b *Builder) ccExe() []string {
2280 return envList("CC", cfg.DefaultCC(cfg.Goos, cfg.Goarch))
2281 }
2282
2283
2284 func (b *Builder) cxxExe() []string {
2285 return envList("CXX", cfg.DefaultCXX(cfg.Goos, cfg.Goarch))
2286 }
2287
2288
2289 func (b *Builder) fcExe() []string {
2290 return envList("FC", "gfortran")
2291 }
2292
2293
2294
2295 func (b *Builder) compilerCmd(compiler []string, incdir, workdir string) []string {
2296 a := append(compiler, "-I", incdir)
2297
2298
2299
2300 if cfg.Goos != "windows" {
2301 a = append(a, "-fPIC")
2302 }
2303 a = append(a, b.gccArchArgs()...)
2304
2305
2306 if cfg.BuildContext.CgoEnabled {
2307 switch cfg.Goos {
2308 case "windows":
2309 a = append(a, "-mthreads")
2310 default:
2311 a = append(a, "-pthread")
2312 }
2313 }
2314
2315 if cfg.Goos == "aix" {
2316
2317 a = append(a, "-mcmodel=large")
2318 }
2319
2320
2321 if b.gccSupportsFlag(compiler, "-fno-caret-diagnostics") {
2322 a = append(a, "-fno-caret-diagnostics")
2323 }
2324
2325 if b.gccSupportsFlag(compiler, "-Qunused-arguments") {
2326 a = append(a, "-Qunused-arguments")
2327 }
2328
2329
2330
2331
2332 if b.gccSupportsFlag(compiler, "-Wl,--no-gc-sections") {
2333 a = append(a, "-Wl,--no-gc-sections")
2334 }
2335
2336
2337 a = append(a, "-fmessage-length=0")
2338
2339
2340 if b.gccSupportsFlag(compiler, "-fdebug-prefix-map=a=b") {
2341 if workdir == "" {
2342 workdir = b.WorkDir
2343 }
2344 workdir = strings.TrimSuffix(workdir, string(filepath.Separator))
2345 if b.gccSupportsFlag(compiler, "-ffile-prefix-map=a=b") {
2346 a = append(a, "-ffile-prefix-map="+workdir+"=/tmp/go-build")
2347 } else {
2348 a = append(a, "-fdebug-prefix-map="+workdir+"=/tmp/go-build")
2349 }
2350 }
2351
2352
2353
2354 if b.gccSupportsFlag(compiler, "-gno-record-gcc-switches") {
2355 a = append(a, "-gno-record-gcc-switches")
2356 }
2357
2358
2359
2360
2361 if cfg.Goos == "darwin" || cfg.Goos == "ios" {
2362 a = append(a, "-fno-common")
2363 }
2364
2365 return a
2366 }
2367
2368
2369
2370
2371
2372 func (b *Builder) gccNoPie(linker []string) string {
2373 if b.gccSupportsFlag(linker, "-no-pie") {
2374 return "-no-pie"
2375 }
2376 if b.gccSupportsFlag(linker, "-nopie") {
2377 return "-nopie"
2378 }
2379 return ""
2380 }
2381
2382
2383 func (b *Builder) gccSupportsFlag(compiler []string, flag string) bool {
2384
2385
2386
2387 sh := b.BackgroundShell()
2388
2389 key := [2]string{compiler[0], flag}
2390
2391
2392
2393
2394
2395
2396
2397
2398
2399
2400
2401
2402
2403
2404
2405
2406
2407 tmp := os.DevNull
2408 if runtime.GOOS == "windows" || runtime.GOOS == "ios" {
2409 f, err := os.CreateTemp(b.WorkDir, "")
2410 if err != nil {
2411 return false
2412 }
2413 f.Close()
2414 tmp = f.Name()
2415 defer os.Remove(tmp)
2416 }
2417
2418 cmdArgs := str.StringList(compiler, flag)
2419 if strings.HasPrefix(flag, "-Wl,") {
2420 ldflags, err := buildFlags("LDFLAGS", DefaultCFlags, nil, checkLinkerFlags)
2421 if err != nil {
2422 return false
2423 }
2424 cmdArgs = append(cmdArgs, ldflags...)
2425 } else {
2426 cflags, err := buildFlags("CFLAGS", DefaultCFlags, nil, checkCompilerFlags)
2427 if err != nil {
2428 return false
2429 }
2430 cmdArgs = append(cmdArgs, cflags...)
2431 cmdArgs = append(cmdArgs, "-c")
2432 }
2433
2434 cmdArgs = append(cmdArgs, "-x", "c", "-", "-o", tmp)
2435
2436 if cfg.BuildN {
2437 sh.ShowCmd(b.WorkDir, "%s || true", joinUnambiguously(cmdArgs))
2438 return false
2439 }
2440
2441
2442 compilerID, cacheOK := b.gccCompilerID(compiler[0])
2443
2444 b.exec.Lock()
2445 defer b.exec.Unlock()
2446 if b, ok := b.flagCache[key]; ok {
2447 return b
2448 }
2449 if b.flagCache == nil {
2450 b.flagCache = make(map[[2]string]bool)
2451 }
2452
2453
2454 var flagID cache.ActionID
2455 if cacheOK {
2456 flagID = cache.Subkey(compilerID, "gccSupportsFlag "+flag)
2457 if data, _, err := cache.GetBytes(cache.Default(), flagID); err == nil {
2458 supported := string(data) == "true"
2459 b.flagCache[key] = supported
2460 return supported
2461 }
2462 }
2463
2464 if cfg.BuildX {
2465 sh.ShowCmd(b.WorkDir, "%s || true", joinUnambiguously(cmdArgs))
2466 }
2467 cmd := exec.Command(cmdArgs[0], cmdArgs[1:]...)
2468 cmd.Dir = b.WorkDir
2469 cmd.Env = append(cmd.Environ(), "LC_ALL=C")
2470 out, _ := cmd.CombinedOutput()
2471
2472
2473
2474
2475
2476
2477 supported := !bytes.Contains(out, []byte("unrecognized")) &&
2478 !bytes.Contains(out, []byte("unknown")) &&
2479 !bytes.Contains(out, []byte("unrecognised")) &&
2480 !bytes.Contains(out, []byte("is not supported")) &&
2481 !bytes.Contains(out, []byte("not recognized")) &&
2482 !bytes.Contains(out, []byte("unsupported"))
2483
2484 if cacheOK {
2485 s := "false"
2486 if supported {
2487 s = "true"
2488 }
2489 cache.PutBytes(cache.Default(), flagID, []byte(s))
2490 }
2491
2492 b.flagCache[key] = supported
2493 return supported
2494 }
2495
2496
2497 func statString(info os.FileInfo) string {
2498 return fmt.Sprintf("stat %d %x %v %v\n", info.Size(), uint64(info.Mode()), info.ModTime(), info.IsDir())
2499 }
2500
2501
2502
2503
2504
2505
2506 func (b *Builder) gccCompilerID(compiler string) (id cache.ActionID, ok bool) {
2507
2508
2509
2510 sh := b.BackgroundShell()
2511
2512 if cfg.BuildN {
2513 sh.ShowCmd(b.WorkDir, "%s || true", joinUnambiguously([]string{compiler, "--version"}))
2514 return cache.ActionID{}, false
2515 }
2516
2517 b.exec.Lock()
2518 defer b.exec.Unlock()
2519
2520 if id, ok := b.gccCompilerIDCache[compiler]; ok {
2521 return id, ok
2522 }
2523
2524
2525
2526
2527
2528
2529
2530
2531
2532
2533
2534
2535
2536
2537
2538 exe, err := pathcache.LookPath(compiler)
2539 if err != nil {
2540 return cache.ActionID{}, false
2541 }
2542
2543 h := cache.NewHash("gccCompilerID")
2544 fmt.Fprintf(h, "gccCompilerID %q", exe)
2545 key := h.Sum()
2546 data, _, err := cache.GetBytes(cache.Default(), key)
2547 if err == nil && len(data) > len(id) {
2548 stats := strings.Split(string(data[:len(data)-len(id)]), "\x00")
2549 if len(stats)%2 != 0 {
2550 goto Miss
2551 }
2552 for i := 0; i+2 <= len(stats); i++ {
2553 info, err := os.Stat(stats[i])
2554 if err != nil || statString(info) != stats[i+1] {
2555 goto Miss
2556 }
2557 }
2558 copy(id[:], data[len(data)-len(id):])
2559 return id, true
2560 Miss:
2561 }
2562
2563
2564
2565
2566
2567
2568 toolID, exe2, err := b.gccToolID(compiler, "c")
2569 if err != nil {
2570 return cache.ActionID{}, false
2571 }
2572
2573 exes := []string{exe, exe2}
2574 str.Uniq(&exes)
2575 fmt.Fprintf(h, "gccCompilerID %q %q\n", exes, toolID)
2576 id = h.Sum()
2577
2578 var buf bytes.Buffer
2579 for _, exe := range exes {
2580 if exe == "" {
2581 continue
2582 }
2583 info, err := os.Stat(exe)
2584 if err != nil {
2585 return cache.ActionID{}, false
2586 }
2587 buf.WriteString(exe)
2588 buf.WriteString("\x00")
2589 buf.WriteString(statString(info))
2590 buf.WriteString("\x00")
2591 }
2592 buf.Write(id[:])
2593
2594 cache.PutBytes(cache.Default(), key, buf.Bytes())
2595 if b.gccCompilerIDCache == nil {
2596 b.gccCompilerIDCache = make(map[string]cache.ActionID)
2597 }
2598 b.gccCompilerIDCache[compiler] = id
2599 return id, true
2600 }
2601
2602
2603 func (b *Builder) gccArchArgs() []string {
2604 switch cfg.Goarch {
2605 case "386":
2606 return []string{"-m32"}
2607 case "amd64":
2608 if cfg.Goos == "darwin" {
2609 return []string{"-arch", "x86_64", "-m64"}
2610 }
2611 return []string{"-m64"}
2612 case "arm64":
2613 if cfg.Goos == "darwin" {
2614 return []string{"-arch", "arm64"}
2615 }
2616 case "arm":
2617 return []string{"-marm"}
2618 case "s390x":
2619 return []string{"-m64", "-march=z196"}
2620 case "mips64", "mips64le":
2621 args := []string{"-mabi=64"}
2622 if cfg.GOMIPS64 == "hardfloat" {
2623 return append(args, "-mhard-float")
2624 } else if cfg.GOMIPS64 == "softfloat" {
2625 return append(args, "-msoft-float")
2626 }
2627 case "mips", "mipsle":
2628 args := []string{"-mabi=32", "-march=mips32"}
2629 if cfg.GOMIPS == "hardfloat" {
2630 return append(args, "-mhard-float", "-mfp32", "-mno-odd-spreg")
2631 } else if cfg.GOMIPS == "softfloat" {
2632 return append(args, "-msoft-float")
2633 }
2634 case "loong64":
2635 return []string{"-mabi=lp64d"}
2636 case "ppc64":
2637 if cfg.Goos == "aix" {
2638 return []string{"-maix64"}
2639 }
2640 }
2641 return nil
2642 }
2643
2644
2645
2646
2647
2648
2649
2650 func envList(key, def string) []string {
2651 v := cfg.Getenv(key)
2652 if v == "" {
2653 v = def
2654 }
2655 args, err := quoted.Split(v)
2656 if err != nil {
2657 panic(fmt.Sprintf("could not parse environment variable %s with value %q: %v", key, v, err))
2658 }
2659 return args
2660 }
2661
2662
2663 func (b *Builder) CFlags(p *load.Package) (cppflags, cflags, cxxflags, fflags, ldflags []string, err error) {
2664 if cppflags, err = buildFlags("CPPFLAGS", "", p.CgoCPPFLAGS, checkCompilerFlags); err != nil {
2665 return
2666 }
2667 if cflags, err = buildFlags("CFLAGS", DefaultCFlags, p.CgoCFLAGS, checkCompilerFlags); err != nil {
2668 return
2669 }
2670 if cxxflags, err = buildFlags("CXXFLAGS", DefaultCFlags, p.CgoCXXFLAGS, checkCompilerFlags); err != nil {
2671 return
2672 }
2673 if fflags, err = buildFlags("FFLAGS", DefaultCFlags, p.CgoFFLAGS, checkCompilerFlags); err != nil {
2674 return
2675 }
2676 if ldflags, err = buildFlags("LDFLAGS", DefaultCFlags, p.CgoLDFLAGS, checkLinkerFlags); err != nil {
2677 return
2678 }
2679
2680 return
2681 }
2682
2683 func buildFlags(name, defaults string, fromPackage []string, check func(string, string, []string) error) ([]string, error) {
2684 if err := check(name, "#cgo "+name, fromPackage); err != nil {
2685 return nil, err
2686 }
2687 return str.StringList(envList("CGO_"+name, defaults), fromPackage), nil
2688 }
2689
2690 var cgoRe = lazyregexp.New(`[/\\:]`)
2691
2692 func (b *Builder) cgo(a *Action, cgoExe, objdir string, pcCFLAGS, pcLDFLAGS, cgofiles, gccfiles, gxxfiles, mfiles, ffiles []string) (outGo, outObj []string, err error) {
2693 p := a.Package
2694 sh := b.Shell(a)
2695
2696 cgoCPPFLAGS, cgoCFLAGS, cgoCXXFLAGS, cgoFFLAGS, cgoLDFLAGS, err := b.CFlags(p)
2697 if err != nil {
2698 return nil, nil, err
2699 }
2700
2701 cgoCPPFLAGS = append(cgoCPPFLAGS, pcCFLAGS...)
2702 cgoLDFLAGS = append(cgoLDFLAGS, pcLDFLAGS...)
2703
2704 if len(mfiles) > 0 {
2705 cgoLDFLAGS = append(cgoLDFLAGS, "-lobjc")
2706 }
2707
2708
2709
2710
2711 if len(ffiles) > 0 {
2712 fc := cfg.Getenv("FC")
2713 if fc == "" {
2714 fc = "gfortran"
2715 }
2716 if strings.Contains(fc, "gfortran") {
2717 cgoLDFLAGS = append(cgoLDFLAGS, "-lgfortran")
2718 }
2719 }
2720
2721
2722
2723
2724
2725
2726
2727
2728
2729
2730
2731
2732
2733
2734
2735
2736 flagSources := []string{"CGO_CFLAGS", "CGO_CXXFLAGS", "CGO_FFLAGS"}
2737 flagLists := [][]string{cgoCFLAGS, cgoCXXFLAGS, cgoFFLAGS}
2738 if flagsNotCompatibleWithInternalLinking(flagSources, flagLists) {
2739 tokenFile := objdir + "preferlinkext"
2740 if err := sh.writeFile(tokenFile, nil); err != nil {
2741 return nil, nil, err
2742 }
2743 outObj = append(outObj, tokenFile)
2744 }
2745
2746 if cfg.BuildMSan {
2747 cgoCFLAGS = append([]string{"-fsanitize=memory"}, cgoCFLAGS...)
2748 cgoLDFLAGS = append([]string{"-fsanitize=memory"}, cgoLDFLAGS...)
2749 }
2750 if cfg.BuildASan {
2751 cgoCFLAGS = append([]string{"-fsanitize=address"}, cgoCFLAGS...)
2752 cgoLDFLAGS = append([]string{"-fsanitize=address"}, cgoLDFLAGS...)
2753 }
2754
2755
2756
2757 cgoCPPFLAGS = append(cgoCPPFLAGS, "-I", objdir)
2758
2759
2760
2761 gofiles := []string{objdir + "_cgo_gotypes.go"}
2762 cfiles := []string{"_cgo_export.c"}
2763 for _, fn := range cgofiles {
2764 f := strings.TrimSuffix(filepath.Base(fn), ".go")
2765 gofiles = append(gofiles, objdir+f+".cgo1.go")
2766 cfiles = append(cfiles, f+".cgo2.c")
2767 }
2768
2769
2770
2771 cgoflags := []string{}
2772 if p.Standard && p.ImportPath == "runtime/cgo" {
2773 cgoflags = append(cgoflags, "-import_runtime_cgo=false")
2774 }
2775 if p.Standard && (p.ImportPath == "runtime/race" || p.ImportPath == "runtime/msan" || p.ImportPath == "runtime/cgo" || p.ImportPath == "runtime/asan") {
2776 cgoflags = append(cgoflags, "-import_syscall=false")
2777 }
2778
2779
2780
2781
2782
2783
2784
2785
2786
2787
2788
2789 cgoenv := b.cCompilerEnv()
2790 var ldflagsOption []string
2791 if len(cgoLDFLAGS) > 0 {
2792 flags := make([]string, len(cgoLDFLAGS))
2793 for i, f := range cgoLDFLAGS {
2794 flags[i] = strconv.Quote(f)
2795 }
2796 ldflagsOption = []string{"-ldflags=" + strings.Join(flags, " ")}
2797
2798
2799 cgoenv = append(cgoenv, "CGO_LDFLAGS=")
2800 }
2801
2802 if cfg.BuildToolchainName == "gccgo" {
2803 if b.gccSupportsFlag([]string{BuildToolchain.compiler()}, "-fsplit-stack") {
2804 cgoCFLAGS = append(cgoCFLAGS, "-fsplit-stack")
2805 }
2806 cgoflags = append(cgoflags, "-gccgo")
2807 if pkgpath := gccgoPkgpath(p); pkgpath != "" {
2808 cgoflags = append(cgoflags, "-gccgopkgpath="+pkgpath)
2809 }
2810 if !BuildToolchain.(gccgoToolchain).supportsCgoIncomplete(b, a) {
2811 cgoflags = append(cgoflags, "-gccgo_define_cgoincomplete")
2812 }
2813 }
2814
2815 switch cfg.BuildBuildmode {
2816 case "c-archive", "c-shared":
2817
2818
2819
2820 cgoflags = append(cgoflags, "-exportheader="+objdir+"_cgo_install.h")
2821 }
2822
2823
2824
2825 var trimpath []string
2826 for i := range cgofiles {
2827 path := mkAbs(p.Dir, cgofiles[i])
2828 if fsys.Replaced(path) {
2829 actual := fsys.Actual(path)
2830 cgofiles[i] = actual
2831 trimpath = append(trimpath, actual+"=>"+path)
2832 }
2833 }
2834 if len(trimpath) > 0 {
2835 cgoflags = append(cgoflags, "-trimpath", strings.Join(trimpath, ";"))
2836 }
2837
2838 if err := sh.run(p.Dir, p.ImportPath, cgoenv, cfg.BuildToolexec, cgoExe, "-objdir", objdir, "-importpath", p.ImportPath, cgoflags, ldflagsOption, "--", cgoCPPFLAGS, cgoCFLAGS, cgofiles); err != nil {
2839 return nil, nil, err
2840 }
2841 outGo = append(outGo, gofiles...)
2842
2843
2844
2845
2846
2847
2848
2849 oseq := 0
2850 nextOfile := func() string {
2851 oseq++
2852 return objdir + fmt.Sprintf("_x%03d.o", oseq)
2853 }
2854
2855
2856 cflags := str.StringList(cgoCPPFLAGS, cgoCFLAGS)
2857 for _, cfile := range cfiles {
2858 ofile := nextOfile()
2859 if err := b.gcc(a, a.Objdir, ofile, cflags, objdir+cfile); err != nil {
2860 return nil, nil, err
2861 }
2862 outObj = append(outObj, ofile)
2863 }
2864
2865 for _, file := range gccfiles {
2866 ofile := nextOfile()
2867 if err := b.gcc(a, a.Objdir, ofile, cflags, file); err != nil {
2868 return nil, nil, err
2869 }
2870 outObj = append(outObj, ofile)
2871 }
2872
2873 cxxflags := str.StringList(cgoCPPFLAGS, cgoCXXFLAGS)
2874 for _, file := range gxxfiles {
2875 ofile := nextOfile()
2876 if err := b.gxx(a, a.Objdir, ofile, cxxflags, file); err != nil {
2877 return nil, nil, err
2878 }
2879 outObj = append(outObj, ofile)
2880 }
2881
2882 for _, file := range mfiles {
2883 ofile := nextOfile()
2884 if err := b.gcc(a, a.Objdir, ofile, cflags, file); err != nil {
2885 return nil, nil, err
2886 }
2887 outObj = append(outObj, ofile)
2888 }
2889
2890 fflags := str.StringList(cgoCPPFLAGS, cgoFFLAGS)
2891 for _, file := range ffiles {
2892 ofile := nextOfile()
2893 if err := b.gfortran(a, a.Objdir, ofile, fflags, file); err != nil {
2894 return nil, nil, err
2895 }
2896 outObj = append(outObj, ofile)
2897 }
2898
2899 switch cfg.BuildToolchainName {
2900 case "gc":
2901 importGo := objdir + "_cgo_import.go"
2902 dynOutGo, dynOutObj, err := b.dynimport(a, objdir, importGo, cgoExe, cflags, cgoLDFLAGS, outObj)
2903 if err != nil {
2904 return nil, nil, err
2905 }
2906 if dynOutGo != "" {
2907 outGo = append(outGo, dynOutGo)
2908 }
2909 if dynOutObj != "" {
2910 outObj = append(outObj, dynOutObj)
2911 }
2912
2913 case "gccgo":
2914 defunC := objdir + "_cgo_defun.c"
2915 defunObj := objdir + "_cgo_defun.o"
2916 if err := BuildToolchain.cc(b, a, defunObj, defunC); err != nil {
2917 return nil, nil, err
2918 }
2919 outObj = append(outObj, defunObj)
2920
2921 default:
2922 noCompiler()
2923 }
2924
2925
2926
2927
2928
2929
2930 if cfg.BuildToolchainName == "gc" && !cfg.BuildN {
2931 var flags []string
2932 for _, f := range outGo {
2933 if !strings.HasPrefix(filepath.Base(f), "_cgo_") {
2934 continue
2935 }
2936
2937 src, err := os.ReadFile(f)
2938 if err != nil {
2939 return nil, nil, err
2940 }
2941
2942 const cgoLdflag = "//go:cgo_ldflag"
2943 idx := bytes.Index(src, []byte(cgoLdflag))
2944 for idx >= 0 {
2945
2946
2947 start := bytes.LastIndex(src[:idx], []byte("\n"))
2948 if start == -1 {
2949 start = 0
2950 }
2951
2952
2953 end := bytes.Index(src[idx:], []byte("\n"))
2954 if end == -1 {
2955 end = len(src)
2956 } else {
2957 end += idx
2958 }
2959
2960
2961
2962
2963
2964 commentStart := bytes.Index(src[start:], []byte("//"))
2965 commentStart += start
2966
2967
2968 if bytes.HasPrefix(src[commentStart:], []byte(cgoLdflag)) {
2969
2970
2971 flag := string(src[idx+len(cgoLdflag) : end])
2972 flag = strings.TrimSpace(flag)
2973 flag = strings.Trim(flag, `"`)
2974 flags = append(flags, flag)
2975 }
2976 src = src[end:]
2977 idx = bytes.Index(src, []byte(cgoLdflag))
2978 }
2979 }
2980
2981
2982 if len(cgoLDFLAGS) > 0 {
2983 outer:
2984 for i := range flags {
2985 for j, f := range cgoLDFLAGS {
2986 if f != flags[i+j] {
2987 continue outer
2988 }
2989 }
2990 flags = append(flags[:i], flags[i+len(cgoLDFLAGS):]...)
2991 break
2992 }
2993 }
2994
2995 if err := checkLinkerFlags("LDFLAGS", "go:cgo_ldflag", flags); err != nil {
2996 return nil, nil, err
2997 }
2998 }
2999
3000 return outGo, outObj, nil
3001 }
3002
3003
3004
3005
3006
3007
3008
3009
3010 func flagsNotCompatibleWithInternalLinking(sourceList []string, flagListList [][]string) bool {
3011 for i := range sourceList {
3012 sn := sourceList[i]
3013 fll := flagListList[i]
3014 if err := checkCompilerFlagsForInternalLink(sn, sn, fll); err != nil {
3015 return true
3016 }
3017 }
3018 return false
3019 }
3020
3021
3022
3023
3024
3025
3026 func (b *Builder) dynimport(a *Action, objdir, importGo, cgoExe string, cflags, cgoLDFLAGS, outObj []string) (dynOutGo, dynOutObj string, err error) {
3027 p := a.Package
3028 sh := b.Shell(a)
3029
3030 cfile := objdir + "_cgo_main.c"
3031 ofile := objdir + "_cgo_main.o"
3032 if err := b.gcc(a, objdir, ofile, cflags, cfile); err != nil {
3033 return "", "", err
3034 }
3035
3036
3037 var syso []string
3038 seen := make(map[*Action]bool)
3039 var gatherSyso func(*Action)
3040 gatherSyso = func(a1 *Action) {
3041 if seen[a1] {
3042 return
3043 }
3044 seen[a1] = true
3045 if p1 := a1.Package; p1 != nil {
3046 syso = append(syso, mkAbsFiles(p1.Dir, p1.SysoFiles)...)
3047 }
3048 for _, a2 := range a1.Deps {
3049 gatherSyso(a2)
3050 }
3051 }
3052 gatherSyso(a)
3053 sort.Strings(syso)
3054 str.Uniq(&syso)
3055 linkobj := str.StringList(ofile, outObj, syso)
3056 dynobj := objdir + "_cgo_.o"
3057
3058 ldflags := cgoLDFLAGS
3059 if (cfg.Goarch == "arm" && cfg.Goos == "linux") || cfg.Goos == "android" {
3060 if !slices.Contains(ldflags, "-no-pie") {
3061
3062
3063 ldflags = append(ldflags, "-pie")
3064 }
3065 if slices.Contains(ldflags, "-pie") && slices.Contains(ldflags, "-static") {
3066
3067
3068 n := make([]string, 0, len(ldflags)-1)
3069 for _, flag := range ldflags {
3070 if flag != "-static" {
3071 n = append(n, flag)
3072 }
3073 }
3074 ldflags = n
3075 }
3076 }
3077 if err := b.gccld(a, objdir, dynobj, ldflags, linkobj); err != nil {
3078
3079
3080
3081
3082
3083
3084 fail := objdir + "dynimportfail"
3085 if err := sh.writeFile(fail, nil); err != nil {
3086 return "", "", err
3087 }
3088 return "", fail, nil
3089 }
3090
3091
3092 var cgoflags []string
3093 if p.Standard && p.ImportPath == "runtime/cgo" {
3094 cgoflags = []string{"-dynlinker"}
3095 }
3096 err = sh.run(base.Cwd(), p.ImportPath, b.cCompilerEnv(), cfg.BuildToolexec, cgoExe, "-dynpackage", p.Name, "-dynimport", dynobj, "-dynout", importGo, cgoflags)
3097 if err != nil {
3098 return "", "", err
3099 }
3100 return importGo, "", nil
3101 }
3102
3103
3104
3105
3106 func (b *Builder) swig(a *Action, objdir string, pcCFLAGS []string) (outGo, outC, outCXX []string, err error) {
3107 p := a.Package
3108
3109 if err := b.swigVersionCheck(); err != nil {
3110 return nil, nil, nil, err
3111 }
3112
3113 intgosize, err := b.swigIntSize(objdir)
3114 if err != nil {
3115 return nil, nil, nil, err
3116 }
3117
3118 for _, f := range p.SwigFiles {
3119 goFile, cFile, err := b.swigOne(a, f, objdir, pcCFLAGS, false, intgosize)
3120 if err != nil {
3121 return nil, nil, nil, err
3122 }
3123 if goFile != "" {
3124 outGo = append(outGo, goFile)
3125 }
3126 if cFile != "" {
3127 outC = append(outC, cFile)
3128 }
3129 }
3130 for _, f := range p.SwigCXXFiles {
3131 goFile, cxxFile, err := b.swigOne(a, f, objdir, pcCFLAGS, true, intgosize)
3132 if err != nil {
3133 return nil, nil, nil, err
3134 }
3135 if goFile != "" {
3136 outGo = append(outGo, goFile)
3137 }
3138 if cxxFile != "" {
3139 outCXX = append(outCXX, cxxFile)
3140 }
3141 }
3142 return outGo, outC, outCXX, nil
3143 }
3144
3145
3146 var (
3147 swigCheckOnce sync.Once
3148 swigCheck error
3149 )
3150
3151 func (b *Builder) swigDoVersionCheck() error {
3152 sh := b.BackgroundShell()
3153 out, err := sh.runOut(".", nil, "swig", "-version")
3154 if err != nil {
3155 return err
3156 }
3157 re := regexp.MustCompile(`[vV]ersion +(\d+)([.]\d+)?([.]\d+)?`)
3158 matches := re.FindSubmatch(out)
3159 if matches == nil {
3160
3161 return nil
3162 }
3163
3164 major, err := strconv.Atoi(string(matches[1]))
3165 if err != nil {
3166
3167 return nil
3168 }
3169 const errmsg = "must have SWIG version >= 3.0.6"
3170 if major < 3 {
3171 return errors.New(errmsg)
3172 }
3173 if major > 3 {
3174
3175 return nil
3176 }
3177
3178
3179 if len(matches[2]) > 0 {
3180 minor, err := strconv.Atoi(string(matches[2][1:]))
3181 if err != nil {
3182 return nil
3183 }
3184 if minor > 0 {
3185
3186 return nil
3187 }
3188 }
3189
3190
3191 if len(matches[3]) > 0 {
3192 patch, err := strconv.Atoi(string(matches[3][1:]))
3193 if err != nil {
3194 return nil
3195 }
3196 if patch < 6 {
3197
3198 return errors.New(errmsg)
3199 }
3200 }
3201
3202 return nil
3203 }
3204
3205 func (b *Builder) swigVersionCheck() error {
3206 swigCheckOnce.Do(func() {
3207 swigCheck = b.swigDoVersionCheck()
3208 })
3209 return swigCheck
3210 }
3211
3212
3213 var (
3214 swigIntSizeOnce sync.Once
3215 swigIntSize string
3216 swigIntSizeError error
3217 )
3218
3219
3220 const swigIntSizeCode = `
3221 package main
3222 const i int = 1 << 32
3223 `
3224
3225
3226
3227 func (b *Builder) swigDoIntSize(objdir string) (intsize string, err error) {
3228 if cfg.BuildN {
3229 return "$INTBITS", nil
3230 }
3231 src := filepath.Join(b.WorkDir, "swig_intsize.go")
3232 if err = os.WriteFile(src, []byte(swigIntSizeCode), 0666); err != nil {
3233 return
3234 }
3235 srcs := []string{src}
3236
3237 p := load.GoFilesPackage(context.TODO(), load.PackageOpts{}, srcs)
3238
3239 if _, _, e := BuildToolchain.gc(b, &Action{Mode: "swigDoIntSize", Package: p, Objdir: objdir}, "", nil, nil, "", false, "", srcs); e != nil {
3240 return "32", nil
3241 }
3242 return "64", nil
3243 }
3244
3245
3246
3247 func (b *Builder) swigIntSize(objdir string) (intsize string, err error) {
3248 swigIntSizeOnce.Do(func() {
3249 swigIntSize, swigIntSizeError = b.swigDoIntSize(objdir)
3250 })
3251 return swigIntSize, swigIntSizeError
3252 }
3253
3254
3255 func (b *Builder) swigOne(a *Action, file, objdir string, pcCFLAGS []string, cxx bool, intgosize string) (outGo, outC string, err error) {
3256 p := a.Package
3257 sh := b.Shell(a)
3258
3259 cgoCPPFLAGS, cgoCFLAGS, cgoCXXFLAGS, _, _, err := b.CFlags(p)
3260 if err != nil {
3261 return "", "", err
3262 }
3263
3264 var cflags []string
3265 if cxx {
3266 cflags = str.StringList(cgoCPPFLAGS, pcCFLAGS, cgoCXXFLAGS)
3267 } else {
3268 cflags = str.StringList(cgoCPPFLAGS, pcCFLAGS, cgoCFLAGS)
3269 }
3270
3271 n := 5
3272 if cxx {
3273 n = 8
3274 }
3275 base := file[:len(file)-n]
3276 goFile := base + ".go"
3277 gccBase := base + "_wrap."
3278 gccExt := "c"
3279 if cxx {
3280 gccExt = "cxx"
3281 }
3282
3283 gccgo := cfg.BuildToolchainName == "gccgo"
3284
3285
3286 args := []string{
3287 "-go",
3288 "-cgo",
3289 "-intgosize", intgosize,
3290 "-module", base,
3291 "-o", objdir + gccBase + gccExt,
3292 "-outdir", objdir,
3293 }
3294
3295 for _, f := range cflags {
3296 if len(f) > 3 && f[:2] == "-I" {
3297 args = append(args, f)
3298 }
3299 }
3300
3301 if gccgo {
3302 args = append(args, "-gccgo")
3303 if pkgpath := gccgoPkgpath(p); pkgpath != "" {
3304 args = append(args, "-go-pkgpath", pkgpath)
3305 }
3306 }
3307 if cxx {
3308 args = append(args, "-c++")
3309 }
3310
3311 out, err := sh.runOut(p.Dir, nil, "swig", args, file)
3312 if err != nil && (bytes.Contains(out, []byte("-intgosize")) || bytes.Contains(out, []byte("-cgo"))) {
3313 return "", "", errors.New("must have SWIG version >= 3.0.6")
3314 }
3315 if err := sh.reportCmd("", "", out, err); err != nil {
3316 return "", "", err
3317 }
3318
3319
3320
3321
3322
3323
3324
3325
3326
3327 goFile = objdir + goFile
3328 newGoFile := objdir + "_" + base + "_swig.go"
3329 if cfg.BuildX || cfg.BuildN {
3330 sh.ShowCmd("", "mv %s %s", goFile, newGoFile)
3331 }
3332 if !cfg.BuildN {
3333 if err := os.Rename(goFile, newGoFile); err != nil {
3334 return "", "", err
3335 }
3336 }
3337 return newGoFile, objdir + gccBase + gccExt, nil
3338 }
3339
3340
3341
3342
3343
3344
3345
3346
3347
3348
3349
3350 func (b *Builder) disableBuildID(ldflags []string) []string {
3351 switch cfg.Goos {
3352 case "android", "dragonfly", "linux", "netbsd":
3353 ldflags = append(ldflags, "-Wl,--build-id=none")
3354 }
3355 return ldflags
3356 }
3357
3358
3359
3360
3361 func mkAbsFiles(dir string, files []string) []string {
3362 abs := make([]string, len(files))
3363 for i, f := range files {
3364 if !filepath.IsAbs(f) {
3365 f = filepath.Join(dir, f)
3366 }
3367 abs[i] = f
3368 }
3369 return abs
3370 }
3371
3372
3373 func actualFiles(files []string) []string {
3374 a := make([]string, len(files))
3375 for i, f := range files {
3376 a[i] = fsys.Actual(f)
3377 }
3378 return a
3379 }
3380
3381
3382
3383
3384
3385
3386
3387
3388 func passLongArgsInResponseFiles(cmd *exec.Cmd) (cleanup func()) {
3389 cleanup = func() {}
3390
3391 var argLen int
3392 for _, arg := range cmd.Args {
3393 argLen += len(arg)
3394 }
3395
3396
3397
3398 if !useResponseFile(cmd.Path, argLen) {
3399 return
3400 }
3401
3402 tf, err := os.CreateTemp("", "args")
3403 if err != nil {
3404 log.Fatalf("error writing long arguments to response file: %v", err)
3405 }
3406 cleanup = func() { os.Remove(tf.Name()) }
3407 var buf bytes.Buffer
3408 for _, arg := range cmd.Args[1:] {
3409 fmt.Fprintf(&buf, "%s\n", encodeArg(arg))
3410 }
3411 if _, err := tf.Write(buf.Bytes()); err != nil {
3412 tf.Close()
3413 cleanup()
3414 log.Fatalf("error writing long arguments to response file: %v", err)
3415 }
3416 if err := tf.Close(); err != nil {
3417 cleanup()
3418 log.Fatalf("error writing long arguments to response file: %v", err)
3419 }
3420 cmd.Args = []string{cmd.Args[0], "@" + tf.Name()}
3421 return cleanup
3422 }
3423
3424 func useResponseFile(path string, argLen int) bool {
3425
3426
3427
3428 prog := strings.TrimSuffix(filepath.Base(path), ".exe")
3429 switch prog {
3430 case "compile", "link", "cgo", "asm", "cover":
3431 default:
3432 return false
3433 }
3434
3435 if argLen > sys.ExecArgLengthLimit {
3436 return true
3437 }
3438
3439
3440
3441 isBuilder := os.Getenv("GO_BUILDER_NAME") != ""
3442 if isBuilder && rand.Intn(10) == 0 {
3443 return true
3444 }
3445
3446 return false
3447 }
3448
3449
3450 func encodeArg(arg string) string {
3451
3452 if !strings.ContainsAny(arg, "\\\n") {
3453 return arg
3454 }
3455 var b strings.Builder
3456 for _, r := range arg {
3457 switch r {
3458 case '\\':
3459 b.WriteByte('\\')
3460 b.WriteByte('\\')
3461 case '\n':
3462 b.WriteByte('\\')
3463 b.WriteByte('n')
3464 default:
3465 b.WriteRune(r)
3466 }
3467 }
3468 return b.String()
3469 }
3470
View as plain text