Commit 24dc6ae7 authored by Kenton Varda's avatar Kenton Varda

When an RPC method name matches a C++ keyword, add an underscore to avoid compilation errors.

parent 45f3133d
...@@ -742,6 +742,32 @@ TEST(Capability, Lists) { ...@@ -742,6 +742,32 @@ TEST(Capability, Lists) {
verifyClient(dynamicListReader[2].as<DynamicCapability>(), callCount3, waitScope); verifyClient(dynamicListReader[2].as<DynamicCapability>(), callCount3, waitScope);
} }
TEST(Capability, KeywordMethods) {
// Verify that keywords are only munged where necessary.
kj::EventLoop loop;
kj::WaitScope waitScope(loop);
bool called = false;
class TestKeywordMethodsImpl: public test::TestKeywordMethods::Server {
public:
TestKeywordMethodsImpl(bool& called): called(called) {}
kj::Promise<void> delete_(DeleteContext context) override {
called = true;
return kj::READY_NOW;
}
private:
bool& called;
};
test::TestKeywordMethods::Client client = kj::heap<TestKeywordMethodsImpl>(called);
client.deleteRequest().send().wait(waitScope);
EXPECT_TRUE(called);
}
} // namespace } // namespace
} // namespace _ } // namespace _
} // namespace capnp } // namespace capnp
...@@ -139,6 +139,31 @@ kj::StringPtr baseName(kj::StringPtr path) { ...@@ -139,6 +139,31 @@ kj::StringPtr baseName(kj::StringPtr path) {
} }
} }
kj::String safeIdentifier(kj::StringPtr identifier) {
// Given a desired identifier name, munge it to make it safe for use in generated code.
//
// If the identifier is a keyword, this adds an underscore to the end.
static const std::set<kj::StringPtr> keywords({
"alignas", "alignof", "and", "and_eq", "asm", "auto", "bitand", "bitor", "bool", "break",
"case", "catch", "char", "char16_t", "char32_t", "class", "compl", "const", "constexpr",
"const_cast", "continue", "decltype", "default", "delete", "do", "double", "dynamic_cast",
"else", "enum", "explicit", "export", "extern", "false", "float", "for", "friend", "goto",
"if", "inline", "int", "long", "mutable", "namespace", "new", "noexcept", "not", "not_eq",
"nullptr", "operator", "or", "or_eq", "private", "protected", "public", "register",
"reinterpret_cast", "return", "short", "signed", "sizeof", "static", "static_assert",
"static_cast", "struct", "switch", "template", "this", "thread_local", "throw", "true",
"try", "typedef", "typeid", "typename", "union", "unsigned", "using", "virtual", "void",
"volatile", "wchar_t", "while", "xor", "xor_eq"
});
if (keywords.count(identifier) > 0) {
return kj::str(identifier, '_');
} else {
return kj::heapString(identifier);
}
}
// ======================================================================================= // =======================================================================================
class CapnpcCppMain { class CapnpcCppMain {
...@@ -1197,6 +1222,7 @@ private: ...@@ -1197,6 +1222,7 @@ private:
auto titleCase = toTitleCase(name); auto titleCase = toTitleCase(name);
auto paramSchema = schemaLoader.get(proto.getParamStructType()).asStruct(); auto paramSchema = schemaLoader.get(proto.getParamStructType()).asStruct();
auto resultSchema = schemaLoader.get(proto.getResultStructType()).asStruct(); auto resultSchema = schemaLoader.get(proto.getResultStructType()).asStruct();
auto identifierName = safeIdentifier(name);
auto paramProto = paramSchema.getProto(); auto paramProto = paramSchema.getProto();
auto resultProto = resultSchema.getProto(); auto resultProto = resultSchema.getProto();
...@@ -1228,7 +1254,7 @@ private: ...@@ -1228,7 +1254,7 @@ private:
" typedef ", resultType, " ", titleCase, "Results;\n"), " typedef ", resultType, " ", titleCase, "Results;\n"),
" typedef ::capnp::CallContext<", shortParamType, ", ", shortResultType, "> ", " typedef ::capnp::CallContext<", shortParamType, ", ", shortResultType, "> ",
titleCase, "Context;\n" titleCase, "Context;\n"
" virtual ::kj::Promise<void> ", name, "(", titleCase, "Context context);\n"), " virtual ::kj::Promise<void> ", identifierName, "(", titleCase, "Context context);\n"),
kj::strTree(), kj::strTree(),
...@@ -1238,7 +1264,7 @@ private: ...@@ -1238,7 +1264,7 @@ private:
" return newCall<", paramType, ", ", resultType, ">(\n" " return newCall<", paramType, ", ", resultType, ">(\n"
" 0x", interfaceIdHex, "ull, ", methodId, ", sizeHint);\n" " 0x", interfaceIdHex, "ull, ", methodId, ", sizeHint);\n"
"}\n" "}\n"
"::kj::Promise<void> ", interfaceName, "::Server::", name, "(", titleCase, "Context) {\n" "::kj::Promise<void> ", interfaceName, "::Server::", identifierName, "(", titleCase, "Context) {\n"
" return ::capnp::Capability::Server::internalUnimplemented(\n" " return ::capnp::Capability::Server::internalUnimplemented(\n"
" \"", interfaceProto.getDisplayName(), "\", \"", name, "\",\n" " \"", interfaceProto.getDisplayName(), "\", \"", name, "\",\n"
" 0x", interfaceIdHex, "ull, ", methodId, ");\n" " 0x", interfaceIdHex, "ull, ", methodId, ");\n"
...@@ -1246,7 +1272,7 @@ private: ...@@ -1246,7 +1272,7 @@ private:
kj::strTree( kj::strTree(
" case ", methodId, ":\n" " case ", methodId, ":\n"
" return ", name, "(::capnp::Capability::Server::internalGetTypedContext<\n" " return ", identifierName, "(::capnp::Capability::Server::internalGetTypedContext<\n"
" ", paramType, ", ", resultType, ">(context));\n") " ", paramType, ", ", resultType, ">(context));\n")
}; };
} }
......
...@@ -661,6 +661,13 @@ interface TestMoreStuff extends(TestCallOrder) { ...@@ -661,6 +661,13 @@ interface TestMoreStuff extends(TestCallOrder) {
methodWithDefaults @8 (a :Text, b :UInt32 = 123, c :Text = "foo") -> (d :Text, e :Text = "bar"); methodWithDefaults @8 (a :Text, b :UInt32 = 123, c :Text = "foo") -> (d :Text, e :Text = "bar");
} }
interface TestKeywordMethods {
delete @0 ();
class @1 ();
void @2 ();
return @3 ();
}
struct TestSturdyRefHostId { struct TestSturdyRefHostId {
host @0 :Text; host @0 :Text;
} }
......
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