Skip to content

Commit 3cec330

Browse files
committed
go/types: add UsesCgo config to support _cgo_gotypes.go
(Reland of golang.org/cl/33677.) This CL adds a UsesCgo config setting to go/types to specify that the _cgo_gotypes.go file generated by cmd/cgo has been provided as a source file. The type checker then internally resolves C.bar qualified identifiers to _Cfoo_bar as appropriate. It also adds support to srcimporter to automatically run cgo. Unfortunately, this functionality is not compatible with overriding OpenFile, because cmd/cgo and gcc will directly open files. Updates #16623. Updates #35721. Change-Id: Ib179d55c8c589916f98ceeae0b9a3e746157253a Reviewed-on: https://go-review.googlesource.com/c/go/+/231459 Run-TryBot: Matthew Dempsky <[email protected]> TryBot-Result: Gobot Gobot <[email protected]> Reviewed-by: Robert Griesemer <[email protected]>
1 parent b565d1e commit 3cec330

File tree

10 files changed

+175
-21
lines changed

10 files changed

+175
-21
lines changed

src/go/internal/srcimporter/srcimporter.go

+59-1
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,11 @@ import (
1414
"go/token"
1515
"go/types"
1616
"io"
17+
"io/ioutil"
1718
"os"
19+
"os/exec"
1820
"path/filepath"
21+
"strings"
1922
"sync"
2023
)
2124

@@ -115,7 +118,6 @@ func (p *Importer) ImportFrom(path, srcDir string, mode types.ImportMode) (*type
115118
var firstHardErr error
116119
conf := types.Config{
117120
IgnoreFuncBodies: true,
118-
FakeImportC: true,
119121
// continue type-checking after the first error
120122
Error: func(err error) {
121123
if firstHardErr == nil && !err.(types.Error).Soft {
@@ -125,6 +127,21 @@ func (p *Importer) ImportFrom(path, srcDir string, mode types.ImportMode) (*type
125127
Importer: p,
126128
Sizes: p.sizes,
127129
}
130+
if len(bp.CgoFiles) > 0 {
131+
if p.ctxt.OpenFile != nil {
132+
// cgo, gcc, pkg-config, etc. do not support
133+
// build.Context's VFS.
134+
conf.FakeImportC = true
135+
} else {
136+
conf.UsesCgo = true
137+
file, err := p.cgo(bp)
138+
if err != nil {
139+
return nil, err
140+
}
141+
files = append(files, file)
142+
}
143+
}
144+
128145
pkg, err = conf.Check(bp.ImportPath, p.fset, files, nil)
129146
if err != nil {
130147
// If there was a hard error it is possibly unsafe
@@ -181,6 +198,47 @@ func (p *Importer) parseFiles(dir string, filenames []string) ([]*ast.File, erro
181198
return files, nil
182199
}
183200

201+
func (p *Importer) cgo(bp *build.Package) (*ast.File, error) {
202+
tmpdir, err := ioutil.TempDir("", "srcimporter")
203+
if err != nil {
204+
return nil, err
205+
}
206+
defer os.RemoveAll(tmpdir)
207+
208+
args := []string{"go", "tool", "cgo", "-objdir", tmpdir}
209+
if bp.Goroot {
210+
switch bp.ImportPath {
211+
case "runtime/cgo":
212+
args = append(args, "-import_runtime_cgo=false", "-import_syscall=false")
213+
case "runtime/race":
214+
args = append(args, "-import_syscall=false")
215+
}
216+
}
217+
args = append(args, "--")
218+
args = append(args, strings.Fields(os.Getenv("CGO_CPPFLAGS"))...)
219+
args = append(args, bp.CgoCPPFLAGS...)
220+
if len(bp.CgoPkgConfig) > 0 {
221+
cmd := exec.Command("pkg-config", append([]string{"--cflags"}, bp.CgoPkgConfig...)...)
222+
out, err := cmd.CombinedOutput()
223+
if err != nil {
224+
return nil, err
225+
}
226+
args = append(args, strings.Fields(string(out))...)
227+
}
228+
args = append(args, "-I", tmpdir)
229+
args = append(args, strings.Fields(os.Getenv("CGO_CFLAGS"))...)
230+
args = append(args, bp.CgoCFLAGS...)
231+
args = append(args, bp.CgoFiles...)
232+
233+
cmd := exec.Command(args[0], args[1:]...)
234+
cmd.Dir = bp.Dir
235+
if err := cmd.Run(); err != nil {
236+
return nil, err
237+
}
238+
239+
return parser.ParseFile(p.fset, filepath.Join(tmpdir, "_cgo_gotypes.go"), nil, 0)
240+
}
241+
184242
// context-controlled file system operations
185243

186244
func (p *Importer) absPath(path string) (string, error) {

src/go/internal/srcimporter/srcimporter_test.go

+11
Original file line numberDiff line numberDiff line change
@@ -232,3 +232,14 @@ func TestIssue23092(t *testing.T) {
232232
func TestIssue24392(t *testing.T) {
233233
testImportPath(t, "go/internal/srcimporter/testdata/issue24392")
234234
}
235+
236+
func TestCgo(t *testing.T) {
237+
testenv.MustHaveGoBuild(t)
238+
testenv.MustHaveCGO(t)
239+
240+
importer := New(&build.Default, token.NewFileSet(), make(map[string]*types.Package))
241+
_, err := importer.ImportFrom("./misc/cgo/test", runtime.GOROOT(), 0)
242+
if err != nil {
243+
t.Fatalf("Import failed: %v", err)
244+
}
245+
}

src/go/types/api.go

+10-1
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,15 @@ type Config struct {
105105
// Do not use casually!
106106
FakeImportC bool
107107

108+
// If UsesCgo is set, the type checker expects the
109+
// _cgo_gotypes.go file generated by running cmd/cgo to be
110+
// provided as a package source file. Qualified identifiers
111+
// referring to package C will be resolved to cgo-provided
112+
// declarations within _cgo_gotypes.go.
113+
//
114+
// It is an error to set both FakeImportC and UsesCgo.
115+
UsesCgo bool
116+
108117
// If Error != nil, it is called with each error found
109118
// during type checking; err has dynamic type Error.
110119
// Secondary errors (for instance, to enumerate all types
@@ -281,7 +290,7 @@ func (tv TypeAndValue) IsBuiltin() bool {
281290
// nil Value.
282291
func (tv TypeAndValue) IsValue() bool {
283292
switch tv.mode {
284-
case constant_, variable, mapindex, value, commaok:
293+
case constant_, variable, mapindex, value, commaok, commaerr:
285294
return true
286295
}
287296
return false

src/go/types/assignments.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ func (check *Checker) assignment(x *operand, T Type, context string) {
2222
switch x.mode {
2323
case invalid:
2424
return // error reported before
25-
case constant_, variable, mapindex, value, commaok:
25+
case constant_, variable, mapindex, value, commaok, commaerr:
2626
// ok
2727
default:
2828
unreachable()

src/go/types/call.go

+67-12
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ package types
99
import (
1010
"go/ast"
1111
"go/token"
12+
"strings"
1213
"unicode"
1314
)
1415

@@ -55,6 +56,8 @@ func (check *Checker) call(x *operand, e *ast.CallExpr) exprKind {
5556

5657
default:
5758
// function/method call
59+
cgocall := x.mode == cgofunc
60+
5861
sig, _ := x.typ.Underlying().(*Signature)
5962
if sig == nil {
6063
check.invalidOp(x.pos(), "cannot call non-function %s", x)
@@ -75,7 +78,11 @@ func (check *Checker) call(x *operand, e *ast.CallExpr) exprKind {
7578
case 0:
7679
x.mode = novalue
7780
case 1:
78-
x.mode = value
81+
if cgocall {
82+
x.mode = commaerr
83+
} else {
84+
x.mode = value
85+
}
7986
x.typ = sig.results.vars[0].typ // unpack tuple
8087
default:
8188
x.mode = value
@@ -193,10 +200,13 @@ func unpack(get getter, n int, allowCommaOk bool) (getter, int, bool) {
193200
}, t.Len(), false
194201
}
195202

196-
if x0.mode == mapindex || x0.mode == commaok {
203+
if x0.mode == mapindex || x0.mode == commaok || x0.mode == commaerr {
197204
// comma-ok value
198205
if allowCommaOk {
199206
a := [2]Type{x0.typ, Typ[UntypedBool]}
207+
if x0.mode == commaerr {
208+
a[1] = universeError
209+
}
200210
return func(x *operand, i int) {
201211
x.mode = value
202212
x.expr = x0.expr
@@ -303,6 +313,17 @@ func (check *Checker) argument(sig *Signature, i int, x *operand, ellipsis token
303313
check.assignment(x, typ, context)
304314
}
305315

316+
var cgoPrefixes = [...]string{
317+
"_Ciconst_",
318+
"_Cfconst_",
319+
"_Csconst_",
320+
"_Ctype_",
321+
"_Cvar_", // actually a pointer to the var
322+
"_Cfpvar_fp_",
323+
"_Cfunc_",
324+
"_Cmacro_", // function to evaluate the expanded expression
325+
}
326+
306327
func (check *Checker) selector(x *operand, e *ast.SelectorExpr) {
307328
// these must be declared before the "goto Error" statements
308329
var (
@@ -323,16 +344,43 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) {
323344
check.recordUse(ident, pname)
324345
pname.used = true
325346
pkg := pname.imported
326-
exp := pkg.scope.Lookup(sel)
327-
if exp == nil {
328-
if !pkg.fake {
329-
check.errorf(e.Sel.Pos(), "%s not declared by package %s", sel, pkg.name)
347+
348+
var exp Object
349+
funcMode := value
350+
if pkg.cgo {
351+
// cgo special cases C.malloc: it's
352+
// rewritten to _CMalloc and does not
353+
// support two-result calls.
354+
if sel == "malloc" {
355+
sel = "_CMalloc"
356+
} else {
357+
funcMode = cgofunc
358+
}
359+
for _, prefix := range cgoPrefixes {
360+
// cgo objects are part of the current package (in file
361+
// _cgo_gotypes.go). Use regular lookup.
362+
_, exp = check.scope.LookupParent(prefix+sel, check.pos)
363+
if exp != nil {
364+
break
365+
}
366+
}
367+
if exp == nil {
368+
check.errorf(e.Sel.Pos(), "%s not declared by package C", sel)
369+
goto Error
370+
}
371+
check.objDecl(exp, nil)
372+
} else {
373+
exp = pkg.scope.Lookup(sel)
374+
if exp == nil {
375+
if !pkg.fake {
376+
check.errorf(e.Sel.Pos(), "%s not declared by package %s", sel, pkg.name)
377+
}
378+
goto Error
379+
}
380+
if !exp.Exported() {
381+
check.errorf(e.Sel.Pos(), "%s not exported by package %s", sel, pkg.name)
382+
// ok to continue
330383
}
331-
goto Error
332-
}
333-
if !exp.Exported() {
334-
check.errorf(e.Sel.Pos(), "%s not exported by package %s", sel, pkg.name)
335-
// ok to continue
336384
}
337385
check.recordUse(e.Sel, exp)
338386

@@ -350,9 +398,16 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) {
350398
case *Var:
351399
x.mode = variable
352400
x.typ = exp.typ
401+
if pkg.cgo && strings.HasPrefix(exp.name, "_Cvar_") {
402+
x.typ = x.typ.(*Pointer).base
403+
}
353404
case *Func:
354-
x.mode = value
405+
x.mode = funcMode
355406
x.typ = exp.typ
407+
if pkg.cgo && strings.HasPrefix(exp.name, "_Cmacro_") {
408+
x.mode = value
409+
x.typ = x.typ.(*Signature).results.vars[0].typ
410+
}
356411
case *Builtin:
357412
x.mode = builtin
358413
x.typ = exp.typ

src/go/types/check.go

+8-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
package types
88

99
import (
10+
"errors"
1011
"go/ast"
1112
"go/constant"
1213
"go/token"
@@ -247,7 +248,13 @@ func (check *Checker) handleBailout(err *error) {
247248
// Files checks the provided files as part of the checker's package.
248249
func (check *Checker) Files(files []*ast.File) error { return check.checkFiles(files) }
249250

251+
var errBadCgo = errors.New("cannot use FakeImportC and UsesCgo together")
252+
250253
func (check *Checker) checkFiles(files []*ast.File) (err error) {
254+
if check.conf.FakeImportC && check.conf.UsesCgo {
255+
return errBadCgo
256+
}
257+
251258
defer check.handleBailout(&err)
252259

253260
check.initFiles(files)
@@ -348,7 +355,7 @@ func (check *Checker) recordCommaOkTypes(x ast.Expr, a [2]Type) {
348355
if a[0] == nil || a[1] == nil {
349356
return
350357
}
351-
assert(isTyped(a[0]) && isTyped(a[1]) && isBoolean(a[1]))
358+
assert(isTyped(a[0]) && isTyped(a[1]) && (isBoolean(a[1]) || a[1] == universeError))
352359
if m := check.Types; m != nil {
353360
for {
354361
tv := m[x]

src/go/types/operand.go

+10
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,8 @@ const (
2727
mapindex // operand is a map index expression (acts like a variable on lhs, commaok on rhs of an assignment)
2828
value // operand is a computed value
2929
commaok // like value, but operand may be used in a comma,ok expression
30+
commaerr // like commaok, but second value is error, not boolean
31+
cgofunc // operand is a cgo function
3032
)
3133

3234
var operandModeString = [...]string{
@@ -39,6 +41,8 @@ var operandModeString = [...]string{
3941
mapindex: "map index expression",
4042
value: "value",
4143
commaok: "comma, ok expression",
44+
commaerr: "comma, error expression",
45+
cgofunc: "cgo function",
4246
}
4347

4448
// An operand represents an intermediate value during type checking.
@@ -94,6 +98,12 @@ func (x *operand) pos() token.Pos {
9498
// commaok <expr> (<untyped kind> <mode> )
9599
// commaok <expr> ( <mode> of type <typ>)
96100
//
101+
// commaerr <expr> (<untyped kind> <mode> )
102+
// commaerr <expr> ( <mode> of type <typ>)
103+
//
104+
// cgofunc <expr> (<untyped kind> <mode> )
105+
// cgofunc <expr> ( <mode> of type <typ>)
106+
//
97107
func operandString(x *operand, qf Qualifier) string {
98108
var buf bytes.Buffer
99109

src/go/types/package.go

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ type Package struct {
1717
complete bool
1818
imports []*Package
1919
fake bool // scope lookup errors are silently dropped if package is fake (internal use only)
20+
cgo bool // uses of this package will be rewritten into uses of declarations from _cgo_gotypes.go
2021
}
2122

2223
// NewPackage returns a new Package for the given package path and name.

src/go/types/resolver.go

+3-2
Original file line numberDiff line numberDiff line change
@@ -141,9 +141,10 @@ func (check *Checker) importPackage(pos token.Pos, path, dir string) *Package {
141141
}
142142

143143
// no package yet => import it
144-
if path == "C" && check.conf.FakeImportC {
144+
if path == "C" && (check.conf.FakeImportC || check.conf.UsesCgo) {
145145
imp = NewPackage("C", "C")
146-
imp.fake = true
146+
imp.fake = true // package scope is not populated
147+
imp.cgo = check.conf.UsesCgo
147148
} else {
148149
// ordinary import
149150
var err error

src/go/types/universe.go

+5-3
Original file line numberDiff line numberDiff line change
@@ -21,9 +21,10 @@ var Universe *Scope
2121
var Unsafe *Package
2222

2323
var (
24-
universeIota *Const
25-
universeByte *Basic // uint8 alias, but has name "byte"
26-
universeRune *Basic // int32 alias, but has name "rune"
24+
universeIota *Const
25+
universeByte *Basic // uint8 alias, but has name "byte"
26+
universeRune *Basic // int32 alias, but has name "rune"
27+
universeError *Named
2728
)
2829

2930
// Typ contains the predeclared *Basic types indexed by their
@@ -200,6 +201,7 @@ func init() {
200201
universeIota = Universe.Lookup("iota").(*Const)
201202
universeByte = Universe.Lookup("byte").(*TypeName).typ.(*Basic)
202203
universeRune = Universe.Lookup("rune").(*TypeName).typ.(*Basic)
204+
universeError = Universe.Lookup("error").(*TypeName).typ.(*Named)
203205
}
204206

205207
// Objects with names containing blanks are internal and not entered into

0 commit comments

Comments
 (0)