Send IR to Rust to generate rust source code using quote!
To pass IR across the FFI boundary we serialize it to a json string, pass the string to Rust, and return a string with generated Rust source code back.
Alternatives considered:
Protobuf instead of json
------------------------
using protobuf as a serialization format would have some advantages (defining types only once, we wouldn't have to write our own serialization logic in C++), we decided not to use it because:
* there is no approved way of using protobufs in Rust in google3 today, and we didn't want to special case us. We could still use protobuf on the C++ side and generate json using it, so at least we don't have to write serialization code ourselves.
* we felt going with json and manual serialization will be more flexible in the uncertain future
* we tossed a coin and json won
Implementing our own Rust code generator in C++
-----------------------------------------------
We sketched the code in unknown commit, and we decided going with Rust solution is more readable and maintainable. We also plan to use Clang syntax trees to generate C++ source code, of which quote!, proc_macro2, and syn are moral equivalents.
PiperOrigin-RevId: 389566607
diff --git a/rs_bindings_from_cc/ir.h b/rs_bindings_from_cc/ir.h
index 43bf33b..8c5cb68 100644
--- a/rs_bindings_from_cc/ir.h
+++ b/rs_bindings_from_cc/ir.h
@@ -16,6 +16,7 @@
#include "base/logging.h"
#include "third_party/absl/strings/cord.h"
+#include "third_party/json/src/json.hpp"
namespace rs_bindings_from_cc {
@@ -35,6 +36,8 @@
const absl::Cord &RsName() const { return rs_name_; }
+ nlohmann::json ToJson() const;
+
private:
absl::Cord rs_name_;
};
@@ -56,6 +59,8 @@
const absl::Cord &Ident() const { return identifier_; }
+ nlohmann::json ToJson() const;
+
private:
absl::Cord identifier_;
};
@@ -73,31 +78,18 @@
const Type &ParamType() const { return type_; }
const Identifier &Ident() const { return identifier_; }
+ nlohmann::json ToJson() const;
+
private:
Type type_;
Identifier identifier_;
};
-// All parameters of a function.
-//
-// Invariants:
-// `params` can be empty.
-class FuncParams {
- public:
- explicit FuncParams(std::vector<FuncParam> params)
- : params_(std::move(params)) {}
-
- const std::vector<FuncParam> &Params() const { return params_; }
-
- private:
- std::vector<FuncParam> params_;
-};
-
// A function involved in the bindings.
class Func {
public:
explicit Func(Identifier identifier, absl::Cord mangled_name,
- Type return_type, FuncParams params)
+ Type return_type, std::vector<FuncParam> params)
: identifier_(std::move(identifier)),
mangled_name_(std::move(mangled_name)),
return_type_(std::move(return_type)),
@@ -107,13 +99,15 @@
const Type &ReturnType() const { return return_type_; }
const Identifier &Ident() const { return identifier_; }
- const FuncParams &Params() const { return params_; }
+ const std::vector<FuncParam> &Params() const { return params_; }
+
+ nlohmann::json ToJson() const;
private:
Identifier identifier_;
absl::Cord mangled_name_;
Type return_type_;
- FuncParams params_;
+ std::vector<FuncParam> params_;
};
// A complete intermediate representation of bindings for publicly accessible
@@ -122,6 +116,8 @@
public:
explicit IR(std::vector<Func> functions) : functions_(std::move(functions)) {}
+ nlohmann::json ToJson() const;
+
private:
std::vector<Func> functions_;
};