interanal/core/runtime: move building logic from cue package
- Simplify logic
- Lazy load packages
- Cleanup interim types used for cue package transition
The lazy loading of builtin packages allows:
- adding builtin packages to a used Runtime
- isolate performance measurements and metrics for
CUE code that doesn't use imports.
Change-Id: Idd4578100047f83397be7c1d69f66b083974e6c0
Reviewed-on: https://cue-review.googlesource.com/c/cue/+/7421
Reviewed-by: CUE cueckoo <cueckoo@gmail.com>
Reviewed-by: Marcel van Lohuizen <mpvl@golang.org>
diff --git a/internal/core/runtime/build.go b/internal/core/runtime/build.go
new file mode 100644
index 0000000..c167e16
--- /dev/null
+++ b/internal/core/runtime/build.go
@@ -0,0 +1,94 @@
+// Copyright 2020 CUE Authors
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package runtime
+
+import (
+ "strings"
+
+ "cuelang.org/go/cue/ast"
+ "cuelang.org/go/cue/ast/astutil"
+ "cuelang.org/go/cue/build"
+ "cuelang.org/go/cue/errors"
+ "cuelang.org/go/internal/core/adt"
+ "cuelang.org/go/internal/core/compile"
+)
+
+// Build builds b and all its transitive dependencies, insofar they have not
+// been build yet.
+func (x *Runtime) Build(b *build.Instance) (v *adt.Vertex, errs errors.Error) {
+ if v := x.GetNodeFromInstance(b); v != nil {
+ return v, b.Err
+ }
+ // TODO: clear cache of old implementation.
+ // if s := b.ImportPath; s != "" {
+ // // Use cached result, if available.
+ // if v, err := x.LoadImport(s); v != nil || err != nil {
+ // return v, err
+ // }
+ // }
+
+ errs = b.Err
+
+ // Build transitive dependencies.
+ for _, file := range b.Files {
+ file.VisitImports(func(d *ast.ImportDecl) {
+ for _, s := range d.Specs {
+ errs = errors.Append(errs, x.buildSpec(b, s))
+ }
+ })
+ }
+
+ err := x.ResolveFiles(b)
+ errs = errors.Append(errs, err)
+
+ v, err = compile.Files(nil, x, b.ID(), b.Files...)
+ errs = errors.Append(errs, err)
+
+ if errs != nil {
+ v = adt.ToVertex(&adt.Bottom{Err: errs})
+ b.Err = errs
+ }
+
+ x.AddInst(b.ImportPath, v, b)
+
+ return v, errs
+}
+
+func (x *Runtime) buildSpec(b *build.Instance, spec *ast.ImportSpec) (errs errors.Error) {
+ info, err := astutil.ParseImportSpec(spec)
+ if err != nil {
+ return errors.Promote(err, "invalid import path")
+ }
+
+ pkg := b.LookupImport(info.ID)
+ if pkg == nil {
+ if strings.Contains(info.ID, ".") {
+ return errors.Newf(spec.Pos(),
+ "package %q imported but not defined in %s",
+ info.ID, b.ImportPath)
+ }
+ return nil // TODO: check the builtin package exists here.
+ }
+
+ if v := x.index.importsByBuild[pkg]; v != nil {
+ return pkg.Err
+ }
+
+ if _, err := x.Build(pkg); err != nil {
+ return err
+ }
+
+ return nil
+}
diff --git a/internal/core/runtime/go.go b/internal/core/runtime/go.go
new file mode 100644
index 0000000..4b84b3f
--- /dev/null
+++ b/internal/core/runtime/go.go
@@ -0,0 +1,53 @@
+// Copyright 2020 CUE Authors
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package runtime
+
+import (
+ "reflect"
+
+ "cuelang.org/go/cue/ast"
+ "cuelang.org/go/internal/core/adt"
+)
+
+func (x *Runtime) StoreType(t reflect.Type, src ast.Expr, expr adt.Expr) {
+ if expr == nil {
+ x.index.StoreType(t, src)
+ } else {
+ x.index.StoreType(t, expr)
+ }
+}
+
+func (x *Runtime) LoadType(t reflect.Type) (src ast.Expr, expr adt.Expr, ok bool) {
+ v, ok := x.index.LoadType(t)
+ if ok {
+ switch x := v.(type) {
+ case ast.Expr:
+ return x, nil, true
+ case adt.Expr:
+ src, _ = x.Source().(ast.Expr)
+ return src, x, true
+ }
+ }
+ return nil, nil, false
+}
+
+func (x *index) StoreType(t reflect.Type, v interface{}) {
+ x.typeCache.Store(t, v)
+}
+
+func (x *index) LoadType(t reflect.Type) (v interface{}, ok bool) {
+ v, ok = x.typeCache.Load(t)
+ return v, ok
+}
diff --git a/internal/core/runtime/imports.go b/internal/core/runtime/imports.go
new file mode 100644
index 0000000..4f25c96
--- /dev/null
+++ b/internal/core/runtime/imports.go
@@ -0,0 +1,128 @@
+// Copyright 2020 CUE Authors
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package runtime
+
+import (
+ "path"
+ "sync"
+
+ "cuelang.org/go/cue/build"
+ "cuelang.org/go/cue/errors"
+ "cuelang.org/go/internal/core/adt"
+)
+
+type PackageFunc func(ctx adt.Runtime) (*adt.Vertex, errors.Error)
+
+func RegisterBuiltin(importPath string, f PackageFunc) {
+ sharedIndex.RegisterBuiltin(importPath, f)
+}
+
+func (x *index) RegisterBuiltin(importPath string, f PackageFunc) {
+ if x.builtins == nil {
+ x.builtins = map[string]PackageFunc{}
+ }
+ x.builtins[importPath] = f
+}
+
+var SharedRuntime = &Runtime{index: sharedIndex}
+
+func (x *Runtime) IsBuiltinPackage(path string) bool {
+ return x.index.isBuiltin(path)
+}
+
+// sharedIndex is used for indexing builtins and any other labels common to
+// all instances.
+var sharedIndex = newIndex()
+
+// index maps conversions from label names to internal codes.
+//
+// All instances belonging to the same package should share this index.
+type index struct {
+ // Change this to Instance at some point.
+ // From *structLit/*Vertex -> Instance
+ imports map[*adt.Vertex]*build.Instance
+ importsByPath map[string]*adt.Vertex
+ importsByBuild map[*build.Instance]*adt.Vertex
+ builtins map[string]PackageFunc
+
+ // mutex sync.Mutex
+ typeCache sync.Map // map[reflect.Type]evaluated
+
+}
+
+func newIndex() *index {
+ i := &index{
+ imports: map[*adt.Vertex]*build.Instance{},
+ importsByPath: map[string]*adt.Vertex{},
+ importsByBuild: map[*build.Instance]*adt.Vertex{},
+ }
+ return i
+}
+
+func (x *index) isBuiltin(id string) bool {
+ if x == nil || x.builtins == nil {
+ return false
+ }
+ _, ok := x.builtins[id]
+ return ok
+}
+
+func (r *Runtime) AddInst(path string, key *adt.Vertex, p *build.Instance) {
+ x := r.index
+ if key == nil {
+ panic("key must not be nil")
+ }
+ x.imports[key] = p
+ x.importsByBuild[p] = key
+ if path != "" {
+ x.importsByPath[path] = key
+ }
+}
+
+func (r *Runtime) GetInstanceFromNode(key *adt.Vertex) *build.Instance {
+ return r.index.imports[key]
+}
+
+func (r *Runtime) GetNodeFromInstance(key *build.Instance) *adt.Vertex {
+ return r.index.importsByBuild[key]
+}
+
+func (r *Runtime) LoadImport(importPath string) (*adt.Vertex, errors.Error) {
+ x := r.index
+
+ key := x.importsByPath[importPath]
+ if key != nil {
+ return key, nil
+ }
+
+ if x.builtins != nil {
+ if f := x.builtins[importPath]; f != nil {
+ p, err := f(r)
+ if err != nil {
+ return adt.ToVertex(&adt.Bottom{Err: err}), nil
+ }
+ inst := &build.Instance{
+ ImportPath: importPath,
+ PkgName: path.Base(importPath),
+ }
+ x.imports[p] = inst
+ x.importsByPath[importPath] = p
+ x.importsByBuild[inst] = p
+ return p, nil
+ }
+ }
+
+ return key, nil
+}
diff --git a/internal/core/runtime/index.go b/internal/core/runtime/index.go
index a8e381c..bf5867c 100644
--- a/internal/core/runtime/index.go
+++ b/internal/core/runtime/index.go
@@ -15,122 +15,30 @@
package runtime
import (
- "reflect"
"sync"
- "cuelang.org/go/cue/ast"
"cuelang.org/go/internal"
"cuelang.org/go/internal/core/adt"
)
-// Index maps conversions from label names to internal codes.
-//
-// All instances belonging to the same package should share this Index.
-//
-// INDEX IS A TRANSITIONAL TYPE TO BRIDGE THE OLD AND NEW
-// IMPLEMENTATIONS. USE RUNTIME.
-type Index struct {
- labelMap map[string]int64
- labels []string
-
- // Change this to Instance at some point.
- // From *structLit/*Vertex -> Instance
- imports map[interface{}]interface{}
- importsByPath map[string]interface{}
- // imports map[string]*adt.Vertex
-
- offset int64
- parent *Index
-
- // mutex sync.Mutex
- typeCache sync.Map // map[reflect.Type]evaluated
+func (r *Runtime) IndexToString(i int64) string {
+ return r.index.IndexToString(i)
}
-// SharedIndex is used for indexing builtins and any other labels common to
-// all instances.
-var SharedIndex = newSharedIndex()
-
-var SharedIndexNew = newSharedIndex()
-
-var SharedRuntimeNew = &Runtime{index: SharedIndexNew}
-
-func newSharedIndex() *Index {
- i := &Index{
- labelMap: map[string]int64{"": 0},
- labels: []string{""},
- imports: map[interface{}]interface{}{},
- importsByPath: map[string]interface{}{},
- }
- return i
+func (r *Runtime) StringToIndex(s string) int64 {
+ return getKey(s)
}
-// NewIndex creates a new index.
-func NewIndex(parent *Index) *Index {
- i := &Index{
- labelMap: map[string]int64{},
- imports: map[interface{}]interface{}{},
- importsByPath: map[string]interface{}{},
- offset: int64(len(parent.labels)) + parent.offset,
- parent: parent,
- }
- return i
+func (r *Runtime) LabelStr(l adt.Feature) string {
+ return l.IdentString(r)
}
-func (x *Index) IndexToString(i int64) string {
- for ; i < x.offset; x = x.parent {
- }
- return x.labels[i-x.offset]
+func (r *Runtime) StrLabel(str string) adt.Feature {
+ return r.Label(str, false)
}
-func (x *Index) StringToIndex(s string) int64 {
- for p := x; p != nil; p = p.parent {
- if f, ok := p.labelMap[s]; ok {
- return int64(f)
- }
- }
- index := int64(len(x.labelMap)) + x.offset
- x.labelMap[s] = index
- x.labels = append(x.labels, s)
- return int64(index)
-}
-
-func (x *Index) HasLabel(s string) (ok bool) {
- for c := x; c != nil; c = c.parent {
- _, ok = c.labelMap[s]
- if ok {
- break
- }
- }
- return ok
-}
-
-func (x *Index) StoreType(t reflect.Type, v interface{}) {
- x.typeCache.Store(t, v)
-}
-
-func (x *Index) LoadType(t reflect.Type) (v interface{}, ok bool) {
- v, ok = x.typeCache.Load(t)
- return v, ok
-}
-
-func (x *Index) StrLabel(str string) adt.Feature {
- return x.Label(str, false)
-}
-
-func (x *Index) NodeLabel(n ast.Node) (f adt.Feature, ok bool) {
- switch label := n.(type) {
- case *ast.BasicLit:
- name, _, err := ast.LabelName(label)
- return x.Label(name, false), err == nil
- case *ast.Ident:
- name, err := ast.ParseIdent(label)
- return x.Label(name, true), err == nil
- }
- return 0, false
-}
-
-func (x *Index) Label(s string, isIdent bool) adt.Feature {
- index := x.StringToIndex(s)
+func (r *Runtime) Label(s string, isIdent bool) adt.Feature {
+ index := r.StringToIndex(s)
typ := adt.StringLabel
if isIdent {
switch {
@@ -146,32 +54,39 @@
return f
}
-func (idx *Index) LabelStr(l adt.Feature) string {
- return l.IdentString(idx)
+// TODO: move to Runtime as fields.
+var (
+ labelMap = map[string]int{}
+ labels = make([]string, 0, 1000)
+ mutex sync.RWMutex
+)
+
+func init() {
+ getKey("")
}
-func (x *Index) AddInst(path string, key, p interface{}) {
- if key == nil {
- panic("key must not be nil")
+func getKey(s string) int64 {
+ mutex.RLock()
+ p, ok := labelMap[s]
+ mutex.RUnlock()
+ if ok {
+ return int64(p)
}
- x.imports[key] = p
- if path != "" {
- x.importsByPath[path] = key
+ mutex.Lock()
+ defer mutex.Unlock()
+ p, ok = labelMap[s]
+ if ok {
+ return int64(p)
}
+ p = len(labels)
+ labels = append(labels, s)
+ labelMap[s] = p
+ return int64(p)
}
-func (x *Index) GetImportFromNode(key interface{}) interface{} {
- imp := x.imports[key]
- if imp == nil && x.parent != nil {
- return x.parent.GetImportFromNode(key)
- }
- return imp
-}
-
-func (x *Index) GetImportFromPath(id string) interface{} {
- key := x.importsByPath[id]
- if key == nil && x.parent != nil {
- return x.parent.GetImportFromPath(id)
- }
- return key
+func (x *index) IndexToString(i int64) string {
+ mutex.RLock()
+ s := labels[i]
+ mutex.RUnlock()
+ return s
}
diff --git a/internal/core/runtime/resolve.go b/internal/core/runtime/resolve.go
index 29f19e4..94188db 100644
--- a/internal/core/runtime/resolve.go
+++ b/internal/core/runtime/resolve.go
@@ -23,15 +23,9 @@
"cuelang.org/go/cue/errors"
)
-func lineStr(idx *Index, n ast.Node) string {
- return n.Pos().String()
-}
+func (r *Runtime) ResolveFiles(p *build.Instance) (errs errors.Error) {
+ idx := r.index
-func ResolveFiles(
- idx *Index,
- p *build.Instance,
- isBuiltin func(s string) bool,
-) (errs errors.Error) {
// Link top-level declarations. As top-level entries get unified, an entry
// may be linked to any top-level entry of any of the files.
allFields := map[string]ast.Node{}
@@ -45,18 +39,17 @@
}
}
for _, f := range p.Files {
- err := ResolveFile(idx, f, p, allFields, isBuiltin)
+ err := resolveFile(idx, f, p, allFields)
errs = errors.Append(errs, err)
}
return errs
}
-func ResolveFile(
- idx *Index,
+func resolveFile(
+ idx *index,
f *ast.File,
p *build.Instance,
allFields map[string]ast.Node,
- isBuiltin func(s string) bool,
) errors.Error {
unresolved := map[string][]*ast.Ident{}
for _, u := range f.Unresolved {
@@ -82,7 +75,7 @@
name := path.Base(id)
if imp := p.LookupImport(id); imp != nil {
name = imp.PkgName
- } else if !isBuiltin(id) {
+ } else if _, ok := idx.builtins[id]; !ok {
errs = errors.Append(errs,
nodeErrorf(spec, "package %q not found", id))
continue
@@ -162,3 +155,7 @@
// }
return errs
}
+
+func lineStr(idx *index, n ast.Node) string {
+ return n.Pos().String()
+}
diff --git a/internal/core/runtime/resolve_test.go b/internal/core/runtime/resolve_test.go
index 42ec1dd..9bc7240 100644
--- a/internal/core/runtime/resolve_test.go
+++ b/internal/core/runtime/resolve_test.go
@@ -33,8 +33,6 @@
Path: ast.NewString(importPath),
}
- isBuiltin := func(s string) bool { return false }
-
f := &ast.File{
Decls: []ast.Decl{
&ast.ImportDecl{Specs: []*ast.ImportSpec{spec1, spec2}},
@@ -50,12 +48,12 @@
Imports: []*ast.ImportSpec{spec1, spec2},
}
- err := ResolveFile(nil, f, &build.Instance{
+ err := resolveFile(nil, f, &build.Instance{
Imports: []*build.Instance{{
ImportPath: importPath,
PkgName: "foo",
}},
- }, map[string]ast.Node{}, isBuiltin)
+ }, map[string]ast.Node{})
if err != nil {
t.Errorf("exected no error, found %v", err)
diff --git a/internal/core/runtime/runtime.go b/internal/core/runtime/runtime.go
index 6361e6b..448dc30 100644
--- a/internal/core/runtime/runtime.go
+++ b/internal/core/runtime/runtime.go
@@ -14,124 +14,18 @@
package runtime
-import (
- "reflect"
- "strings"
-
- "cuelang.org/go/cue/ast"
- "cuelang.org/go/cue/ast/astutil"
- "cuelang.org/go/cue/build"
- "cuelang.org/go/cue/errors"
- "cuelang.org/go/internal/core/adt"
- "cuelang.org/go/internal/core/compile"
-)
-
// A Runtime maintains data structures for indexing and resuse for evaluation.
type Runtime struct {
- index *Index
+ index *index
// Data holds the legacy index strut. It is for transitional purposes only.
Data interface{}
}
-// New creates a new Runtime.
+// New creates a new Runtime. The builtins registered with RegisterBuiltin
+// are available for
func New() *Runtime {
return &Runtime{
- index: NewIndex(SharedIndexNew),
+ index: sharedIndex,
}
}
-
-func NewWithIndex(x *Index) *Runtime {
- return &Runtime{index: x}
-}
-
-func (x *Runtime) IndexToString(i int64) string {
- return x.index.IndexToString(i)
-}
-
-func (x *Runtime) StringToIndex(s string) int64 {
- return x.index.StringToIndex(s)
-}
-
-func (x *Runtime) Build(b *build.Instance) (v *adt.Vertex, errs errors.Error) {
- if s := b.ImportPath; s != "" {
- // Use cached result, if available.
- if v, err := x.LoadImport(s); v != nil || err != nil {
- return v, err
- }
- // Cache the result if any.
- defer func() {
- if errs == nil && v != nil {
- x.index.AddInst(b.ImportPath, v, b)
- }
- }()
- }
-
- // Build transitive dependencies.
- for _, file := range b.Files {
- file.VisitImports(func(d *ast.ImportDecl) {
- for _, s := range d.Specs {
- errs = errors.Append(errs, x.buildSpec(b, s))
- }
- })
- }
-
- if errs != nil {
- return nil, errs
- }
-
- return compile.Files(nil, x, b.ID(), b.Files...)
-}
-
-func (x *Runtime) buildSpec(b *build.Instance, spec *ast.ImportSpec) (errs errors.Error) {
- info, err := astutil.ParseImportSpec(spec)
- if err != nil {
- return errors.Promote(err, "invalid import path")
- }
-
- pkg := b.LookupImport(info.ID)
- if pkg == nil {
- if strings.Contains(info.ID, ".") {
- return errors.Newf(spec.Pos(),
- "package %q imported but not defined in %s",
- info.ID, b.ImportPath)
- }
- return nil // TODO: check the builtin package exists here.
- }
-
- if _, err := x.Build(pkg); err != nil {
- return err
- }
-
- return nil
-}
-
-func (x *Runtime) LoadImport(importPath string) (*adt.Vertex, errors.Error) {
- v := x.index.GetImportFromPath(importPath)
- if v == nil {
- return nil, nil
- }
- return v.(*adt.Vertex), nil
-}
-
-func (x *Runtime) StoreType(t reflect.Type, src ast.Expr, expr adt.Expr) {
- if expr == nil {
- x.index.StoreType(t, src)
- } else {
- x.index.StoreType(t, expr)
- }
-}
-
-func (x *Runtime) LoadType(t reflect.Type) (src ast.Expr, expr adt.Expr, ok bool) {
- v, ok := x.index.LoadType(t)
- if ok {
- switch x := v.(type) {
- case ast.Expr:
- return x, nil, true
- case adt.Expr:
- src, _ = x.Source().(ast.Expr)
- return src, x, true
- }
- }
- return nil, nil, false
-}