Commit 2e17fd43 authored by Kenton Varda's avatar Kenton Varda

Interface code generation. Also update method grammar to allow multiple…

Interface code generation.  Also update method grammar to allow multiple returns, and allow specifying request/response structs instead of parameter lists.
parent d8c5b805
...@@ -106,7 +106,7 @@ namespace { ...@@ -106,7 +106,7 @@ namespace {
class DummyClientHook final: public ClientHook { class DummyClientHook final: public ClientHook {
public: public:
Request<ObjectPointer, TypelessAnswer> newCall( Request<ObjectPointer, TypelessResults> newCall(
uint64_t interfaceId, uint16_t methodId) const override { uint64_t interfaceId, uint16_t methodId) const override {
KJ_FAIL_REQUIRE("Calling capability that was extracted from a message that had no " KJ_FAIL_REQUIRE("Calling capability that was extracted from a message that had no "
"capability context."); "capability context.");
......
...@@ -22,10 +22,35 @@ ...@@ -22,10 +22,35 @@
// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include "capability.h" #include "capability.h"
#include <kj/debug.h>
namespace capnp { namespace capnp {
TypelessAnswer::Pipeline TypelessAnswer::Pipeline::getPointerField( kj::Promise<void> Capability::Server::internalUnimplemented(
const char* actualInterfaceName, uint64_t requestedTypeId) {
KJ_FAIL_REQUIRE("Requested interface not implemented.", actualInterfaceName, requestedTypeId) {
// Recoverable exception will be caught by promise framework.
return kj::READY_NOW;
}
}
kj::Promise<void> Capability::Server::internalUnimplemented(
const char* interfaceName, uint64_t typeId, uint16_t methodId) {
KJ_FAIL_REQUIRE("Method not implemented.", interfaceName, typeId, methodId) {
// Recoverable exception will be caught by promise framework.
return kj::READY_NOW;
}
}
kj::Promise<void> Capability::Server::internalUnimplemented(
const char* interfaceName, const char* methodName, uint64_t typeId, uint16_t methodId) {
KJ_FAIL_REQUIRE("Method not implemented.", interfaceName, typeId, methodName, methodId) {
// Recoverable exception will be caught by promise framework.
return kj::READY_NOW;
}
}
TypelessResults::Pipeline TypelessResults::Pipeline::getPointerField(
uint16_t pointerIndex) const { uint16_t pointerIndex) const {
auto newOps = kj::heapArray<PipelineOp>(ops.size() + 1); auto newOps = kj::heapArray<PipelineOp>(ops.size() + 1);
for (auto i: kj::indices(ops)) { for (auto i: kj::indices(ops)) {
......
This diff is collapsed.
This diff is collapsed.
...@@ -175,11 +175,11 @@ private: ...@@ -175,11 +175,11 @@ private:
case schema::Type::LIST: case schema::Type::LIST:
return kj::strTree("List(", genType(type.getList().getElementType(), scope), ")"); return kj::strTree("List(", genType(type.getList().getElementType(), scope), ")");
case schema::Type::ENUM: case schema::Type::ENUM:
return nodeName(scope.getDependency(type.getEnum().getTypeId()), scope); return nodeName(schemaLoader.get(type.getEnum().getTypeId()), scope);
case schema::Type::STRUCT: case schema::Type::STRUCT:
return nodeName(scope.getDependency(type.getStruct().getTypeId()), scope); return nodeName(schemaLoader.get(type.getStruct().getTypeId()), scope);
case schema::Type::INTERFACE: case schema::Type::INTERFACE:
return nodeName(scope.getDependency(type.getInterface().getTypeId()), scope); return nodeName(schemaLoader.get(type.getInterface().getTypeId()), scope);
case schema::Type::OBJECT: return kj::strTree("Object"); case schema::Type::OBJECT: return kj::strTree("Object");
} }
return kj::strTree(); return kj::strTree();
...@@ -262,7 +262,7 @@ private: ...@@ -262,7 +262,7 @@ private:
} }
case schema::Value::ENUM: { case schema::Value::ENUM: {
KJ_REQUIRE(type.isEnum(), "type/value mismatch"); KJ_REQUIRE(type.isEnum(), "type/value mismatch");
auto enumNode = scope.getDependency(type.getEnum().getTypeId()).asEnum().getProto(); auto enumNode = schemaLoader.get(type.getEnum().getTypeId()).asEnum().getProto();
auto enumerants = enumNode.getEnum().getEnumerants(); auto enumerants = enumNode.getEnum().getEnumerants();
KJ_REQUIRE(value.getEnum() < enumerants.size(), KJ_REQUIRE(value.getEnum() < enumerants.size(),
"Enum value out-of-range.", value.getEnum(), enumNode.getDisplayName()); "Enum value out-of-range.", value.getEnum(), enumNode.getDisplayName());
...@@ -271,7 +271,7 @@ private: ...@@ -271,7 +271,7 @@ private:
case schema::Value::STRUCT: { case schema::Value::STRUCT: {
KJ_REQUIRE(type.isStruct(), "type/value mismatch"); KJ_REQUIRE(type.isStruct(), "type/value mismatch");
auto structValue = value.getStruct().getAs<DynamicStruct>( auto structValue = value.getStruct().getAs<DynamicStruct>(
scope.getDependency(type.getStruct().getTypeId()).asStruct()); schemaLoader.get(type.getStruct().getTypeId()).asStruct());
return kj::strTree(structValue); return kj::strTree(structValue);
} }
case schema::Value::INTERFACE: { case schema::Value::INTERFACE: {
...@@ -386,7 +386,7 @@ private: ...@@ -386,7 +386,7 @@ private:
"\n"); "\n");
} }
case schema::Field::GROUP: { case schema::Field::GROUP: {
auto group = scope.getDependency(field.getGroup().getTypeId()).asStruct(); auto group = schemaLoader.get(field.getGroup().getTypeId()).asStruct();
return kj::strTree( return kj::strTree(
indent, field.getName(), indent, field.getName(),
" :group", genAnnotations(field.getAnnotations(), scope), " {", " :group", genAnnotations(field.getAnnotations(), scope), " {",
...@@ -400,6 +400,25 @@ private: ...@@ -400,6 +400,25 @@ private:
return kj::strTree(); return kj::strTree();
} }
kj::StringTree genParamList(InterfaceSchema interface, StructSchema schema) {
if (schema.getProto().getScopeId() == 0) {
// A named parameter list.
return kj::strTree("(", kj::StringTree(
KJ_MAP(field, schema.getFields()) {
auto proto = field.getProto();
auto slot = proto.getSlot();
return kj::strTree(
proto.getName(), " :", genType(slot.getType(), interface),
isEmptyValue(slot.getDefaultValue()) ? kj::strTree("") :
kj::strTree(" = ", genValue(
slot.getType(), slot.getDefaultValue(), interface)));
}, ", "), ")");
} else {
return nodeName(schema, interface);
}
}
kj::StringTree genDecl(Schema schema, Text::Reader name, uint64_t scopeId, Indent indent) { kj::StringTree genDecl(Schema schema, Text::Reader name, uint64_t scopeId, Indent indent) {
auto proto = schema.getProto(); auto proto = schema.getProto();
if (proto.getScopeId() != scopeId) { if (proto.getScopeId() != scopeId) {
...@@ -439,28 +458,17 @@ private: ...@@ -439,28 +458,17 @@ private:
indent, "}\n"); indent, "}\n");
} }
case schema::Node::INTERFACE: { case schema::Node::INTERFACE: {
auto interface = schema.asInterface();
return kj::strTree( return kj::strTree(
indent, "interface ", name, " @0x", kj::hex(proto.getId()), indent, "interface ", name, " @0x", kj::hex(proto.getId()),
genAnnotations(schema), " {\n", genAnnotations(schema), " {\n",
KJ_MAP(method, sortByCodeOrder(schema.asInterface().getMethods())) { KJ_MAP(method, sortByCodeOrder(interface.getMethods())) {
int i = 0;
auto methodProto = method.getProto(); auto methodProto = method.getProto();
auto params = schemaLoader.get(methodProto.getParamStructType()).asStruct();
auto results = schemaLoader.get(methodProto.getResultStructType()).asStruct();
return kj::strTree( return kj::strTree(
indent.next(), methodProto.getName(), " @", method.getIndex(), "(", indent.next(), methodProto.getName(), " @", method.getIndex(), " ",
KJ_MAP(param, methodProto.getParams()) { genParamList(interface, params), " -> ", genParamList(interface, results), ";\n");
bool hasDefault = i >= methodProto.getRequiredParamCount() ||
!isEmptyValue(param.getDefaultValue());
return kj::strTree(
i++ > 0 ? ", " : "",
param.getName(), ": ", genType(param.getType(), schema),
hasDefault
? kj::strTree(" = ", genValue(
param.getType(), param.getDefaultValue(), schema))
: kj::strTree(),
genAnnotations(param.getAnnotations(), schema));
},
") :", genType(methodProto.getReturnType(), schema),
genAnnotations(methodProto.getAnnotations(), schema), ";\n");
}, },
genNestedDecls(schema, indent.next()), genNestedDecls(schema, indent.next()),
indent, "}\n"); indent, "}\n");
......
...@@ -199,7 +199,8 @@ private: ...@@ -199,7 +199,8 @@ private:
void traverseAnnotations(const List<schema::Annotation>::Reader& annotations, uint eagerness, void traverseAnnotations(const List<schema::Annotation>::Reader& annotations, uint eagerness,
std::unordered_map<const Node*, uint>& seen) const; std::unordered_map<const Node*, uint>& seen) const;
void traverseDependency(uint64_t depId, uint eagerness, void traverseDependency(uint64_t depId, uint eagerness,
std::unordered_map<const Node*, uint>& seen) const; std::unordered_map<const Node*, uint>& seen,
bool ignoreIfNotFound = false) const;
// Helpers for traverse(). // Helpers for traverse().
}; };
...@@ -730,11 +731,8 @@ void Compiler::Node::traverseNodeDependencies( ...@@ -730,11 +731,8 @@ void Compiler::Node::traverseNodeDependencies(
} }
} }
for (auto method: interface.getMethods()) { for (auto method: interface.getMethods()) {
for (auto param: method.getParams()) { traverseDependency(method.getParamStructType(), eagerness, seen, true);
traverseType(param.getType(), eagerness, seen); traverseDependency(method.getResultStructType(), eagerness, seen, true);
traverseAnnotations(param.getAnnotations(), eagerness, seen);
}
traverseType(method.getReturnType(), eagerness, seen);
traverseAnnotations(method.getAnnotations(), eagerness, seen); traverseAnnotations(method.getAnnotations(), eagerness, seen);
} }
break; break;
...@@ -771,10 +769,11 @@ void Compiler::Node::traverseType(const schema::Type::Reader& type, uint eagerne ...@@ -771,10 +769,11 @@ void Compiler::Node::traverseType(const schema::Type::Reader& type, uint eagerne
} }
void Compiler::Node::traverseDependency(uint64_t depId, uint eagerness, void Compiler::Node::traverseDependency(uint64_t depId, uint eagerness,
std::unordered_map<const Node*, uint>& seen) const { std::unordered_map<const Node*, uint>& seen,
bool ignoreIfNotFound) const {
KJ_IF_MAYBE(node, module->getCompiler().findNode(depId)) { KJ_IF_MAYBE(node, module->getCompiler().findNode(depId)) {
node->traverse(eagerness, seen); node->traverse(eagerness, seen);
} else { } else if (!ignoreIfNotFound) {
KJ_FAIL_ASSERT("Dependency ID not present in compiler?", depId); KJ_FAIL_ASSERT("Dependency ID not present in compiler?", depId);
} }
} }
...@@ -872,14 +871,31 @@ static void findImports(Declaration::Reader decl, std::set<kj::StringPtr>& outpu ...@@ -872,14 +871,31 @@ static void findImports(Declaration::Reader decl, std::set<kj::StringPtr>& outpu
break; break;
case Declaration::METHOD: { case Declaration::METHOD: {
auto method = decl.getMethod(); auto method = decl.getMethod();
for (auto param: method.getParams()) {
auto params = method.getParams();
if (params.isNamedList()) {
for (auto param: params.getNamedList()) {
findImports(param.getType(), output); findImports(param.getType(), output);
for (auto ann: param.getAnnotations()) { for (auto ann: param.getAnnotations()) {
findImports(ann.getName(), output); findImports(ann.getName(), output);
} }
} }
if (method.getReturnType().isExpression()) { } else {
findImports(method.getReturnType().getExpression(), output); findImports(params.getType(), output);
}
if (method.getResults().isExplicit()) {
auto results = method.getResults().getExplicit();
if (results.isNamedList()) {
for (auto param: results.getNamedList()) {
findImports(param.getType(), output);
for (auto ann: param.getAnnotations()) {
findImports(ann.getName(), output);
}
}
} else {
findImports(results.getType(), output);
}
} }
break; break;
} }
......
...@@ -170,10 +170,10 @@ struct Declaration { ...@@ -170,10 +170,10 @@ struct Declaration {
extends @21 :List(DeclName); extends @21 :List(DeclName);
} }
method :group { method :group {
params @22 :List(Param); params @22 :ParamList;
returnType :union { results :union {
none @23 :Void; none @23 :Void;
expression @24 :TypeExpression; explicit @24 :ParamList;
} }
} }
...@@ -221,6 +221,19 @@ struct Declaration { ...@@ -221,6 +221,19 @@ struct Declaration {
builtinObject @55 :Void; builtinObject @55 :Void;
} }
struct ParamList {
# A list of method parameters or method returns.
union {
namedList @0 :List(Param);
type @1 :DeclName;
# Specified some other struct type instead of a named list.
}
startByte @2 :UInt32;
endByte @3 :UInt32;
}
struct Param { struct Param {
name @0 :LocatedText; # If null, param failed to parse. name @0 :LocatedText; # If null, param failed to parse.
type @1 :TypeExpression; type @1 :TypeExpression;
...@@ -229,6 +242,9 @@ struct Declaration { ...@@ -229,6 +242,9 @@ struct Declaration {
none @3 :Void; none @3 :Void;
value @4 :ValueExpression; value @4 :ValueExpression;
} }
startByte @5 :UInt32;
endByte @6 :UInt32;
} }
} }
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -90,7 +90,7 @@ public: ...@@ -90,7 +90,7 @@ public:
kj::Array<schema::Node::Reader> auxNodes; kj::Array<schema::Node::Reader> auxNodes;
// Auxiliary nodes that were produced when translating this node and should be loaded along // 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 // with it. In particular, structs that contain groups (or named unions) spawn extra nodes
// representing those. // representing those, and interfaces spawn struct nodes representing method params/results.
}; };
NodeSet getBootstrapNode(); NodeSet getBootstrapNode();
...@@ -119,6 +119,9 @@ private: ...@@ -119,6 +119,9 @@ private:
// If this is a struct node and it contains groups, these are the nodes for those groups, which // 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. // must be loaded together with the top-level node.
kj::Vector<Orphan<schema::Node>> paramStructs;
// If this is an interface, these are the auto-generated structs representing params and results.
struct UnfinishedValue { struct UnfinishedValue {
ValueExpression::Reader source; ValueExpression::Reader source;
schema::Type::Reader type; schema::Type::Reader type;
...@@ -151,6 +154,10 @@ private: ...@@ -151,6 +154,10 @@ private:
// The `members` arrays contain only members with ordinal numbers, in code order. Other members // The `members` arrays contain only members with ordinal numbers, in code order. Other members
// are handled elsewhere. // are handled elsewhere.
uint64_t compileParamList(kj::StringPtr methodName, uint16_t ordinal, bool isResults,
Declaration::ParamList::Reader paramList);
// Compile a param (or result) list and return the type ID of the struct type.
bool compileType(TypeExpression::Reader source, schema::Type::Builder target); bool compileType(TypeExpression::Reader source, schema::Type::Builder target);
// Returns false if there was a problem, in which case value expressions of this type should // Returns false if there was a problem, in which case value expressions of this type should
// not be parsed. // not be parsed.
......
...@@ -95,6 +95,33 @@ uint64_t generateGroupId(uint64_t parentId, uint16_t groupIndex) { ...@@ -95,6 +95,33 @@ uint64_t generateGroupId(uint64_t parentId, uint16_t groupIndex) {
return result | (1ull << 63); return result | (1ull << 63);
} }
uint64_t generateMethodParamsId(uint64_t parentId, uint16_t methodOrdinal, bool isResults) {
// Compute ID by MD5 hashing the concatenation of the parent ID, the method ordinal, and a
// boolean indicating whether this is the params or the results, and then taking the first 8
// bytes.
kj::byte bytes[sizeof(uint64_t) + sizeof(uint16_t) + 1];
for (uint i = 0; i < sizeof(uint64_t); i++) {
bytes[i] = (parentId >> (i * 8)) & 0xff;
}
for (uint i = 0; i < sizeof(uint16_t); i++) {
bytes[sizeof(uint64_t) + i] = (methodOrdinal >> (i * 8)) & 0xff;
}
bytes[sizeof(bytes) - 1] = isResults;
Md5 md5;
md5.update(bytes);
kj::ArrayPtr<const kj::byte> resultBytes = md5.finish();
uint64_t result = 0;
for (uint i = 0; i < sizeof(uint64_t); i++) {
result = (result << 8) | resultBytes[i];
}
return result | (1ull << 63);
}
void parseFile(List<Statement>::Reader statements, ParsedFile::Builder result, void parseFile(List<Statement>::Reader statements, ParsedFile::Builder result,
const ErrorReporter& errorReporter) { const ErrorReporter& errorReporter) {
CapnpParser parser(Orphanage::getForMessageContaining(result), errorReporter); CapnpParser parser(Orphanage::getForMessageContaining(result), errorReporter);
...@@ -813,17 +840,20 @@ CapnpParser::CapnpParser(Orphanage orphanageParam, const ErrorReporter& errorRep ...@@ -813,17 +840,20 @@ CapnpParser::CapnpParser(Orphanage orphanageParam, const ErrorReporter& errorRep
return DeclParserResult(kj::mv(decl), parsers.interfaceLevelDecl); return DeclParserResult(kj::mv(decl), parsers.interfaceLevelDecl);
})); }));
parsers.param = arena.copy(p::transform( parsers.param = arena.copy(p::transformWithLocation(
p::sequence(identifier, op(":"), parsers.typeExpression, p::sequence(identifier, op(":"), parsers.typeExpression,
p::optional(p::sequence(op("="), parsers.valueExpression)), p::optional(p::sequence(op("="), parsers.valueExpression)),
p::many(parsers.annotation)), p::many(parsers.annotation)),
[this](Located<Text::Reader>&& name, Orphan<TypeExpression>&& type, [this](kj::parse::Span<List<Token>::Reader::Iterator> location,
Located<Text::Reader>&& name, Orphan<TypeExpression>&& type,
kj::Maybe<Orphan<ValueExpression>>&& defaultValue, kj::Maybe<Orphan<ValueExpression>>&& defaultValue,
kj::Array<Orphan<Declaration::AnnotationApplication>>&& annotations) kj::Array<Orphan<Declaration::AnnotationApplication>>&& annotations)
-> Orphan<Declaration::Param> { -> Orphan<Declaration::Param> {
auto result = orphanage.newOrphan<Declaration::Param>(); auto result = orphanage.newOrphan<Declaration::Param>();
auto builder = result.get(); auto builder = result.get();
initLocation(location, builder);
name.copyTo(builder.initName()); name.copyTo(builder.initName());
builder.adoptType(kj::mv(type)); builder.adoptType(kj::mv(type));
builder.adoptAnnotations(arrayToList(orphanage, kj::mv(annotations))); builder.adoptAnnotations(arrayToList(orphanage, kj::mv(annotations)));
...@@ -836,14 +866,39 @@ CapnpParser::CapnpParser(Orphanage orphanageParam, const ErrorReporter& errorRep ...@@ -836,14 +866,39 @@ CapnpParser::CapnpParser(Orphanage orphanageParam, const ErrorReporter& errorRep
return kj::mv(result); return kj::mv(result);
})); }));
auto& paramList = arena.copy(p::oneOf(
p::transform(parenthesizedList(parsers.param, errorReporter),
[this](Located<kj::Array<kj::Maybe<Orphan<Declaration::Param>>>>&& params)
-> Orphan<Declaration::ParamList> {
auto decl = orphanage.newOrphan<Declaration::ParamList>();
auto builder = decl.get();
params.copyLocationTo(builder);
auto listBuilder = builder.initNamedList(params.value.size());
for (uint i: kj::indices(params.value)) {
KJ_IF_MAYBE(param, params.value[i]) {
listBuilder.adoptWithCaveats(i, kj::mv(*param));
}
}
return decl;
}),
p::transform(parsers.declName,
[this](Orphan<DeclName>&& name) -> Orphan<Declaration::ParamList> {
auto decl = orphanage.newOrphan<Declaration::ParamList>();
auto builder = decl.get();
auto nameReader = name.getReader();
builder.setStartByte(nameReader.getStartByte());
builder.setEndByte(nameReader.getEndByte());
builder.adoptType(kj::mv(name));
return decl;
})));
parsers.methodDecl = arena.copy(p::transform( parsers.methodDecl = arena.copy(p::transform(
p::sequence(identifier, parsers.ordinal, p::sequence(identifier, parsers.ordinal, paramList,
parenthesizedList(parsers.param, errorReporter), p::optional(p::sequence(op("->"), paramList)),
p::optional(p::sequence(op(":"), parsers.typeExpression)),
p::many(parsers.annotation)), p::many(parsers.annotation)),
[this](Located<Text::Reader>&& name, Orphan<LocatedInteger>&& ordinal, [this](Located<Text::Reader>&& name, Orphan<LocatedInteger>&& ordinal,
Located<kj::Array<kj::Maybe<Orphan<Declaration::Param>>>>&& params, Orphan<Declaration::ParamList>&& params,
kj::Maybe<Orphan<TypeExpression>>&& returnType, kj::Maybe<Orphan<Declaration::ParamList>>&& results,
kj::Array<Orphan<Declaration::AnnotationApplication>>&& annotations) kj::Array<Orphan<Declaration::AnnotationApplication>>&& annotations)
-> DeclParserResult { -> DeclParserResult {
auto decl = orphanage.newOrphan<Declaration>(); auto decl = orphanage.newOrphan<Declaration>();
...@@ -851,18 +906,14 @@ CapnpParser::CapnpParser(Orphanage orphanageParam, const ErrorReporter& errorRep ...@@ -851,18 +906,14 @@ CapnpParser::CapnpParser(Orphanage orphanageParam, const ErrorReporter& errorRep
initMemberDecl(decl.get(), kj::mv(name), kj::mv(ordinal), kj::mv(annotations)) initMemberDecl(decl.get(), kj::mv(name), kj::mv(ordinal), kj::mv(annotations))
.initMethod(); .initMethod();
auto paramsBuilder = builder.initParams(params.value.size()); builder.adoptParams(kj::mv(params));
for (uint i = 0; i < params.value.size(); i++) {
KJ_IF_MAYBE(param, params.value[i]) {
paramsBuilder.adoptWithCaveats(i, kj::mv(*param));
}
}
KJ_IF_MAYBE(t, returnType) { KJ_IF_MAYBE(r, results) {
builder.getReturnType().adoptExpression(kj::mv(*t)); builder.getResults().adoptExplicit(kj::mv(*r));
} else { } else {
builder.getReturnType().setNone(); builder.getResults().setNone();
} }
return DeclParserResult(kj::mv(decl)); return DeclParserResult(kj::mv(decl));
})); }));
......
...@@ -48,6 +48,9 @@ uint64_t generateChildId(uint64_t parentId, kj::StringPtr childName); ...@@ -48,6 +48,9 @@ uint64_t generateChildId(uint64_t parentId, kj::StringPtr childName);
uint64_t generateGroupId(uint64_t parentId, uint16_t groupIndex); uint64_t generateGroupId(uint64_t parentId, uint16_t groupIndex);
// Generate the ID for a group within a struct. // Generate the ID for a group within a struct.
uint64_t generateMethodParamsId(uint64_t parentId, uint16_t methodOrdinal, bool isResults);
// Generate the ID for a struct representing method params / results.
// //
// TODO(cleanup): Move generate*Id() somewhere more sensible. // TODO(cleanup): Move generate*Id() somewhere more sensible.
......
...@@ -30,6 +30,8 @@ ...@@ -30,6 +30,8 @@
#include <kj/debug.h> #include <kj/debug.h>
#include <kj/exception.h> #include <kj/exception.h>
#include <kj/arena.h> #include <kj/arena.h>
#include <kj/vector.h>
#include <algorithm>
namespace capnp { namespace capnp {
...@@ -386,17 +388,8 @@ private: ...@@ -386,17 +388,8 @@ private:
"invalid codeOrder"); "invalid codeOrder");
sawCodeOrder[method.getCodeOrder()] = true; sawCodeOrder[method.getCodeOrder()] = true;
auto params = method.getParams(); validateTypeId(method.getParamStructType(), schema::Node::STRUCT);
for (auto param: params) { validateTypeId(method.getResultStructType(), schema::Node::STRUCT);
KJ_CONTEXT("validating parameter", param.getName());
uint dummy1;
bool dummy2;
validate(param.getType(), param.getDefaultValue(), &dummy1, &dummy2);
}
VALIDATE_SCHEMA(method.getRequiredParamCount() <= params.size(),
"invalid requiredParamCount");
validate(method.getReturnType());
} }
} }
...@@ -754,6 +747,43 @@ private: ...@@ -754,6 +747,43 @@ private:
void checkCompatibility(const schema::Node::Interface::Reader& interfaceNode, void checkCompatibility(const schema::Node::Interface::Reader& interfaceNode,
const schema::Node::Interface::Reader& replacement) { const schema::Node::Interface::Reader& replacement) {
{
// Check superclasses.
kj::Vector<uint64_t> extends;
kj::Vector<uint64_t> replacementExtends;
for (uint64_t extend: interfaceNode.getExtends()) {
extends.add(extend);
}
for (uint64_t extend: replacement.getExtends()) {
replacementExtends.add(extend);
}
std::sort(extends.begin(), extends.end());
std::sort(replacementExtends.begin(), replacementExtends.end());
auto iter = extends.begin();
auto replacementIter = replacementExtends.begin();
while (iter != extends.end() || replacementIter != replacementExtends.end()) {
if (iter == extends.end()) {
replacementIsNewer();
break;
} else if (replacementIter == replacementExtends.end()) {
replacementIsOlder();
break;
} else if (*iter < *replacementIter) {
replacementIsOlder();
++iter;
} else if (*iter > *replacementIter) {
replacementIsNewer();
++replacementIter;
} else {
++iter;
++replacementIter;
}
}
}
auto methods = interfaceNode.getMethods(); auto methods = interfaceNode.getMethods();
auto replacementMethods = replacement.getMethods(); auto replacementMethods = replacement.getMethods();
...@@ -774,40 +804,11 @@ private: ...@@ -774,40 +804,11 @@ private:
const schema::Method::Reader& replacement) { const schema::Method::Reader& replacement) {
KJ_CONTEXT("comparing method", method.getName()); KJ_CONTEXT("comparing method", method.getName());
auto params = method.getParams(); // TODO(someday): Allow named parameter list to be replaced by compatible struct type.
auto replacementParams = replacement.getParams(); VALIDATE_SCHEMA(method.getParamStructType() == replacement.getParamStructType(),
"Updated method has different parameters.");
if (replacementParams.size() > params.size()) { VALIDATE_SCHEMA(method.getResultStructType() == replacement.getResultStructType(),
replacementIsNewer(); "Updated method has different results.");
} else if (replacementParams.size() < params.size()) {
replacementIsOlder();
}
uint count = std::min(params.size(), replacementParams.size());
for (uint i = 0; i < count; i++) {
auto param = params[i];
auto replacementParam = replacementParams[i];
KJ_CONTEXT("comparing parameter", param.getName());
checkCompatibility(param.getType(), replacementParam.getType(),
NO_UPGRADE_TO_STRUCT);
checkDefaultCompatibility(param.getDefaultValue(), replacementParam.getDefaultValue());
}
// Before checking that the required parameter counts are equal, check if the user added new
// parameters without defaulting them, as this is the most common reason for this error and we
// can provide a nicer error message.
VALIDATE_SCHEMA(replacement.getRequiredParamCount() <= count &&
method.getRequiredParamCount() <= count,
"Updated method signature contains additional parameters that lack default values");
VALIDATE_SCHEMA(replacement.getRequiredParamCount() == method.getRequiredParamCount(),
"Updated method signature has different number of required parameters (parameters without "
"default values)");
checkCompatibility(method.getReturnType(), replacement.getReturnType(),
ALLOW_UPGRADE_TO_STRUCT);
} }
void checkCompatibility(const schema::Node::Const::Reader& constNode, void checkCompatibility(const schema::Node::Const::Reader& constNode,
......
...@@ -191,6 +191,12 @@ struct Field { ...@@ -191,6 +191,12 @@ struct Field {
type @5 :Type; type @5 :Type;
defaultValue @6 :Value; defaultValue @6 :Value;
hadExplicitDefault @10 :Bool;
# Whether the default value was specified explicitly. Non-explicit default values are always
# zero or empty values. Usually, whether the default value was explicit shouldn't matter.
# The main use case for this flag is for structs representing method parameters:
# explicitly-defaulted parameters may be allowed to be omitted when calling the method.
} }
group :group { group :group {
...@@ -232,22 +238,16 @@ struct Method { ...@@ -232,22 +238,16 @@ struct Method {
# Specifies order in which the methods were declared in the code. # Specifies order in which the methods were declared in the code.
# Like Struct.Field.codeOrder. # Like Struct.Field.codeOrder.
params @2 :List(Param); paramStructType @2 :Id;
struct Param { # ID of the parameter struct type. If a named parameter list was specified in the method
name @0 :Text; # declaration (rather than a single struct parameter type) then a corresponding struct type is
type @1 :Type; # auto-generated. Such an auto-generated type will not be listed in the interface's
defaultValue @2 :Value; # `nestedNodes` and its `scopeId` will be zero -- it is completely detached from the namespace.
annotations @3 :List(Annotation);
}
requiredParamCount @3 :UInt16; resultStructType @3 :Id;
# One plus the index of the last parameter that has no default value. In languages where # ID of the return struct type; similar to `paramStructType`.
# method calls look like function calls, this is the minimum number of parameters that must
# always be specified, while subsequent parameters are optional.
returnType @4 :Type; annotations @4 :List(Annotation);
annotations @5 :List(Annotation);
} }
struct Type { struct Type {
......
This diff is collapsed.
This diff is collapsed.
...@@ -574,11 +574,13 @@ const derivedConstant :TestAllTypes = ( ...@@ -574,11 +574,13 @@ const derivedConstant :TestAllTypes = (
structList = TestConstants.structListConst); structList = TestConstants.structListConst);
interface TestInterface { interface TestInterface {
foo @0 (i :UInt32, j :Bool) :Text; foo @0 (i :UInt32, j :Bool) -> (x: Text);
bar @1 () :Void; bar @1 () -> ();
baz @2 (s: TestAllTypes); baz @2 (s: TestAllTypes);
} }
interface TestExtends extends(TestInterface) { interface TestExtends extends(TestInterface) {
qux @0 (); qux @0 ();
corge @1 TestAllTypes -> ();
grault @2 () -> TestAllTypes;
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment