...

Text file src/cmd/go/testdata/script/cgo_suspect_flag_force_external.txt

Documentation: cmd/go/testdata/script

     1# Test case to verify that when we have a package that uses CGO in
     2# combination with selected "unusual" flags (involving plugins, LTO)
     3# that we force external linking.  See related
     4# issues 58619,  58620, and 58848.
     5
     6[compiler:gccgo] skip # only external linking for gccgo
     7
     8[!cgo] skip 'test verifies behavior that depends on CGO_CFLAGS'
     9[mustlinkext] skip 'test expects internal linking for non-cgo programs'
    10
    11# Here we build three program: one with explicit CGO use, one with no
    12# CGO use, and one that uses a stdlib package ("runtime/cgo") that has
    13# CGO in it. It used to be that only the explicit use of CGO would
    14# trigger external linking, and that the program that only used
    15# "runtime/cgo" would always be handled with internal linking. This caused
    16# issues when users included odd/unusual flags (ex: -fplugin, -flto)
    17# in CGO_CFLAGS, causing the Go linker to have to read and interpret
    18# non-standard host objects.
    19#
    20# As of 1.21 we continue to use internal linking for programs whose
    21# CGO use comes only from stdlib packages in the absence of any flag
    22# funny business, however if the Go command sees flags that may be suspicious,
    23# it signals the Go linker to invoke the external linker.
    24
    25# The next few tests run builds passing "-n" to the Go command, then
    26# checking the output to see if the Go command is trying to pass a
    27# "preferlinkext" token to the linker to request external linking.
    28
    29#-----------------------
    30
    31# Use a fresh GOCACHE for these next steps, so as to have the real
    32# actions for the runtime/cgo package appear in the "-n -x" output.
    33env GOCACHE=$WORK/gocache
    34mkdir $GOCACHE
    35
    36# First build: there is no CGO in use, so no token should be present regardless
    37# of weird CGO flags.
    38go build -x -n -o dummy.exe ./noUseOfCgo
    39! stderr preferlinkext
    40env CGO_CFLAGS=-flto
    41go build -x -n -o dummy.exe ./noUseOfCgo
    42! stderr preferlinkext
    43env CGO_CFLAGS=
    44
    45# Second build uses CGO, so we expect to see the token present in the
    46# -n output only when strange flags are used.
    47go build -x -n -o dummy.exe ./usesInternalCgo
    48! stderr preferlinkext
    49env CGO_CFLAGS=-flto
    50go build -x -n -o dummy.exe ./usesInternalCgo
    51stderr preferlinkext
    52env CGO_CFLAGS=-fplugin
    53go build -x -n -o dummy.exe ./usesInternalCgo
    54stderr preferlinkext
    55env CGO_CFLAGS=-fprofile-instr-generate
    56go build -x -n -o dummy.exe ./usesInternalCgo
    57stderr preferlinkext
    58
    59# The -fdebug-prefix-map=path is permitted for internal linking.
    60env CGO_CFLAGS=-fdebug-prefix-map=/some/sandbox/execroot/workspace=/tmp/new
    61go build -x -n -o dummy.exe ./usesInternalCgo
    62! stderr preferlinkext
    63env CGO_CFLAGS=-fdebug-prefix-map=/Users/someone/.cache/bazel/_bazel_someone/3fa7e4650c43657ead684537951f49e2/sandbox/linux-sandbox/10/execroot/rules_go_static=.
    64go build -x -n -o dummy.exe ./usesInternalCgo
    65! stderr preferlinkext
    66# The -ffile-prefix-map=path is permitted for internal linking too.
    67env CGO_CFLAGS=-ffile-prefix-map=/Users/someone/.cache/bazel/_bazel_someone/3fa7e4650c43657ead684537951f49e2/sandbox/linux-sandbox/10/execroot/rules_go_static/bazel-out/aarch64-fastbuild-ST-b33d65c724e6/bin/external/io_bazel_rules_go/stdlib_=.
    68go build -x -n -o dummy.exe ./usesInternalCgo
    69! stderr preferlinkext
    70# Verifying that -fdebug-prefix-map=path, -ffile-prefix-map, -no-canonical-prefixes
    71# and -fno-canonical-systemd-headers are permitted for internal linking.
    72env CGO_CFLAGS=-fdebug-prefix-map=old=/tmp/new
    73go build -x -n -o dummy.exe ./usesInternalCgo
    74! stderr preferlinkext
    75env CGO_CFLAGS=-ffile-prefix-map=/Users/someone/_11233/things=new
    76go build -x -n -o dummy.exe ./usesInternalCgo
    77! stderr preferlinkext
    78env CGO_CFLAGS=-no-canonical-prefixes
    79go build -x -n -o dummy.exe ./usesInternalCgo
    80! stderr preferlinkext
    81env CGO_CFLAGS=-fno-canonical-system-headers
    82go build -x -n -o dummy.exe ./usesInternalCgo
    83! stderr preferlinkext
    84env CGO_CFLAGS=
    85
    86[short] skip
    87
    88# In the remaining tests below we do actual builds (without -n) to
    89# verify that the Go linker is going the right thing in addition to the
    90# Go command. Here the idea is to pass "-tmpdir" to the linker, then
    91# check after the link is done for the presence of the file
    92# <tmpdir>/go.o, which the Go linker creates prior to kicking off the
    93# external linker.
    94
    95mkdir tmp1
    96mkdir tmp2
    97mkdir tmp3
    98mkdir tmp4
    99mkdir tmp5
   100
   101# First build: no external linking expected
   102go build -ldflags=-tmpdir=tmp1 -o $devnull ./noUseOfCgo &
   103
   104# Second build: using only "runtime/cgo", expect internal linking.
   105go build -ldflags=-tmpdir=tmp2 -o $devnull ./usesInternalCgo &
   106
   107# Third build: program uses only "runtime/cgo", so we would normally
   108# expect internal linking, except that cflags contain suspicious entries
   109# (in this case, a flag that does not appear on the allow list).
   110env CGO_CFLAGS=-fmerge-all-constants
   111env CGO_LDFLAGS=-fmerge-all-constants
   112go build -ldflags=-tmpdir=tmp3 -o $devnull ./usesInternalCgo &
   113env CGO_CFLAGS=
   114env CGO_LDFLAGS=
   115
   116# Fourth build: explicit CGO, expect external linking.
   117go build -ldflags=-tmpdir=tmp4 -o $devnull ./usesExplicitCgo &
   118
   119# Fifth build: explicit CGO, but we specifically asked for internal linking
   120# via a flag, so using internal linking it is.
   121[cgolinkext] go list ./usesInternalCgo
   122[!cgolinkext] go build '-ldflags=-tmpdir=tmp5 -linkmode=internal' -o $devnull ./usesInternalCgo &
   123
   124# Sixth build: explicit CGO use in a non-main package.
   125go build -o p.a ./nonMainPackageUsesExplicitCgo &
   126
   127wait
   128
   129# Check first build: no external linking expected
   130! exists tmp1/go.o
   131
   132# Check second build: using only "runtime/cgo", expect internal linking.
   133[!cgolinkext] ! exists tmp2/go.o
   134[cgolinkext] exists tmp2/go.o
   135
   136# Check third build: has suspicious flag.
   137exists tmp3/go.o
   138
   139# Fourth build: explicit CGO, expect external linking.
   140exists tmp4/go.o
   141
   142# Fifth build: explicit CGO, -linkmode=internal.
   143! exists tmp5/go.o
   144
   145# Sixth build: make sure that "go tool nm" doesn't get confused
   146# by the presence of the "preferlinkext" sentinel.
   147go tool nm p.a
   148
   149-- go.mod --
   150
   151module cgo.example
   152
   153go 1.20
   154
   155-- noUseOfCgo/main.go --
   156
   157package main
   158
   159func main() {
   160	println("clean as a whistle")
   161}
   162
   163-- usesInternalCgo/main.go --
   164
   165package main
   166
   167import (
   168	"runtime/cgo"
   169)
   170
   171func main() {
   172	q := "hello"
   173	h := cgo.NewHandle(q)
   174	h.Delete()
   175}
   176
   177-- usesExplicitCgo/main.go --
   178
   179package main
   180
   181/*
   182int meaningOfLife() { return 42; }
   183*/
   184import "C"
   185
   186func main() {
   187     println(C.meaningOfLife())
   188}
   189
   190-- nonMainPackageUsesExplicitCgo/main.go --
   191
   192package p
   193
   194/*
   195int meaningOfLife() { return 42; }
   196*/
   197import "C"
   198
   199func PrintIt() {
   200     println(C.meaningOfLife())
   201}

View as plain text