blob: 229e6372a5e817af552a0a76719cb7ecbb344b94 [file] [log] [blame]
// Part of the Crubit project, under the Apache License v2.0 with LLVM
// Exceptions. See /LICENSE for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
// Parses C++ headers and generates:
// * a Rust source file with bindings for the C++ API
// * a C++ source file with the implementation of the bindings
#include <string>
#include <utility>
#include <vector>
#include "third_party/absl/container/flat_hash_map.h"
#include "third_party/absl/flags/parse.h"
#include "third_party/absl/status/status.h"
#include "third_party/absl/status/statusor.h"
#include "third_party/absl/strings/string_view.h"
#include "rs_bindings_from_cc/bazel_types.h"
#include "rs_bindings_from_cc/cmdline.h"
#include "rs_bindings_from_cc/ir.h"
#include "rs_bindings_from_cc/ir_from_cc.h"
#include "rs_bindings_from_cc/src_code_gen.h"
#include "rs_bindings_from_cc/util/status_macros.h"
#include "third_party/llvm/llvm-project/llvm/include/llvm/Support/FormatVariadic.h"
#include "third_party/llvm/llvm-project/llvm/include/llvm/Support/JSON.h"
#include "third_party/llvm/llvm-project/llvm/include/llvm/Support/raw_ostream.h"
namespace {
absl::Status SetFileContents(absl::string_view path,
absl::string_view contents) {
std::error_code error_code;
llvm::raw_fd_ostream stream(path, error_code);
if (error_code) {
return absl::Status(absl::StatusCode::kInternal, error_code.message());
}
stream << contents;
stream.close();
if (stream.has_error()) {
return absl::Status(absl::StatusCode::kInternal, stream.error().message());
}
return absl::OkStatus();
}
absl::Status Main(std::vector<char*> args) {
using rs_bindings_from_cc::Cmdline;
using rs_bindings_from_cc::IR;
CRUBIT_ASSIGN_OR_RETURN(Cmdline cmdline, Cmdline::Create());
if (cmdline.do_nothing()) {
CRUBIT_RETURN_IF_ERROR(SetFileContents(
cmdline.rs_out(),
"// intentionally left empty because --do_nothing was passed."));
CRUBIT_RETURN_IF_ERROR(SetFileContents(
cmdline.cc_out(),
"// intentionally left empty because --do_nothing was passed."));
return absl::OkStatus();
}
std::vector<absl::string_view> args_as_string_views;
args_as_string_views.insert(args_as_string_views.end(), args.begin(),
args.end());
CRUBIT_ASSIGN_OR_RETURN(
IR ir, rs_bindings_from_cc::IrFromCc(
/* extra_source_code= */ "", cmdline.current_target(),
cmdline.public_headers(),
/* virtual_headers_contents= */ {},
cmdline.headers_to_targets(), args_as_string_views));
if (!cmdline.ir_out().empty()) {
CRUBIT_RETURN_IF_ERROR(SetFileContents(
cmdline.ir_out(), std::string(llvm::formatv("{0:2}", ir.ToJson()))));
}
rs_bindings_from_cc::Bindings bindings =
rs_bindings_from_cc::GenerateBindings(ir);
CRUBIT_RETURN_IF_ERROR(SetFileContents(cmdline.rs_out(), bindings.rs_api));
CRUBIT_RETURN_IF_ERROR(
SetFileContents(cmdline.cc_out(), bindings.rs_api_impl));
return absl::OkStatus();
}
} // namespace
int main(int argc, char* argv[]) {
auto args = absl::ParseCommandLine(argc, argv);
absl::Status status = Main(std::move(args));
if (!status.ok()) {
llvm::errs() << status.message() << "\n";
return -1;
}
return 0;
}