blob: 1fd91668a12506e9242a541403d3534b237e009e [file] [log] [blame]
Googler741ed9c2021-10-01 08:00:49 +00001// Part of the Crubit project, under the Apache License v2.0 with LLVM
2// Exceptions. See /LICENSE for license information.
3// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
4
5#include "rs_bindings_from_cc/ir_from_cc.h"
6
Marcel Hlopko20f4ce42021-11-02 08:20:00 +00007#include <memory>
8#include <string>
9#include <utility>
10#include <vector>
11
Googler741ed9c2021-10-01 08:00:49 +000012#include "third_party/absl/container/flat_hash_map.h"
Marcel Hlopko20f4ce42021-11-02 08:20:00 +000013#include "third_party/absl/status/status.h"
14#include "third_party/absl/status/statusor.h"
15#include "third_party/absl/strings/string_view.h"
Googler741ed9c2021-10-01 08:00:49 +000016#include "third_party/absl/strings/substitute.h"
Marcel Hlopko20f4ce42021-11-02 08:20:00 +000017#include "third_party/absl/types/span.h"
Marco Polettic61bcc42022-04-08 12:54:30 -070018#include "common/check.h"
Marcel Hlopko3b254b32022-03-09 14:10:49 +000019#include "rs_bindings_from_cc/bazel_types.h"
20#include "rs_bindings_from_cc/frontend_action.h"
21#include "rs_bindings_from_cc/importer.h"
22#include "rs_bindings_from_cc/ir.h"
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000023#include "third_party/llvm/llvm-project/clang/include/clang/Basic/FileManager.h"
24#include "third_party/llvm/llvm-project/clang/include/clang/Basic/FileSystemOptions.h"
Marcel Hlopko20f4ce42021-11-02 08:20:00 +000025#include "third_party/llvm/llvm-project/clang/include/clang/Frontend/FrontendAction.h"
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000026#include "third_party/llvm/llvm-project/clang/include/clang/Tooling/Tooling.h"
Googler741ed9c2021-10-01 08:00:49 +000027
Marcel Hlopkof15e8ce2022-04-08 08:46:09 -070028namespace crubit {
Googler741ed9c2021-10-01 08:00:49 +000029
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000030static constexpr absl::string_view kVirtualHeaderPath =
31 "ir_from_cc_virtual_header.h";
Googler741ed9c2021-10-01 08:00:49 +000032static constexpr absl::string_view kVirtualInputPath =
Michael Forster3f323be2021-10-11 07:13:28 +000033 "ir_from_cc_virtual_input.cc";
Googler741ed9c2021-10-01 08:00:49 +000034
Michael Forster3f323be2021-10-11 07:13:28 +000035absl::StatusOr<IR> IrFromCc(
Googler6c3de122022-03-28 11:40:41 +000036 const absl::string_view extra_source_code, const BazelLabel current_target,
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000037 absl::Span<const HeaderName> public_headers,
38 absl::flat_hash_map<const HeaderName, const std::string>
39 virtual_headers_contents,
Googler6c3de122022-03-28 11:40:41 +000040 absl::flat_hash_map<const HeaderName, const BazelLabel> headers_to_targets,
Michael Forster3f323be2021-10-11 07:13:28 +000041 absl::Span<const absl::string_view> args) {
Lukasz Anforowicz34ad7f72022-03-17 16:05:28 +000042 // Caller should verify that the inputs are not empty.
43 CRUBIT_CHECK(!extra_source_code.empty() || !public_headers.empty());
44 CRUBIT_CHECK(!extra_source_code.empty() || !headers_to_targets.empty());
45
Googler7ea9fbb2022-03-11 02:56:03 +000046 std::vector<HeaderName> entrypoint_headers(public_headers.begin(),
47 public_headers.end());
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000048 clang::tooling::FileContentMappings file_contents;
Googler741ed9c2021-10-01 08:00:49 +000049
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000050 for (auto const& name_and_content : virtual_headers_contents) {
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000051 file_contents.push_back({std::string(name_and_content.first.IncludePath()),
52 name_and_content.second});
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000053 }
54 if (!extra_source_code.empty()) {
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000055 file_contents.push_back(
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000056 {std::string(kVirtualHeaderPath), std::string(extra_source_code)});
57 HeaderName header_name = HeaderName(std::string(kVirtualHeaderPath));
58 entrypoint_headers.push_back(header_name);
59 headers_to_targets.insert({header_name, current_target});
Googler741ed9c2021-10-01 08:00:49 +000060 }
61
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000062 std::string virtual_input_file_content;
63 for (const HeaderName& header_name : entrypoint_headers) {
Michael Forster3f323be2021-10-11 07:13:28 +000064 absl::SubstituteAndAppend(&virtual_input_file_content, "#include \"$0\"\n",
Marcel Hlopko7aa38a72021-11-11 07:39:51 +000065 header_name.IncludePath());
Michael Forster3f323be2021-10-11 07:13:28 +000066 }
Googler741ed9c2021-10-01 08:00:49 +000067
Michael Forster3f323be2021-10-11 07:13:28 +000068 std::vector<std::string> args_as_strings{
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000069 // Parse non-doc comments that are used as documention
Michael Forster3f323be2021-10-11 07:13:28 +000070 "-fparse-all-comments"};
71 args_as_strings.insert(args_as_strings.end(), args.begin(), args.end());
Googler741ed9c2021-10-01 08:00:49 +000072
Michael Forstera49d2e62022-01-28 07:26:40 +000073 if (Importer::Invocation invocation(current_target, entrypoint_headers,
74 headers_to_targets);
Rosica Dejanovska5fc4caf2022-01-31 13:54:32 +000075 clang::tooling::runToolOnCodeWithArgs(
76 std::make_unique<FrontendAction>(invocation),
77 virtual_input_file_content, args_as_strings, kVirtualInputPath,
78 "rs_bindings_from_cc",
79 std::make_shared<clang::PCHContainerOperations>(), file_contents)) {
Michael Forstera49d2e62022-01-28 07:26:40 +000080 return invocation.ir_;
Michael Forster3f323be2021-10-11 07:13:28 +000081 } else {
82 return absl::Status(absl::StatusCode::kInvalidArgument,
83 "Could not compile header contents");
84 }
Googler741ed9c2021-10-01 08:00:49 +000085}
86
Marcel Hlopkof15e8ce2022-04-08 08:46:09 -070087} // namespace crubit