1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30 package work
31
32 import (
33 "fmt"
34 "internal/lazyregexp"
35 "regexp"
36 "strings"
37
38 "cmd/go/internal/cfg"
39 "cmd/go/internal/load"
40 )
41
42 var re = lazyregexp.New
43
44 var validCompilerFlags = []*lazyregexp.Regexp{
45 re(`-D([A-Za-z_][A-Za-z0-9_]*)(=[^@\-]*)?`),
46 re(`-U([A-Za-z_][A-Za-z0-9_]*)`),
47 re(`-F([^@\-].*)`),
48 re(`-I([^@\-].*)`),
49 re(`-O`),
50 re(`-O([^@\-].*)`),
51 re(`-W`),
52 re(`-W([^@,]+)`),
53 re(`-Wa,-mbig-obj`),
54 re(`-Wp,-D([A-Za-z_][A-Za-z0-9_]*)(=[^@,\-]*)?`),
55 re(`-Wp,-U([A-Za-z_][A-Za-z0-9_]*)`),
56 re(`-ansi`),
57 re(`-f(no-)?asynchronous-unwind-tables`),
58 re(`-f(no-)?blocks`),
59 re(`-f(no-)builtin-[a-zA-Z0-9_]*`),
60 re(`-f(no-)?common`),
61 re(`-f(no-)?constant-cfstrings`),
62 re(`-fdebug-prefix-map=([^@]+)=([^@]+)`),
63 re(`-fdiagnostics-show-note-include-stack`),
64 re(`-ffile-prefix-map=([^@]+)=([^@]+)`),
65 re(`-fno-canonical-system-headers`),
66 re(`-f(no-)?eliminate-unused-debug-types`),
67 re(`-f(no-)?exceptions`),
68 re(`-f(no-)?fast-math`),
69 re(`-f(no-)?inline-functions`),
70 re(`-finput-charset=([^@\-].*)`),
71 re(`-f(no-)?fat-lto-objects`),
72 re(`-f(no-)?keep-inline-dllexport`),
73 re(`-f(no-)?lto`),
74 re(`-fmacro-backtrace-limit=(.+)`),
75 re(`-fmessage-length=(.+)`),
76 re(`-f(no-)?modules`),
77 re(`-f(no-)?objc-arc`),
78 re(`-f(no-)?objc-nonfragile-abi`),
79 re(`-f(no-)?objc-legacy-dispatch`),
80 re(`-f(no-)?omit-frame-pointer`),
81 re(`-f(no-)?openmp(-simd)?`),
82 re(`-f(no-)?permissive`),
83 re(`-f(no-)?(pic|PIC|pie|PIE)`),
84 re(`-f(no-)?plt`),
85 re(`-f(no-)?rtti`),
86 re(`-f(no-)?split-stack`),
87 re(`-f(no-)?stack-(.+)`),
88 re(`-f(no-)?strict-aliasing`),
89 re(`-f(un)signed-char`),
90 re(`-f(no-)?use-linker-plugin`),
91 re(`-f(no-)?visibility-inlines-hidden`),
92 re(`-fsanitize=(.+)`),
93 re(`-fsanitize-undefined-strip-path-components=(-)?[0-9]+`),
94 re(`-ftemplate-depth-(.+)`),
95 re(`-ftls-model=(global-dynamic|local-dynamic|initial-exec|local-exec)`),
96 re(`-fvisibility=(.+)`),
97 re(`-g([^@\-].*)?`),
98 re(`-m32`),
99 re(`-m64`),
100 re(`-m(abi|arch|cpu|fpu|simd|tls-dialect|tune)=([^@\-].*)`),
101 re(`-m(no-)?v?aes`),
102 re(`-marm`),
103 re(`-mcmodel=[0-9a-z-]+`),
104 re(`-mfloat-abi=([^@\-].*)`),
105 re(`-m(soft|single|double)-float`),
106 re(`-mfpmath=[0-9a-z,+]*`),
107 re(`-m(no-)?avx[0-9a-z.]*`),
108 re(`-m(no-)?ms-bitfields`),
109 re(`-m(no-)?stack-(.+)`),
110 re(`-mmacosx-(.+)`),
111 re(`-m(no-)?relax`),
112 re(`-m(no-)?strict-align`),
113 re(`-m(no-)?(lsx|lasx|frecipe|div32|lam-bh|lamcas|ld-seq-sa)`),
114 re(`-mios-simulator-version-min=(.+)`),
115 re(`-miphoneos-version-min=(.+)`),
116 re(`-mlarge-data-threshold=[0-9]+`),
117 re(`-mtvos-simulator-version-min=(.+)`),
118 re(`-mtvos-version-min=(.+)`),
119 re(`-mwatchos-simulator-version-min=(.+)`),
120 re(`-mwatchos-version-min=(.+)`),
121 re(`-mnop-fun-dllimport`),
122 re(`-m(no-)?sse[0-9.]*`),
123 re(`-m(no-)?ssse3`),
124 re(`-mthumb(-interwork)?`),
125 re(`-mthreads`),
126 re(`-mwindows`),
127 re(`-no-canonical-prefixes`),
128 re(`--param=ssp-buffer-size=[0-9]*`),
129 re(`-pedantic(-errors)?`),
130 re(`-pipe`),
131 re(`-pthread`),
132 re(`--static`),
133 re(`-?-std=([^@\-].*)`),
134 re(`-?-stdlib=([^@\-].*)`),
135 re(`--sysroot=([^@\-].*)`),
136 re(`-w`),
137 re(`-x([^@\-].*)`),
138 re(`-v`),
139 }
140
141 var validCompilerFlagsWithNextArg = []string{
142 "-arch",
143 "-D",
144 "-U",
145 "-I",
146 "-F",
147 "-framework",
148 "-include",
149 "-isysroot",
150 "-isystem",
151 "--sysroot",
152 "-target",
153 "-x",
154 }
155
156 var invalidLinkerFlags = []*lazyregexp.Regexp{
157
158
159 re(`-lto_library`),
160 }
161
162 var validLinkerFlags = []*lazyregexp.Regexp{
163 re(`-F([^@\-].*)`),
164 re(`-l([^@\-].*)`),
165 re(`-L([^@\-].*)`),
166 re(`-O`),
167 re(`-O([^@\-].*)`),
168 re(`-f(no-)?(pic|PIC|pie|PIE)`),
169 re(`-f(no-)?openmp(-simd)?`),
170 re(`-fsanitize=([^@\-].*)`),
171 re(`-flat_namespace`),
172 re(`-g([^@\-].*)?`),
173 re(`-headerpad_max_install_names`),
174 re(`-m(abi|arch|cpu|fpu|simd|tls-dialect|tune)=([^@\-].*)`),
175 re(`-mcmodel=[0-9a-z-]+`),
176 re(`-mfloat-abi=([^@\-].*)`),
177 re(`-m(soft|single|double)-float`),
178 re(`-m(no-)?relax`),
179 re(`-m(no-)?strict-align`),
180 re(`-m(no-)?(lsx|lasx|frecipe|div32|lam-bh|lamcas|ld-seq-sa)`),
181 re(`-mmacosx-(.+)`),
182 re(`-mios-simulator-version-min=(.+)`),
183 re(`-miphoneos-version-min=(.+)`),
184 re(`-mthreads`),
185 re(`-mwindows`),
186 re(`-(pic|PIC|pie|PIE)`),
187 re(`-pthread`),
188 re(`-rdynamic`),
189 re(`-shared`),
190 re(`-?-static([-a-z0-9+]*)`),
191 re(`-?-stdlib=([^@\-].*)`),
192 re(`-v`),
193
194
195
196
197
198
199 re(`-Wl,--(no-)?allow-multiple-definition`),
200 re(`-Wl,--(no-)?allow-shlib-undefined`),
201 re(`-Wl,--(no-)?as-needed`),
202 re(`-Wl,-Bdynamic`),
203 re(`-Wl,-berok`),
204 re(`-Wl,-Bstatic`),
205 re(`-Wl,-Bsymbolic-functions`),
206 re(`-Wl,-O[0-9]+`),
207 re(`-Wl,-d[ny]`),
208 re(`-Wl,--disable-new-dtags`),
209 re(`-Wl,-e[=,][a-zA-Z0-9]+`),
210 re(`-Wl,--enable-new-dtags`),
211 re(`-Wl,--end-group`),
212 re(`-Wl,--(no-)?export-dynamic`),
213 re(`-Wl,-E`),
214 re(`-Wl,-framework,[^,@\-][^,]*`),
215 re(`-Wl,--hash-style=(sysv|gnu|both)`),
216 re(`-Wl,-headerpad_max_install_names`),
217 re(`-Wl,--no-undefined`),
218 re(`-Wl,--pop-state`),
219 re(`-Wl,--push-state`),
220 re(`-Wl,-R,?([^@\-,][^,@]*$)`),
221 re(`-Wl,--just-symbols[=,]([^,@\-][^,@]*)`),
222 re(`-Wl,-rpath(-link)?[=,]([^,@\-][^,]*)`),
223 re(`-Wl,-s`),
224 re(`-Wl,-search_paths_first`),
225 re(`-Wl,-sectcreate,([^,@\-][^,]*),([^,@\-][^,]*),([^,@\-][^,]*)`),
226 re(`-Wl,--start-group`),
227 re(`-Wl,-?-static`),
228 re(`-Wl,-?-subsystem,(native|windows|console|posix|xbox)`),
229 re(`-Wl,-syslibroot[=,]([^,@\-][^,]*)`),
230 re(`-Wl,-undefined[=,]([^,@\-][^,]*)`),
231 re(`-Wl,-?-unresolved-symbols=[^,]+`),
232 re(`-Wl,--(no-)?warn-([^,]+)`),
233 re(`-Wl,-?-wrap[=,][^,@\-][^,]*`),
234 re(`-Wl(,-z,(relro|now|(no)?execstack))+`),
235
236 re(`[a-zA-Z0-9_/].*\.(a|o|obj|dll|dylib|so|tbd)`),
237 re(`\./.*\.(a|o|obj|dll|dylib|so|tbd)`),
238 }
239
240 var validLinkerFlagsWithNextArg = []string{
241 "-arch",
242 "-F",
243 "-l",
244 "-L",
245 "-framework",
246 "-isysroot",
247 "--sysroot",
248 "-target",
249 "-Wl,-framework",
250 "-Wl,-rpath",
251 "-Wl,-R",
252 "-Wl,--just-symbols",
253 "-Wl,-undefined",
254 }
255
256 var validPkgConfigFlags = []*lazyregexp.Regexp{
257 re(`--atleast-pkgconfig-version=\d+\.\d+\.\d+`),
258 re(`--atleast-version=\d+\.\d+\.\d+`),
259 re(`--cflags-only-I`),
260 re(`--cflags`),
261 re(`--define-prefix`),
262 re(`--define-variable=[A-Za-z_][A-Za-z0-9_]*=[^@\-]*`),
263 re(`--digraph`),
264 re(`--dont-define-prefix`),
265 re(`--dont-relocate-paths`),
266 re(`--dump-personality`),
267 re(`--env-only`),
268 re(`--errors-to-stdout`),
269 re(`--exact-version=\d+\.\d+\.\d+`),
270 re(`--exists`),
271 re(`--fragment-filter=[A-Za-z_][a-zA-Z0-9_]*`),
272 re(`--ignore-conflicts`),
273 re(`--internal-cflags`),
274 re(`--keep-system-cflags`),
275 re(`--keep-system-libs`),
276 re(`--libs-only-l`),
277 re(`--libs-only-L`),
278 re(`--libs`),
279 re(`--list-all`),
280 re(`--list-package-names`),
281 re(`--max-version=\d+\.\d+\.\d+`),
282 re(`--maximum-traverse-depth=[0-9]+`),
283 re(`--modversion`),
284 re(`--msvc-syntax`),
285 re(`--no-cache`),
286 re(`--no-provides`),
287 re(`--no-uninstalled`),
288 re(`--path`),
289 re(`--personality=(triplet|filename)`),
290 re(`--prefix-variable=[A-Za-z_][a-zA-Z0-9_]*`),
291 re(`--print-errors`),
292 re(`--print-provides`),
293 re(`--print-requires-private`),
294 re(`--print-requires`),
295 re(`--print-variables`),
296 re(`--pure`),
297 re(`--shared`),
298 re(`--short-errors`),
299 re(`--silence-errors`),
300 re(`--simulate`),
301 re(`--static`),
302 re(`--uninstalled`),
303 re(`--validate`),
304 re(`--variable=[A-Za-z_][a-zA-Z0-9_]*`),
305 re(`--with-path=[^@\-].*`),
306 }
307
308 func checkCompilerFlags(name, source string, list []string) error {
309 checkOverrides := true
310 return checkFlags(name, source, list, nil, validCompilerFlags, validCompilerFlagsWithNextArg, checkOverrides)
311 }
312
313 func checkLinkerFlags(name, source string, list []string) error {
314 checkOverrides := true
315 return checkFlags(name, source, list, invalidLinkerFlags, validLinkerFlags, validLinkerFlagsWithNextArg, checkOverrides)
316 }
317
318 func checkPkgConfigFlags(name, source string, list []string) error {
319 checkOverrides := false
320 return checkFlags(name, source, list, nil, validPkgConfigFlags, nil, checkOverrides)
321 }
322
323
324
325
326
327 func checkCompilerFlagsForInternalLink(name, source string, list []string) error {
328 checkOverrides := false
329 if err := checkFlags(name, source, list, nil, validCompilerFlags, validCompilerFlagsWithNextArg, checkOverrides); err != nil {
330 return err
331 }
332
333
334 for _, fl := range list {
335 if strings.HasPrefix(fl, "-flto") {
336 return fmt.Errorf("flag %q triggers external linking", fl)
337 }
338 }
339 return nil
340 }
341
342 func checkFlags(name, source string, list []string, invalid, valid []*lazyregexp.Regexp, validNext []string, checkOverrides bool) error {
343
344 var (
345 allow *regexp.Regexp
346 disallow *regexp.Regexp
347 )
348 if checkOverrides {
349 if env := cfg.Getenv("CGO_" + name + "_ALLOW"); env != "" {
350 r, err := regexp.Compile(env)
351 if err != nil {
352 return fmt.Errorf("parsing $CGO_%s_ALLOW: %v", name, err)
353 }
354 allow = r
355 }
356 if env := cfg.Getenv("CGO_" + name + "_DISALLOW"); env != "" {
357 r, err := regexp.Compile(env)
358 if err != nil {
359 return fmt.Errorf("parsing $CGO_%s_DISALLOW: %v", name, err)
360 }
361 disallow = r
362 }
363 }
364
365 Args:
366 for i := 0; i < len(list); i++ {
367 arg := list[i]
368 if disallow != nil && disallow.FindString(arg) == arg {
369 goto Bad
370 }
371 if allow != nil && allow.FindString(arg) == arg {
372 continue Args
373 }
374 for _, re := range invalid {
375 if re.FindString(arg) == arg {
376 goto Bad
377 }
378 }
379 for _, re := range valid {
380 if match := re.FindString(arg); match == arg {
381 continue Args
382 } else if strings.HasPrefix(arg, "-Wl,--push-state,") {
383
384
385
386 args := strings.Split(arg, ",")
387 for _, a := range args[1:] {
388 a = "-Wl," + a
389 var found bool
390 for _, re := range valid {
391 if re.FindString(a) == a {
392 found = true
393 break
394 }
395 }
396 if !found {
397 goto Bad
398 }
399 for _, re := range invalid {
400 if re.FindString(a) == a {
401 goto Bad
402 }
403 }
404 }
405 continue Args
406 }
407 }
408 for _, x := range validNext {
409 if arg == x {
410 if i+1 < len(list) && load.SafeArg(list[i+1]) {
411 i++
412 continue Args
413 }
414
415
416 if i+1 < len(list) &&
417 strings.HasPrefix(arg, "-Wl,") &&
418 strings.HasPrefix(list[i+1], "-Wl,") &&
419 load.SafeArg(list[i+1][4:]) &&
420 !strings.Contains(list[i+1][4:], ",") {
421 i++
422 continue Args
423 }
424
425
426 if i+1 < len(list) && arg == "-I" {
427 if (strings.HasPrefix(list[i+1], "=") || strings.HasPrefix(list[i+1], "$SYSROOT")) &&
428 load.SafeArg(list[i+1][1:]) {
429 i++
430 continue Args
431 }
432 }
433
434 if i+1 < len(list) {
435 return fmt.Errorf("invalid flag in %s: %s %s (see https://go.dev/s/invalidflag)", source, arg, list[i+1])
436 }
437 return fmt.Errorf("invalid flag in %s: %s without argument (see https://go.dev/s/invalidflag)", source, arg)
438 }
439 }
440 Bad:
441 return fmt.Errorf("invalid flag in %s: %s (see https://go.dev/s/invalidflag)", source, arg)
442 }
443 return nil
444 }
445
View as plain text