| | |
| | |
| | |
| |
|
| | package noder |
| |
|
| | import ( |
| | "fmt" |
| | "internal/buildcfg" |
| | "strings" |
| |
|
| | "cmd/compile/internal/ir" |
| | "cmd/compile/internal/syntax" |
| | ) |
| |
|
| | func isSpace(c rune) bool { |
| | return c == ' ' || c == '\t' || c == '\n' || c == '\r' |
| | } |
| |
|
| | func isQuoted(s string) bool { |
| | return len(s) >= 2 && s[0] == '"' && s[len(s)-1] == '"' |
| | } |
| |
|
| | const ( |
| | funcPragmas = ir.Nointerface | |
| | ir.Noescape | |
| | ir.Norace | |
| | ir.Nosplit | |
| | ir.Noinline | |
| | ir.NoCheckPtr | |
| | ir.RegisterParams | |
| | ir.CgoUnsafeArgs | |
| | ir.UintptrKeepAlive | |
| | ir.UintptrEscapes | |
| | ir.Systemstack | |
| | ir.Nowritebarrier | |
| | ir.Nowritebarrierrec | |
| | ir.Yeswritebarrierrec |
| | ) |
| |
|
| | func pragmaFlag(verb string) ir.PragmaFlag { |
| | switch verb { |
| | case "go:build": |
| | return ir.GoBuildPragma |
| | case "go:nointerface": |
| | if buildcfg.Experiment.FieldTrack { |
| | return ir.Nointerface |
| | } |
| | case "go:noescape": |
| | return ir.Noescape |
| | case "go:norace": |
| | return ir.Norace |
| | case "go:nosplit": |
| | return ir.Nosplit | ir.NoCheckPtr |
| | case "go:noinline": |
| | return ir.Noinline |
| | case "go:nocheckptr": |
| | return ir.NoCheckPtr |
| | case "go:systemstack": |
| | return ir.Systemstack |
| | case "go:nowritebarrier": |
| | return ir.Nowritebarrier |
| | case "go:nowritebarrierrec": |
| | return ir.Nowritebarrierrec | ir.Nowritebarrier |
| | case "go:yeswritebarrierrec": |
| | return ir.Yeswritebarrierrec |
| | case "go:cgo_unsafe_args": |
| | return ir.CgoUnsafeArgs | ir.NoCheckPtr |
| | case "go:uintptrkeepalive": |
| | return ir.UintptrKeepAlive |
| | case "go:uintptrescapes": |
| | |
| | |
| | |
| | return ir.UintptrEscapes | ir.UintptrKeepAlive |
| | case "go:registerparams": |
| | return ir.RegisterParams |
| | } |
| | return 0 |
| | } |
| |
|
| | |
| | func (p *noder) pragcgo(pos syntax.Pos, text string) { |
| | f := pragmaFields(text) |
| |
|
| | verb := strings.TrimPrefix(f[0], "go:") |
| | f[0] = verb |
| |
|
| | switch verb { |
| | case "cgo_export_static", "cgo_export_dynamic": |
| | switch { |
| | case len(f) == 2 && !isQuoted(f[1]): |
| | case len(f) == 3 && !isQuoted(f[1]) && !isQuoted(f[2]): |
| | default: |
| | p.error(syntax.Error{Pos: pos, Msg: fmt.Sprintf(`usage: //go:%s local [remote]`, verb)}) |
| | return |
| | } |
| | case "cgo_import_dynamic": |
| | switch { |
| | case len(f) == 2 && !isQuoted(f[1]): |
| | case len(f) == 3 && !isQuoted(f[1]) && !isQuoted(f[2]): |
| | case len(f) == 4 && !isQuoted(f[1]) && !isQuoted(f[2]) && isQuoted(f[3]): |
| | f[3] = strings.Trim(f[3], `"`) |
| | if buildcfg.GOOS == "aix" && f[3] != "" { |
| | |
| | |
| | n := strings.Split(f[3], "/") |
| | if len(n) != 2 || !strings.HasSuffix(n[0], ".a") || (!strings.HasSuffix(n[1], ".o") && !strings.Contains(n[1], ".so.")) { |
| | p.error(syntax.Error{Pos: pos, Msg: `usage: //go:cgo_import_dynamic local [remote ["lib.a/object.o"]]`}) |
| | return |
| | } |
| | } |
| | default: |
| | p.error(syntax.Error{Pos: pos, Msg: `usage: //go:cgo_import_dynamic local [remote ["library"]]`}) |
| | return |
| | } |
| | case "cgo_import_static": |
| | switch { |
| | case len(f) == 2 && !isQuoted(f[1]): |
| | default: |
| | p.error(syntax.Error{Pos: pos, Msg: `usage: //go:cgo_import_static local`}) |
| | return |
| | } |
| | case "cgo_dynamic_linker": |
| | switch { |
| | case len(f) == 2 && isQuoted(f[1]): |
| | f[1] = strings.Trim(f[1], `"`) |
| | default: |
| | p.error(syntax.Error{Pos: pos, Msg: `usage: //go:cgo_dynamic_linker "path"`}) |
| | return |
| | } |
| | case "cgo_ldflag": |
| | switch { |
| | case len(f) == 2 && isQuoted(f[1]): |
| | f[1] = strings.Trim(f[1], `"`) |
| | default: |
| | p.error(syntax.Error{Pos: pos, Msg: `usage: //go:cgo_ldflag "arg"`}) |
| | return |
| | } |
| | default: |
| | return |
| | } |
| | p.pragcgobuf = append(p.pragcgobuf, f) |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| | func pragmaFields(s string) []string { |
| | var a []string |
| | inQuote := false |
| | fieldStart := -1 |
| | for i, c := range s { |
| | switch { |
| | case c == '"': |
| | if inQuote { |
| | inQuote = false |
| | a = append(a, s[fieldStart:i+1]) |
| | fieldStart = -1 |
| | } else { |
| | inQuote = true |
| | if fieldStart >= 0 { |
| | a = append(a, s[fieldStart:i]) |
| | } |
| | fieldStart = i |
| | } |
| | case !inQuote && isSpace(c): |
| | if fieldStart >= 0 { |
| | a = append(a, s[fieldStart:i]) |
| | fieldStart = -1 |
| | } |
| | default: |
| | if fieldStart == -1 { |
| | fieldStart = i |
| | } |
| | } |
| | } |
| | if !inQuote && fieldStart >= 0 { |
| | a = append(a, s[fieldStart:]) |
| | } |
| | return a |
| | } |
| |
|