node-translator.h 13.2 KB
Newer Older
Kenton Varda's avatar
Kenton Varda committed
1 2
// Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors
// Licensed under the MIT License:
Kenton Varda's avatar
Kenton Varda committed
3
//
Kenton Varda's avatar
Kenton Varda committed
4 5 6 7 8 9
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
Kenton Varda's avatar
Kenton Varda committed
10
//
Kenton Varda's avatar
Kenton Varda committed
11 12
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
Kenton Varda's avatar
Kenton Varda committed
13
//
Kenton Varda's avatar
Kenton Varda committed
14 15 16 17 18 19 20
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
Kenton Varda's avatar
Kenton Varda committed
21 22 23 24 25 26

#ifndef CAPNP_COMPILER_NODE_TRANSLATOR_H_
#define CAPNP_COMPILER_NODE_TRANSLATOR_H_

#include <capnp/orphan.h>
#include <capnp/compiler/grammar.capnp.h>
Kenton Varda's avatar
Kenton Varda committed
27
#include <capnp/schema.capnp.h>
Kenton Varda's avatar
Kenton Varda committed
28 29
#include <capnp/dynamic.h>
#include <kj/vector.h>
Kenton Varda's avatar
Kenton Varda committed
30
#include <kj/one-of.h>
Kenton Varda's avatar
Kenton Varda committed
31 32 33 34 35 36 37 38 39 40 41 42 43
#include "error-reporter.h"

namespace capnp {
namespace compiler {

class NodeTranslator {
  // Translates one node in the schema from AST form to final schema form.  A "node" is anything
  // that has a unique ID, such as structs, enums, constants, and annotations, but not fields,
  // unions, enumerants, or methods (the latter set have 16-bit ordinals but not 64-bit global IDs).

public:
  class Resolver {
    // Callback class used to find other nodes relative to this one.
44 45 46 47
    //
    // TODO(cleanup): This has evolved into being a full interface for traversing the node tree.
    //   Maybe we should rename it as such, and move it out of NodeTranslator. See also
    //   TODO(cleanup) on NodeTranslator::BrandedDecl.
Kenton Varda's avatar
Kenton Varda committed
48 49

  public:
Kenton Varda's avatar
Kenton Varda committed
50
    struct ResolvedDecl {
Kenton Varda's avatar
Kenton Varda committed
51
      uint64_t id;
Kenton Varda's avatar
Kenton Varda committed
52 53
      uint genericParamCount;
      uint64_t scopeId;
54
      Declaration::Which kind;
Kenton Varda's avatar
Kenton Varda committed
55
      Resolver* resolver;
56 57 58 59 60 61

      kj::Maybe<schema::Brand::Reader> brand;
      // If present, then it is necessary to replace the brand scope with the given brand before
      // using the target type. This happens when the decl resolved to an alias; all other fields
      // of `ResolvedDecl` refer to the target of the alias, except for `scopeId` which is the
      // scope that contained the alias.
Kenton Varda's avatar
Kenton Varda committed
62 63
    };

Kenton Varda's avatar
Kenton Varda committed
64 65 66 67 68
    struct ResolvedParameter {
      uint64_t id;  // ID of the node declaring the parameter.
      uint index;   // Index of the parameter.
    };

69
    typedef kj::OneOf<ResolvedDecl, ResolvedParameter> ResolveResult;
Kenton Varda's avatar
Kenton Varda committed
70

71
    virtual kj::Maybe<ResolveResult> resolve(kj::StringPtr name) = 0;
Kenton Varda's avatar
Kenton Varda committed
72 73 74
    // Look up the given name, relative to this node, and return basic information about the
    // target.

75
    virtual kj::Maybe<ResolveResult> resolveMember(kj::StringPtr name) = 0;
Kenton Varda's avatar
Kenton Varda committed
76 77
    // Look up a member of this node.

78 79 80
    virtual ResolvedDecl resolveBuiltin(Declaration::Which which) = 0;
    virtual ResolvedDecl resolveId(uint64_t id) = 0;

81 82 83
    virtual kj::Maybe<ResolvedDecl> getParent() = 0;
    // Returns the parent of this scope, or null if this is the top scope.

Kenton Varda's avatar
Kenton Varda committed
84 85 86
    virtual ResolvedDecl getTopScope() = 0;
    // Get the top-level scope containing this node.

87
    virtual kj::Maybe<Schema> resolveBootstrapSchema(uint64_t id, schema::Brand::Reader brand) = 0;
88
    // Get the schema for the given ID.  If a schema is returned, it must be safe to traverse its
89 90
    // dependencies via the Schema API.  A schema that is only at the bootstrap stage is
    // acceptable.
91 92 93
    //
    // Throws an exception if the id is not one that was found by calling resolve() or by
    // traversing other schemas.  Returns null if the ID is recognized, but the corresponding
94
    // schema node failed to be built for reasons that were already reported.
Kenton Varda's avatar
Kenton Varda committed
95

96
    virtual kj::Maybe<schema::Node::Reader> resolveFinalSchema(uint64_t id) = 0;
97 98 99 100
    // Get the final schema for the given ID.  A bootstrap schema is not acceptable.  A raw
    // node reader is returned rather than a Schema object because using a Schema object built
    // by the final schema loader could trigger lazy initialization of dependencies which could
    // lead to a cycle and deadlock.
101 102 103 104
    //
    // Throws an exception if the id is not one that was found by calling resolve() or by
    // traversing other schemas.  Returns null if the ID is recognized, but the corresponding
    // schema node failed to be built for reasons that were already reported.
105

Kenton Varda's avatar
Kenton Varda committed
106
    virtual kj::Maybe<ResolvedDecl> resolveImport(kj::StringPtr name) = 0;
107
    // Get the ID of an imported file given the import path.
108 109 110 111 112 113 114

    virtual kj::Maybe<Type> resolveBootstrapType(schema::Type::Reader type, Schema scope) = 0;
    // Compile a schema::Type into a Type whose dependencies may safely be traversed via the schema
    // API. These dependencies may have only bootstrap schemas. Returns null if the type could not
    // be constructed due to already-reported errors.
    //
    // `scope` is the schema
Kenton Varda's avatar
Kenton Varda committed
115 116
  };

117
  NodeTranslator(Resolver& resolver, ErrorReporter& errorReporter,
Kenton Varda's avatar
Kenton Varda committed
118
                 const Declaration::Reader& decl, Orphan<schema::Node> wipNode,
119
                 bool compileAnnotations);
Kenton Varda's avatar
Kenton Varda committed
120 121 122 123
  // Construct a NodeTranslator to translate the given declaration.  The wipNode starts out with
  // `displayName`, `id`, `scopeId`, and `nestedNodes` already initialized.  The `NodeTranslator`
  // fills in the rest.

Kenton Varda's avatar
Kenton Varda committed
124
  ~NodeTranslator() noexcept(false);
125

Kenton Varda's avatar
Kenton Varda committed
126
  struct NodeSet {
Kenton Varda's avatar
Kenton Varda committed
127
    schema::Node::Reader node;
Kenton Varda's avatar
Kenton Varda committed
128 129
    // The main node.

Kenton Varda's avatar
Kenton Varda committed
130
    kj::Array<schema::Node::Reader> auxNodes;
Kenton Varda's avatar
Kenton Varda committed
131 132
    // Auxiliary nodes that were produced when translating this node and should be loaded along
    // with it.  In particular, structs that contain groups (or named unions) spawn extra nodes
133
    // representing those, and interfaces spawn struct nodes representing method params/results.
Kenton Varda's avatar
Kenton Varda committed
134 135 136
  };

  NodeSet getBootstrapNode();
Kenton Varda's avatar
Kenton Varda committed
137 138 139 140 141 142 143 144
  // Get an incomplete version of the node in which pointer-typed value expressions have not yet
  // been translated.  Instead, for all `schema.Value` objects representing pointer-type values,
  // the value is set to an appropriate "empty" value.  This version of the schema can be used to
  // bootstrap the dynamic API which can then in turn be used to encode the missing complex values.
  //
  // If the final node has already been built, this will actually return the final node (in fact,
  // it's the same node object).

Kenton Varda's avatar
Kenton Varda committed
145
  NodeSet finish();
Kenton Varda's avatar
Kenton Varda committed
146 147 148
  // Finish translating the node (including filling in all the pieces that are missing from the
  // bootstrap node) and return it.

149 150 151 152 153 154 155 156 157
  static kj::Maybe<Resolver::ResolveResult> compileDecl(
      uint64_t scopeId, uint scopeParameterCount, Resolver& resolver, ErrorReporter& errorReporter,
      Expression::Reader expression, schema::Brand::Builder brandBuilder);
  // Compile a one-off declaration expression without building a NodeTranslator. Used for
  // evaluating aliases.
  //
  // `brandBuilder` may be used to construct a message which will fill in ResolvedDecl::brand in
  // the result.

Kenton Varda's avatar
Kenton Varda committed
158
private:
159 160 161 162
  class DuplicateNameDetector;
  class DuplicateOrdinalDetector;
  class StructLayout;
  class StructTranslator;
163 164
  class BrandedDecl;
  class BrandScope;
165

166 167
  Resolver& resolver;
  ErrorReporter& errorReporter;
168
  Orphanage orphanage;
169
  bool compileAnnotations;
170
  kj::Own<BrandScope> localBrand;
Kenton Varda's avatar
Kenton Varda committed
171

Kenton Varda's avatar
Kenton Varda committed
172
  Orphan<schema::Node> wipNode;
Kenton Varda's avatar
Kenton Varda committed
173 174
  // The work-in-progress schema node.

Kenton Varda's avatar
Kenton Varda committed
175
  kj::Vector<Orphan<schema::Node>> groups;
Kenton Varda's avatar
Kenton Varda committed
176 177 178
  // If this is a struct node and it contains groups, these are the nodes for those groups,  which
  // must be loaded together with the top-level node.

179 180 181
  kj::Vector<Orphan<schema::Node>> paramStructs;
  // If this is an interface, these are the auto-generated structs representing params and results.

Kenton Varda's avatar
Kenton Varda committed
182
  struct UnfinishedValue {
Kenton Varda's avatar
Kenton Varda committed
183
    Expression::Reader source;
Kenton Varda's avatar
Kenton Varda committed
184
    schema::Type::Reader type;
185
    Schema typeScope;
Kenton Varda's avatar
Kenton Varda committed
186
    schema::Value::Builder target;
Kenton Varda's avatar
Kenton Varda committed
187 188 189 190 191 192 193
  };
  kj::Vector<UnfinishedValue> unfinishedValues;
  // List of values in `wipNode` which have not yet been interpreted, because they are structs
  // or lists and as such interpreting them require using the types' schemas (to take advantage
  // of the dynamic API).  Once bootstrap schemas have been built, they can be used to interpret
  // these values.

Kenton Varda's avatar
Kenton Varda committed
194
  void compileNode(Declaration::Reader decl, schema::Node::Builder builder);
Kenton Varda's avatar
Kenton Varda committed
195

Kenton Varda's avatar
Kenton Varda committed
196
  void compileConst(Declaration::Const::Reader decl, schema::Node::Const::Builder builder);
Kenton Varda's avatar
Kenton Varda committed
197
  void compileAnnotation(Declaration::Annotation::Reader decl,
Kenton Varda's avatar
Kenton Varda committed
198
                         schema::Node::Annotation::Builder builder);
Kenton Varda's avatar
Kenton Varda committed
199

200
  void compileEnum(Void decl, List<Declaration>::Reader members,
Kenton Varda's avatar
Kenton Varda committed
201
                   schema::Node::Builder builder);
202
  void compileStruct(Void decl, List<Declaration>::Reader members,
Kenton Varda's avatar
Kenton Varda committed
203
                     schema::Node::Builder builder);
204 205
  void compileInterface(Declaration::Interface::Reader decl,
                        List<Declaration>::Reader members,
Kenton Varda's avatar
Kenton Varda committed
206
                        schema::Node::Builder builder);
Kenton Varda's avatar
Kenton Varda committed
207 208 209
  // The `members` arrays contain only members with ordinal numbers, in code order.  Other members
  // are handled elsewhere.

210 211 212 213 214 215 216 217 218 219 220 221 222 223 224
  struct ImplicitParams {
    // Represents a set of implicit parameters visible in the current context.

    uint64_t scopeId;
    // If zero, then any reference to an implciit param in this context should be compiled to a
    // `implicitMethodParam` AnyPointer. If non-zero, it should be complied to a `parameter`
    // AnyPointer.

    List<Declaration::BrandParameter>::Reader params;
  };

  static inline ImplicitParams noImplicitParams() {
    return { 0, List<Declaration::BrandParameter>::Reader() };
  }

225
  template <typename InitBrandFunc>
226
  uint64_t compileParamList(kj::StringPtr methodName, uint16_t ordinal, bool isResults,
Kenton Varda's avatar
Kenton Varda committed
227
                            Declaration::ParamList::Reader paramList,
228
                            List<Declaration::BrandParameter>::Reader implicitParams,
229
                            InitBrandFunc&& initBrand);
230 231
  // Compile a param (or result) list and return the type ID of the struct type.

232 233
  kj::Maybe<BrandedDecl> compileDeclExpression(
      Expression::Reader source, ImplicitParams implicitMethodParams);
Kenton Varda's avatar
Kenton Varda committed
234 235
  // Compile an expression which is expected to resolve to a declaration or type expression.

236 237
  bool compileType(Expression::Reader source, schema::Type::Builder target,
                   ImplicitParams implicitMethodParams);
Kenton Varda's avatar
Kenton Varda committed
238 239 240
  // Returns false if there was a problem, in which case value expressions of this type should
  // not be parsed.

Kenton Varda's avatar
Kenton Varda committed
241
  void compileDefaultDefaultValue(schema::Type::Reader type, schema::Value::Builder target);
Kenton Varda's avatar
Kenton Varda committed
242 243
  // Initializes `target` to contain the "default default" value for `type`.

244 245 246
  void compileBootstrapValue(
      Expression::Reader source, schema::Type::Reader type, schema::Value::Builder target,
      Schema typeScope = Schema());
247 248
  // Calls compileValue() if this value should be interpreted at bootstrap time.  Otheriwse,
  // adds the value to `unfinishedValues` for later evaluation.
249 250 251 252 253 254
  //
  // If `type` comes from some other node, `typeScope` is the schema for that node. This is only
  // really needed for looking up generic parameter bindings, therefore if the type comes from
  // the node being built, an empty "Schema" (the default) works here because the node being built
  // is of course being built for all possible bindings and thus none of its generic parameters are
  // bound.
Kenton Varda's avatar
Kenton Varda committed
255

Kenton Varda's avatar
Kenton Varda committed
256
  void compileValue(Expression::Reader source, schema::Type::Reader type,
257
                    Schema typeScope, schema::Value::Builder target, bool isBootstrap);
258 259
  // Interprets the value expression and initializes `target` with the result.

Kenton Varda's avatar
Kenton Varda committed
260
  kj::Maybe<DynamicValue::Reader> readConstant(Expression::Reader name, bool isBootstrap);
261 262 263 264 265 266 267 268 269 270 271 272
  // Get the value of the given constant.  May return null if some error occurs, which will already
  // have been reported.

  Orphan<List<schema::Annotation>> compileAnnotationApplications(
      List<Declaration::AnnotationApplication>::Reader annotations,
      kj::StringPtr targetsFlagName);
};

class ValueTranslator {
public:
  class Resolver {
  public:
Kenton Varda's avatar
Kenton Varda committed
273
    virtual kj::Maybe<DynamicValue::Reader> resolveConstant(Expression::Reader name) = 0;
274 275
  };

276
  ValueTranslator(Resolver& resolver, ErrorReporter& errorReporter, Orphanage orphanage)
277 278
      : resolver(resolver), errorReporter(errorReporter), orphanage(orphanage) {}

279
  kj::Maybe<Orphan<DynamicValue>> compileValue(Expression::Reader src, Type type);
280 281 282

private:
  Resolver& resolver;
283
  ErrorReporter& errorReporter;
284
  Orphanage orphanage;
285

286
  Orphan<DynamicValue> compileValueInner(Expression::Reader src, Type type);
287
  // Helper for compileValue().
Kenton Varda's avatar
Kenton Varda committed
288

289
  void fillStructValue(DynamicStruct::Builder builder,
Kenton Varda's avatar
Kenton Varda committed
290
                       List<Expression::Param>::Reader assignments);
291 292
  // Interprets the given assignments and uses them to fill in the given struct builder.

293 294
  kj::String makeNodeName(Schema node);
  kj::String makeTypeName(Type type);
295

Kenton Varda's avatar
Kenton Varda committed
296
  kj::Maybe<ListSchema> makeListSchemaOf(schema::Type::Reader elementType);
Kenton Varda's avatar
Kenton Varda committed
297 298 299 300 301 302
};

}  // namespace compiler
}  // namespace capnp

#endif  // CAPNP_COMPILER_NODE_TRANSLATOR_H_