2013-09-03 13:29:02 -06:00
|
|
|
// Copyright 2013 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2013-09-24 13:08:14 -06:00
|
|
|
// Package serial defines the oracle's schema for structured data
|
|
|
|
// serialization using JSON, XML, etc.
|
|
|
|
package serial
|
2013-09-03 13:29:02 -06:00
|
|
|
|
|
|
|
// All 'pos' strings are of the form "file:line:col".
|
|
|
|
// TODO(adonovan): improve performance by sharing filename strings.
|
|
|
|
// TODO(adonovan): improve precision by providing the start/end
|
|
|
|
// interval when available.
|
|
|
|
//
|
|
|
|
// TODO(adonovan): consider richer encodings of types, functions,
|
|
|
|
// methods, etc.
|
|
|
|
|
|
|
|
// A Peers is the result of a 'peers' query.
|
|
|
|
// If Allocs is empty, the selected channel can't point to anything.
|
|
|
|
type Peers struct {
|
|
|
|
Pos string `json:"pos"` // location of the selected channel op (<-)
|
|
|
|
Type string `json:"type"` // type of the selected channel
|
|
|
|
Allocs []string `json:"allocs,omitempty"` // locations of aliased make(chan) ops
|
|
|
|
Sends []string `json:"sends,omitempty"` // locations of aliased ch<-x ops
|
|
|
|
Receives []string `json:"receives,omitempty"` // locations of aliased <-ch ops
|
2014-10-12 08:08:28 -06:00
|
|
|
Closes []string `json:"closes,omitempty"` // locations of aliased close(ch) ops
|
2013-09-03 13:29:02 -06:00
|
|
|
}
|
|
|
|
|
2013-09-10 12:11:42 -06:00
|
|
|
// A Referrers is the result of a 'referrers' query.
|
|
|
|
type Referrers struct {
|
|
|
|
Pos string `json:"pos"` // location of the query reference
|
|
|
|
ObjPos string `json:"objpos,omitempty"` // location of the definition
|
|
|
|
Desc string `json:"desc"` // description of the denoted object
|
|
|
|
Refs []string `json:"refs,omitempty"` // locations of all references
|
|
|
|
}
|
|
|
|
|
2013-12-13 08:04:55 -07:00
|
|
|
// A Definition is the result of a 'definition' query.
|
|
|
|
type Definition struct {
|
|
|
|
ObjPos string `json:"objpos,omitempty"` // location of the definition
|
|
|
|
Desc string `json:"desc"` // description of the denoted object
|
|
|
|
}
|
|
|
|
|
2013-09-03 13:29:02 -06:00
|
|
|
type CalleesItem struct {
|
|
|
|
Name string `json:"name"` // full name of called function
|
|
|
|
Pos string `json:"pos"` // location of called function
|
|
|
|
}
|
|
|
|
|
|
|
|
// A Callees is the result of a 'callees' query.
|
|
|
|
//
|
|
|
|
// Callees is nonempty unless the call was a dynamic call on a
|
|
|
|
// provably nil func or interface value.
|
|
|
|
type Callees struct {
|
|
|
|
Pos string `json:"pos"` // location of selected call site
|
|
|
|
Desc string `json:"desc"` // description of call site
|
|
|
|
Callees []*CalleesItem `json:"callees,omitempty"` // set of possible call targets
|
|
|
|
}
|
|
|
|
|
|
|
|
// A Caller is one element of the slice returned by a 'callers' query.
|
|
|
|
// (Callstack also contains a similar slice.)
|
|
|
|
//
|
|
|
|
// The root of the callgraph has an unspecified "Caller" string.
|
|
|
|
type Caller struct {
|
|
|
|
Pos string `json:"pos,omitempty"` // location of the calling function
|
|
|
|
Desc string `json:"desc"` // description of call site
|
|
|
|
Caller string `json:"caller"` // full name of calling function
|
|
|
|
}
|
|
|
|
|
|
|
|
// A CallStack is the result of a 'callstack' query.
|
|
|
|
// It indicates an arbitrary path from the root of the callgraph to
|
|
|
|
// the query function.
|
|
|
|
//
|
|
|
|
// If the Callers slice is empty, the function was unreachable in this
|
|
|
|
// analysis scope.
|
|
|
|
type CallStack struct {
|
|
|
|
Pos string `json:"pos"` // location of the selected function
|
|
|
|
Target string `json:"target"` // the selected function
|
|
|
|
Callers []Caller `json:"callers"` // enclosing calls, innermost first.
|
|
|
|
}
|
|
|
|
|
|
|
|
// A FreeVar is one element of the slice returned by a 'freevars'
|
|
|
|
// query. Each one identifies an expression referencing a local
|
|
|
|
// identifier defined outside the selected region.
|
|
|
|
type FreeVar struct {
|
|
|
|
Pos string `json:"pos"` // location of the identifier's definition
|
|
|
|
Kind string `json:"kind"` // one of {var,func,type,const,label}
|
|
|
|
Ref string `json:"ref"` // referring expression (e.g. "x" or "x.y.z")
|
|
|
|
Type string `json:"type"` // type of the expression
|
|
|
|
}
|
|
|
|
|
2013-12-13 16:00:55 -07:00
|
|
|
// An Implements contains the result of an 'implements' query.
|
|
|
|
// It describes the queried type, the set of named non-empty interface
|
|
|
|
// types to which it is assignable, and the set of named/*named types
|
|
|
|
// (concrete or non-empty interface) which may be assigned to it.
|
|
|
|
//
|
2013-09-03 13:29:02 -06:00
|
|
|
type Implements struct {
|
2013-12-13 16:00:55 -07:00
|
|
|
T ImplementsType `json:"type,omitempty"` // the queried type
|
|
|
|
AssignableTo []ImplementsType `json:"to,omitempty"` // types assignable to T
|
|
|
|
AssignableFrom []ImplementsType `json:"from,omitempty"` // interface types assignable from T
|
|
|
|
AssignableFromPtr []ImplementsType `json:"fromptr,omitempty"` // interface types assignable only from *T
|
2015-02-24 16:02:49 -07:00
|
|
|
|
|
|
|
// The following fields are set only if the query was a method.
|
|
|
|
// Assignable{To,From,FromPtr}Method[i] is the corresponding
|
|
|
|
// method of type Assignable{To,From,FromPtr}[i], or blank
|
|
|
|
// {"",""} if that type lacks the method.
|
|
|
|
Method *DescribeMethod `json:"method,omitempty"` // the queried method
|
|
|
|
AssignableToMethod []DescribeMethod `json:"to_method,omitempty"`
|
|
|
|
AssignableFromMethod []DescribeMethod `json:"from_method,omitempty"`
|
|
|
|
AssignableFromPtrMethod []DescribeMethod `json:"fromptr_method,omitempty"`
|
2013-12-13 16:00:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// An ImplementsType describes a single type as part of an 'implements' query.
|
|
|
|
type ImplementsType struct {
|
|
|
|
Name string `json:"name"` // full name of the type
|
|
|
|
Pos string `json:"pos"` // location of its definition
|
|
|
|
Kind string `json:"kind"` // "basic", "array", etc
|
2013-09-03 13:29:02 -06:00
|
|
|
}
|
|
|
|
|
2013-12-13 08:04:55 -07:00
|
|
|
// A SyntaxNode is one element of a stack of enclosing syntax nodes in
|
|
|
|
// a "what" query.
|
|
|
|
type SyntaxNode struct {
|
|
|
|
Description string `json:"desc"` // description of syntax tree
|
2014-11-13 10:34:25 -07:00
|
|
|
Start int `json:"start"` // start byte offset, 0-based
|
|
|
|
End int `json:"end"` // end byte offset
|
2013-12-13 08:04:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// A What is the result of the "what" query, which quickly identifies
|
|
|
|
// the selection, parsing only a single file. It is intended for use
|
|
|
|
// in low-latency GUIs.
|
|
|
|
type What struct {
|
|
|
|
Enclosing []SyntaxNode `json:"enclosing"` // enclosing nodes of syntax tree
|
|
|
|
Modes []string `json:"modes"` // query modes enabled for this selection.
|
|
|
|
SrcDir string `json:"srcdir,omitempty"` // $GOROOT src directory containing queried package
|
|
|
|
ImportPath string `json:"importpath,omitempty"` // import path of queried package
|
|
|
|
}
|
|
|
|
|
|
|
|
// A PointsToLabel describes a pointer analysis label.
|
2013-09-03 13:29:02 -06:00
|
|
|
//
|
|
|
|
// A "label" is an object that may be pointed to by a pointer, map,
|
|
|
|
// channel, 'func', slice or interface. Labels include:
|
|
|
|
// - functions
|
|
|
|
// - globals
|
|
|
|
// - arrays created by literals (e.g. []byte("foo")) and conversions ([]byte(s))
|
|
|
|
// - stack- and heap-allocated variables (including composite literals)
|
|
|
|
// - arrays allocated by append()
|
|
|
|
// - channels, maps and arrays created by make()
|
|
|
|
// - and their subelements, e.g. "alloc.y[*].z"
|
|
|
|
//
|
2013-12-13 08:04:55 -07:00
|
|
|
type PointsToLabel struct {
|
2013-09-03 13:29:02 -06:00
|
|
|
Pos string `json:"pos"` // location of syntax that allocated the object
|
|
|
|
Desc string `json:"desc"` // description of the label
|
|
|
|
}
|
|
|
|
|
2013-12-13 08:04:55 -07:00
|
|
|
// A PointsTo is one element of the result of a 'pointsto' query on an
|
|
|
|
// expression. It describes a single pointer: its type and the set of
|
|
|
|
// "labels" it points to.
|
2013-09-03 13:29:02 -06:00
|
|
|
//
|
2013-12-13 08:04:55 -07:00
|
|
|
// If the pointer is of interface type, it will have one PTS entry
|
2013-09-03 13:29:02 -06:00
|
|
|
// describing each concrete type that it may contain. For each
|
|
|
|
// concrete type that is a pointer, the PTS entry describes the labels
|
go.tools/pointer: reflection, part 1: maps, and some core features.
Core:
reflect.TypeOf
reflect.ValueOf
reflect.Zero
reflect.Value.Interface
Maps:
(reflect.Value).MapIndex
(reflect.Value).MapKeys
(reflect.Value).SetMapIndex
(*reflect.rtype).Elem
(*reflect.rtype).Key
+ tests:
pointer/testdata/mapreflect.go.
oracle/testdata/src/main/reflection.go.
Interface objects (T, V...) have been renamed "tagged objects".
Abstraction: we model reflect.Value similar to
interface{}---as a pointer that points only to tagged
objects---but a reflect.Value may also point to an "indirect
tagged object", one in which the payload V is of type *T not T.
These are required because reflect.Values can hold lvalues,
e.g. when derived via Field() or Elem(), though we won't use
them till we get to structs and pointers.
Solving: each reflection intrinsic defines a new constraint
and resolution rule. Because of the nature of reflection,
generalizing across types, the resolution rules dynamically
create additional complex constraints during solving, where
previously only simple (copy) constraints were created.
This requires some solver changes:
The work done before the main solver loop (to attach new
constraints to the graph) is now done before each iteration,
in processNewConstraints.
Its loop over constraints is broken into two passes:
the first handles base (addr-of) constraints,
the second handles simple and complex constraints.
constraint.init() has been inlined. The only behaviour that
varies across constraints is ptr()
Sadly this will pessimize presolver optimisations, when we get
there; such is the price of reflection.
Objects: reflection intrinsics create objects (i.e. cause
memory allocations) with no SSA operation. We will represent
them as the cgnode of the instrinsic (e.g. reflect.New), so we
extend Labels and node.data to represent objects as a product
(not sum) of ssa.Value and cgnode and pull this out into its
own type, struct object. This simplifies a number of
invariants and saves space. The ntObject flag is now
represented by obj!=nil; the other flags are moved into
object.
cgnodes are now always recorded in objects/Labels for which it
is appropriate (all but those for globals, constants and the
shared contours for functions).
Also:
- Prepopulate the flattenMemo cache to consider reflect.Value
a fake pointer, not a struct.
- Improve accessors and documentation on type Label.
- @conctypes assertions renamed @types (since dyn. types needn't be concrete).
- add oracle 'describe' test on an interface (missing, an oversight).
R=crawshaw
CC=golang-dev
https://golang.org/cl/13418048
2013-09-16 07:49:10 -06:00
|
|
|
// it may point to. The same is true for reflect.Values, except the
|
|
|
|
// dynamic types needn't be concrete.
|
2013-09-03 13:29:02 -06:00
|
|
|
//
|
2013-12-13 08:04:55 -07:00
|
|
|
type PointsTo struct {
|
|
|
|
Type string `json:"type"` // (concrete) type of the pointer
|
|
|
|
NamePos string `json:"namepos,omitempty"` // location of type defn, if Named
|
|
|
|
Labels []PointsToLabel `json:"labels,omitempty"` // pointed-to objects
|
|
|
|
}
|
|
|
|
|
|
|
|
// A DescribeValue is the additional result of a 'describe' query
|
|
|
|
// if the selection indicates a value or expression.
|
2013-09-03 13:29:02 -06:00
|
|
|
type DescribeValue struct {
|
2013-12-13 08:04:55 -07:00
|
|
|
Type string `json:"type"` // type of the expression
|
|
|
|
Value string `json:"value,omitempty"` // value of the expression, if constant
|
|
|
|
ObjPos string `json:"objpos,omitempty"` // location of the definition, if an Ident
|
2013-09-03 13:29:02 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
type DescribeMethod struct {
|
|
|
|
Name string `json:"name"` // method name, as defined by types.Selection.String()
|
|
|
|
Pos string `json:"pos"` // location of the method's definition
|
|
|
|
}
|
|
|
|
|
|
|
|
// A DescribeType is the additional result of a 'describe' query
|
|
|
|
// if the selection indicates a type.
|
|
|
|
type DescribeType struct {
|
|
|
|
Type string `json:"type"` // the string form of the type
|
|
|
|
NamePos string `json:"namepos,omitempty"` // location of definition of type, if named
|
|
|
|
NameDef string `json:"namedef,omitempty"` // underlying definition of type, if named
|
|
|
|
Methods []DescribeMethod `json:"methods,omitempty"` // methods of the type
|
|
|
|
}
|
|
|
|
|
|
|
|
type DescribeMember struct {
|
|
|
|
Name string `json:"name"` // name of member
|
|
|
|
Type string `json:"type,omitempty"` // type of member (underlying, if 'type')
|
|
|
|
Value string `json:"value,omitempty"` // value of member (if 'const')
|
|
|
|
Pos string `json:"pos"` // location of definition of member
|
|
|
|
Kind string `json:"kind"` // one of {var,const,func,type}
|
|
|
|
Methods []DescribeMethod `json:"methods,omitempty"` // methods (if member is a type)
|
|
|
|
}
|
|
|
|
|
|
|
|
// A DescribePackage is the additional result of a 'describe' if
|
|
|
|
// the selection indicates a package.
|
|
|
|
type DescribePackage struct {
|
|
|
|
Path string `json:"path"` // import path of the package
|
|
|
|
Members []*DescribeMember `json:"members,omitempty"` // accessible members of the package
|
|
|
|
}
|
|
|
|
|
|
|
|
// A Describe is the result of a 'describe' query.
|
|
|
|
// It may contain an element describing the selected semantic entity
|
|
|
|
// in detail.
|
|
|
|
type Describe struct {
|
|
|
|
Desc string `json:"desc"` // description of the selected syntax node
|
|
|
|
Pos string `json:"pos"` // location of the selected syntax node
|
|
|
|
Detail string `json:"detail,omitempty"` // one of {package, type, value}, or "".
|
|
|
|
|
|
|
|
// At most one of the following fields is populated:
|
|
|
|
// the one specified by 'detail'.
|
|
|
|
Package *DescribePackage `json:"package,omitempty"`
|
|
|
|
Type *DescribeType `json:"type,omitempty"`
|
|
|
|
Value *DescribeValue `json:"value,omitempty"`
|
|
|
|
}
|
|
|
|
|
2014-12-05 08:51:34 -07:00
|
|
|
// A WhichErrs is the result of a 'whicherrs' query.
|
|
|
|
// It contains the position of the queried error and the possible globals,
|
|
|
|
// constants, and types it may point to.
|
|
|
|
type WhichErrs struct {
|
|
|
|
ErrPos string `json:"errpos,omitempty"` // location of queried error
|
|
|
|
Globals []string `json:"globals,omitempty"` // locations of globals
|
|
|
|
Constants []string `json:"constants,omitempty"` // locations of constants
|
|
|
|
Types []WhichErrsType `json:"types,omitempty"` // Types
|
|
|
|
}
|
|
|
|
|
|
|
|
type WhichErrsType struct {
|
|
|
|
Type string `json:"type,omitempty"`
|
|
|
|
Position string `json:"position,omitempty"`
|
|
|
|
}
|
|
|
|
|
2013-09-03 13:29:02 -06:00
|
|
|
// A Result is the common result of any oracle query.
|
|
|
|
// It contains a query-specific result element.
|
|
|
|
//
|
2013-12-13 16:00:55 -07:00
|
|
|
// TODO(adonovan): perhaps include other info such as: analysis scope,
|
|
|
|
// raw query position, stack of ast nodes, query package, etc.
|
2013-09-03 13:29:02 -06:00
|
|
|
type Result struct {
|
|
|
|
Mode string `json:"mode"` // mode of the query
|
|
|
|
|
|
|
|
// Exactly one of the following fields is populated:
|
|
|
|
// the one specified by 'mode'.
|
2013-12-13 16:00:55 -07:00
|
|
|
Callees *Callees `json:"callees,omitempty"`
|
|
|
|
Callers []Caller `json:"callers,omitempty"`
|
|
|
|
Callstack *CallStack `json:"callstack,omitempty"`
|
|
|
|
Definition *Definition `json:"definition,omitempty"`
|
|
|
|
Describe *Describe `json:"describe,omitempty"`
|
|
|
|
Freevars []*FreeVar `json:"freevars,omitempty"`
|
|
|
|
Implements *Implements `json:"implements,omitempty"`
|
|
|
|
Peers *Peers `json:"peers,omitempty"`
|
|
|
|
PointsTo []PointsTo `json:"pointsto,omitempty"`
|
|
|
|
Referrers *Referrers `json:"referrers,omitempty"`
|
|
|
|
What *What `json:"what,omitempty"`
|
2014-12-05 08:51:34 -07:00
|
|
|
WhichErrs *WhichErrs `json:"whicherrs,omitempty"`
|
2013-09-03 13:29:02 -06:00
|
|
|
}
|