Move `FunctionDeclImporter` to its own file.

PiperOrigin-RevId: 444186354
diff --git a/rs_bindings_from_cc/importers/function.cc b/rs_bindings_from_cc/importers/function.cc
new file mode 100644
index 0000000..b203a74
--- /dev/null
+++ b/rs_bindings_from_cc/importers/function.cc
@@ -0,0 +1,216 @@
+// 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
+
+#include "rs_bindings_from_cc/importers/function.h"
+
+#include "third_party/absl/strings/substitute.h"
+
+namespace crubit {
+
+std::optional<IR::Item> FunctionDeclImporter::Import(
+    clang::FunctionDecl* function_decl) {
+  if (!ictx_.IsFromCurrentTarget(function_decl)) return std::nullopt;
+  if (function_decl->isDeleted()) return std::nullopt;
+
+  clang::tidy::lifetimes::LifetimeSymbolTable lifetime_symbol_table;
+  llvm::Expected<clang::tidy::lifetimes::FunctionLifetimes> lifetimes =
+      clang::tidy::lifetimes::GetLifetimeAnnotations(
+          function_decl, *ictx_.invocation_.lifetime_context_,
+          &lifetime_symbol_table);
+
+  std::vector<FuncParam> params;
+  std::set<std::string> errors;
+  auto add_error = [&errors](std::string msg) {
+    auto result = errors.insert(std::move(msg));
+    CRUBIT_CHECK(result.second && "Duplicated error message");
+  };
+  if (auto* method_decl =
+          clang::dyn_cast<clang::CXXMethodDecl>(function_decl)) {
+    if (!ictx_.type_mapper_.Contains(method_decl->getParent())) {
+      return ictx_.ImportUnsupportedItem(function_decl,
+                                         "Couldn't import the parent");
+    }
+
+    // non-static member functions receive an implicit `this` parameter.
+    if (method_decl->isInstance()) {
+      std::optional<clang::tidy::lifetimes::ValueLifetimes> this_lifetimes;
+      if (lifetimes) {
+        this_lifetimes = lifetimes->GetThisLifetimes();
+      }
+      auto param_type = ictx_.type_mapper_.ConvertQualType(
+          method_decl->getThisType(), this_lifetimes,
+          /*nullable=*/false);
+      if (!param_type.ok()) {
+        add_error(absl::StrCat("`this` parameter is not supported: ",
+                               param_type.status().message()));
+      } else {
+        params.push_back({*std::move(param_type), Identifier("__this")});
+      }
+    }
+  }
+
+  if (lifetimes) {
+    CRUBIT_CHECK(lifetimes->IsValidForDecl(function_decl));
+  }
+
+  for (unsigned i = 0; i < function_decl->getNumParams(); ++i) {
+    const clang::ParmVarDecl* param = function_decl->getParamDecl(i);
+    std::optional<clang::tidy::lifetimes::ValueLifetimes> param_lifetimes;
+    if (lifetimes) {
+      param_lifetimes = lifetimes->GetParamLifetimes(i);
+    }
+    auto param_type =
+        ictx_.type_mapper_.ConvertQualType(param->getType(), param_lifetimes);
+    if (!param_type.ok()) {
+      add_error(absl::Substitute("Parameter #$0 is not supported: $1", i,
+                                 param_type.status().message()));
+      continue;
+    }
+
+    if (const clang::RecordType* record_type =
+            clang::dyn_cast<clang::RecordType>(param->getType())) {
+      if (clang::RecordDecl* record_decl =
+              clang::dyn_cast<clang::RecordDecl>(record_type->getDecl())) {
+        // TODO(b/200067242): non-trivial_abi structs, when passed by value,
+        // have a different representation which needs special support. We
+        // currently do not support it.
+        if (!record_decl->canPassInRegisters()) {
+          add_error(
+              absl::Substitute("Non-trivial_abi type '$0' is not "
+                               "supported by value as parameter #$1",
+                               param->getType().getAsString(), i));
+        }
+      }
+    }
+
+    std::optional<Identifier> param_name = ictx_.GetTranslatedIdentifier(param);
+    CRUBIT_CHECK(param_name.has_value());  // No known failure cases.
+    params.push_back({*param_type, *std::move(param_name)});
+  }
+
+  if (const clang::RecordType* record_return_type =
+          clang::dyn_cast<clang::RecordType>(function_decl->getReturnType())) {
+    if (clang::RecordDecl* record_decl =
+            clang::dyn_cast<clang::RecordDecl>(record_return_type->getDecl())) {
+      // TODO(b/200067242): non-trivial_abi structs, when passed by value,
+      // have a different representation which needs special support. We
+      // currently do not support it.
+      if (!record_decl->canPassInRegisters()) {
+        add_error(
+            absl::Substitute("Non-trivial_abi type '$0' is not supported "
+                             "by value as a return type",
+                             function_decl->getReturnType().getAsString()));
+      }
+    }
+  }
+
+  std::optional<clang::tidy::lifetimes::ValueLifetimes> return_lifetimes;
+  if (lifetimes) {
+    return_lifetimes = lifetimes->GetReturnLifetimes();
+  }
+
+  auto return_type = ictx_.type_mapper_.ConvertQualType(
+      function_decl->getReturnType(), return_lifetimes);
+  if (!return_type.ok()) {
+    add_error(absl::StrCat("Return type is not supported: ",
+                           return_type.status().message()));
+  }
+
+  llvm::DenseSet<clang::tidy::lifetimes::Lifetime> all_free_lifetimes;
+  if (lifetimes) {
+    all_free_lifetimes = lifetimes->AllFreeLifetimes();
+  }
+
+  std::vector<LifetimeName> lifetime_params;
+  for (clang::tidy::lifetimes::Lifetime lifetime : all_free_lifetimes) {
+    std::optional<llvm::StringRef> name =
+        lifetime_symbol_table.LookupLifetime(lifetime);
+    CRUBIT_CHECK(name.has_value());
+    lifetime_params.push_back(
+        {.name = name->str(), .id = LifetimeId(lifetime.Id())});
+  }
+  llvm::sort(lifetime_params,
+             [](const LifetimeName& l1, const LifetimeName& l2) {
+               return l1.name < l2.name;
+             });
+
+  llvm::Optional<MemberFuncMetadata> member_func_metadata;
+  if (auto* method_decl =
+          clang::dyn_cast<clang::CXXMethodDecl>(function_decl)) {
+    switch (method_decl->getAccess()) {
+      case clang::AS_public:
+        break;
+      case clang::AS_protected:
+      case clang::AS_private:
+      case clang::AS_none:
+        // No need for IR to include Func representing private methods.
+        // TODO(lukasza): Revisit this for protected methods.
+        return std::nullopt;
+    }
+    llvm::Optional<MemberFuncMetadata::InstanceMethodMetadata>
+        instance_metadata;
+    if (method_decl->isInstance()) {
+      MemberFuncMetadata::ReferenceQualification reference;
+      switch (method_decl->getRefQualifier()) {
+        case clang::RQ_LValue:
+          reference = MemberFuncMetadata::kLValue;
+          break;
+        case clang::RQ_RValue:
+          reference = MemberFuncMetadata::kRValue;
+          break;
+        case clang::RQ_None:
+          reference = MemberFuncMetadata::kUnqualified;
+          break;
+      }
+      instance_metadata = MemberFuncMetadata::InstanceMethodMetadata{
+          .reference = reference,
+          .is_const = method_decl->isConst(),
+          .is_virtual = method_decl->isVirtual(),
+          .is_explicit_ctor = false,
+      };
+      if (auto* ctor_decl =
+              clang::dyn_cast<clang::CXXConstructorDecl>(function_decl)) {
+        instance_metadata->is_explicit_ctor = ctor_decl->isExplicit();
+      }
+    }
+
+    member_func_metadata = MemberFuncMetadata{
+        .record_id = GenerateItemId(method_decl->getParent()),
+        .instance_method_metadata = instance_metadata};
+  }
+
+  if (!errors.empty()) {
+    return ictx_.ImportUnsupportedItem(function_decl, errors);
+  }
+
+  bool has_c_calling_convention =
+      function_decl->getType()->getAs<clang::FunctionType>()->getCallConv() ==
+      clang::CC_C;
+  std::optional<UnqualifiedIdentifier> translated_name =
+      ictx_.GetTranslatedName(function_decl);
+
+  // Silence ClangTidy, checked above: calling `add_error` if
+  // `!return_type.ok()` and returning early if `!errors.empty()`.
+  CRUBIT_CHECK(return_type.ok());
+
+  if (translated_name.has_value()) {
+    return Func{
+        .name = *translated_name,
+        .owning_target = ictx_.GetOwningTarget(function_decl),
+        .doc_comment = ictx_.GetComment(function_decl),
+        .mangled_name = ictx_.GetMangledName(function_decl),
+        .return_type = *return_type,
+        .params = std::move(params),
+        .lifetime_params = std::move(lifetime_params),
+        .is_inline = function_decl->isInlined(),
+        .member_func_metadata = std::move(member_func_metadata),
+        .has_c_calling_convention = has_c_calling_convention,
+        .source_loc = ictx_.ConvertSourceLocation(function_decl->getBeginLoc()),
+        .id = GenerateItemId(function_decl),
+    };
+  }
+  return std::nullopt;
+}
+
+}  // namespace crubit