| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371 |
- // Part of the Carbon Language 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 "toolchain/check/cpp_thunk.h"
- #include "clang/AST/GlobalDecl.h"
- #include "clang/AST/Mangle.h"
- #include "clang/Sema/Sema.h"
- #include "toolchain/check/call.h"
- #include "toolchain/check/context.h"
- #include "toolchain/check/control_flow.h"
- #include "toolchain/check/literal.h"
- #include "toolchain/check/type.h"
- #include "toolchain/check/type_completion.h"
- #include "toolchain/sem_ir/ids.h"
- #include "toolchain/sem_ir/typed_insts.h"
- namespace Carbon::Check {
- // Returns the C++ thunk mangled name given the callee function.
- static auto GenerateThunkMangledName(
- clang::ASTContext& ast_context,
- const clang::FunctionDecl& callee_function_decl) -> std::string {
- RawStringOstream mangled_name_stream;
- {
- // TODO: Create `MangleContext` once.
- std::unique_ptr<clang::MangleContext> mangle_context(
- ast_context.createMangleContext());
- mangle_context->mangleName(clang::GlobalDecl(&callee_function_decl),
- mangled_name_stream);
- }
- mangled_name_stream << ".carbon_thunk";
- return mangled_name_stream.TakeStr();
- }
- // Returns true if a C++ thunk is required for the given type. A C++ thunk is
- // required for any type except for void, pointer types and signed 32-bit and
- // 64-bit integers.
- static auto IsThunkRequiredForType(Context& context, SemIR::TypeId type_id)
- -> bool {
- if (!type_id.has_value() || type_id == SemIR::ErrorInst::TypeId) {
- return false;
- }
- type_id = context.types().GetUnqualifiedType(type_id);
- switch (context.types().GetAsInst(type_id).kind()) {
- case SemIR::PointerType::Kind: {
- return false;
- }
- case SemIR::ClassType::Kind: {
- if (!context.types().IsComplete(type_id)) {
- // Signed integers of 32 or 64 bits should be completed when imported.
- return true;
- }
- if (!context.types().IsSignedInt(type_id)) {
- return true;
- }
- llvm::APInt bit_width =
- context.ints().Get(context.types().GetIntTypeInfo(type_id).bit_width);
- return bit_width != 32 && bit_width != 64;
- }
- default:
- return true;
- }
- }
- auto IsCppThunkRequired(Context& context, const SemIR::Function& function)
- -> bool {
- if (!function.clang_decl_id.has_value()) {
- return false;
- }
- if (function.self_param_id.has_value()) {
- // TODO: Support member methods.
- return false;
- }
- SemIR::TypeId return_type_id =
- function.GetDeclaredReturnType(context.sem_ir());
- if (return_type_id.has_value()) {
- // TODO: Support non-void return values.
- return false;
- }
- bool thunk_required_for_param = false;
- for (auto param_id :
- context.inst_blocks().GetOrEmpty(function.call_params_id)) {
- if (param_id == SemIR::ErrorInst::InstId) {
- return false;
- }
- if (!thunk_required_for_param &&
- IsThunkRequiredForType(
- context,
- context.insts().GetAs<SemIR::AnyParam>(param_id).type_id)) {
- thunk_required_for_param = true;
- }
- }
- return thunk_required_for_param;
- }
- // Returns whether the type is a pointer or a signed int of 32 or 64 bits.
- static auto IsSimpleAbiType(clang::ASTContext& ast_context,
- clang::QualType type) -> bool {
- if (type->isPointerType()) {
- return true;
- }
- if (const auto* builtin_type = type->getAs<clang::BuiltinType>()) {
- if (builtin_type->isSignedInteger()) {
- uint64_t type_size = ast_context.getIntWidth(type);
- return type_size == 32 || type_size == 64;
- }
- }
- return false;
- }
- // Creates the thunk parameter types given the callee function. Also returns for
- // each type whether it is different from the matching callee function parameter
- // type.
- static auto BuildThunkParameterTypes(
- clang::ASTContext& ast_context,
- const clang::FunctionDecl& callee_function_decl)
- -> std::tuple<llvm::SmallVector<clang::QualType>, llvm::SmallVector<bool>> {
- std::tuple<llvm::SmallVector<clang::QualType>, llvm::SmallVector<bool>>
- result;
- auto& [thunk_param_types, param_type_changed] = result;
- unsigned num_params = callee_function_decl.getNumParams();
- thunk_param_types.reserve(num_params);
- param_type_changed.reserve(num_params);
- for (const clang::ParmVarDecl* callee_param :
- callee_function_decl.parameters()) {
- clang::QualType param_type = callee_param->getType();
- bool is_simple_abi_type = IsSimpleAbiType(ast_context, param_type);
- if (!is_simple_abi_type) {
- clang::QualType pointer_type = ast_context.getPointerType(param_type);
- param_type = ast_context.getAttributedType(
- clang::NullabilityKind::NonNull, pointer_type, pointer_type);
- }
- param_type_changed.push_back(!is_simple_abi_type);
- thunk_param_types.push_back(param_type);
- }
- return result;
- }
- // Returns the thunk parameters using the callee function parameter identifiers.
- static auto BuildThunkParameters(
- clang::ASTContext& ast_context,
- const clang::FunctionDecl& callee_function_decl,
- clang::FunctionDecl* thunk_function_decl)
- -> llvm::SmallVector<clang::ParmVarDecl*> {
- clang::SourceLocation clang_loc = callee_function_decl.getLocation();
- unsigned num_params = thunk_function_decl->getNumParams();
- CARBON_CHECK(callee_function_decl.getNumParams() == num_params);
- const auto* thunk_function_proto_type =
- thunk_function_decl->getFunctionType()->getAs<clang::FunctionProtoType>();
- llvm::SmallVector<clang::ParmVarDecl*> thunk_params;
- thunk_params.reserve(num_params);
- for (unsigned i = 0; i < num_params; ++i) {
- clang::ParmVarDecl* thunk_param = clang::ParmVarDecl::Create(
- ast_context, thunk_function_decl, clang_loc, clang_loc,
- callee_function_decl.getParamDecl(i)->getIdentifier(),
- thunk_function_proto_type->getParamType(i), nullptr, clang::SC_None,
- nullptr);
- thunk_params.push_back(thunk_param);
- }
- return thunk_params;
- }
- // Returns the thunk function declaration given the callee function and the
- // thunk parameter types.
- static auto CreateThunkFunctionDecl(
- clang::ASTContext& ast_context,
- const clang::FunctionDecl& callee_function_decl,
- llvm::ArrayRef<clang::QualType> thunk_param_types) -> clang::FunctionDecl* {
- clang::SourceLocation clang_loc = callee_function_decl.getLocation();
- clang::IdentifierInfo& identifier_info = ast_context.Idents.get(
- callee_function_decl.getNameAsString() + "__carbon_thunk");
- const auto* callee_function_type = callee_function_decl.getFunctionType()
- ->castAs<clang::FunctionProtoType>();
- // TODO: Check whether we need to modify `ExtParameterInfo` in `ExtProtoInfo`.
- clang::QualType thunk_function_type = ast_context.getFunctionType(
- callee_function_decl.getReturnType(), thunk_param_types,
- callee_function_type->getExtProtoInfo());
- // TODO: Thunks should not have external linkage, consider using `SC_Static`.
- clang::FunctionDecl* thunk_function_decl = clang::FunctionDecl::Create(
- ast_context, ast_context.getTranslationUnitDecl(), clang_loc, clang_loc,
- clang::DeclarationName(&identifier_info), thunk_function_type,
- /*TInfo=*/nullptr, clang::SC_Extern);
- thunk_function_decl->setParams(BuildThunkParameters(
- ast_context, callee_function_decl, thunk_function_decl));
- // Set always_inline.
- thunk_function_decl->addAttr(
- clang::AlwaysInlineAttr::CreateImplicit(ast_context));
- // Set asm("<callee function mangled name>.carbon_thunk").
- thunk_function_decl->addAttr(clang::AsmLabelAttr::CreateImplicit(
- ast_context, GenerateThunkMangledName(ast_context, callee_function_decl),
- clang_loc));
- return thunk_function_decl;
- }
- // Takes the thunk function parameters and for each one creates an arg for the
- // callee function which is the thunk parameter or its address.
- static auto BuildCalleeArgs(clang::Sema& sema,
- clang::FunctionDecl* thunk_function_decl,
- llvm::ArrayRef<bool> param_type_changed)
- -> llvm::SmallVector<clang::Expr*> {
- llvm::SmallVector<clang::Expr*> call_args;
- size_t num_params = thunk_function_decl->getNumParams();
- CARBON_CHECK(param_type_changed.size() == num_params);
- call_args.reserve(num_params);
- for (unsigned i = 0; i < num_params; ++i) {
- clang::ParmVarDecl* thunk_param = thunk_function_decl->getParamDecl(i);
- clang::SourceLocation clang_loc = thunk_param->getLocation();
- clang::Expr* call_arg = sema.BuildDeclRefExpr(
- thunk_param, thunk_param->getType(), clang::VK_LValue, clang_loc);
- if (param_type_changed[i]) {
- // TODO: Insert a cast to an rvalue.
- clang::ExprResult deref_result =
- sema.BuildUnaryOp(nullptr, clang_loc, clang::UO_Deref, call_arg);
- CARBON_CHECK(deref_result.isUsable());
- call_arg = deref_result.get();
- }
- call_args.push_back(call_arg);
- }
- return call_args;
- }
- // Builds the thunk function body which calls the callee function using the call
- // args and returns the callee function return value. Returns nullptr on
- // failure.
- static auto BuildThunkBody(clang::Sema& sema,
- clang::FunctionDecl* callee_function_decl,
- llvm::MutableArrayRef<clang::Expr*> call_args)
- -> clang::Stmt* {
- clang::SourceLocation clang_loc = callee_function_decl->getLocation();
- clang::DeclRefExpr* callee_function_ref = sema.BuildDeclRefExpr(
- callee_function_decl, callee_function_decl->getType(), clang::VK_PRValue,
- clang_loc);
- clang::ExprResult call_result = sema.BuildCallExpr(
- nullptr, callee_function_ref, clang_loc, call_args, clang_loc);
- if (!call_result.isUsable()) {
- return nullptr;
- }
- clang::Expr* call = call_result.get();
- clang::StmtResult return_result = sema.BuildReturnStmt(clang_loc, call);
- CARBON_CHECK(return_result.isUsable());
- return return_result.get();
- }
- auto BuildCppThunk(Context& context, const SemIR::Function& callee_function)
- -> clang::FunctionDecl* {
- clang::ASTContext& ast_context = context.ast_context();
- clang::FunctionDecl* callee_function_decl =
- context.sem_ir()
- .clang_decls()
- .Get(callee_function.clang_decl_id)
- .decl->getAsFunction();
- CARBON_CHECK(callee_function_decl);
- // Build the thunk function declaration.
- auto [thunk_param_types, param_type_changed] =
- BuildThunkParameterTypes(ast_context, *callee_function_decl);
- clang::FunctionDecl* thunk_function_decl = CreateThunkFunctionDecl(
- ast_context, *callee_function_decl, thunk_param_types);
- // Build the thunk function body.
- clang::Sema& sema = context.sem_ir().cpp_ast()->getSema();
- clang::Sema::ContextRAII context_raii(sema, thunk_function_decl);
- sema.ActOnStartOfFunctionDef(nullptr, thunk_function_decl);
- llvm::SmallVector<clang::Expr*> call_args =
- BuildCalleeArgs(sema, thunk_function_decl, param_type_changed);
- clang::Stmt* body = BuildThunkBody(sema, callee_function_decl, call_args);
- sema.ActOnFinishFunctionBody(thunk_function_decl, body);
- if (!body) {
- return nullptr;
- }
- return thunk_function_decl;
- }
- auto PerformCppThunkCall(Context& context, SemIR::LocId loc_id,
- SemIR::FunctionId callee_function_id,
- llvm::ArrayRef<SemIR::InstId> callee_arg_ids,
- SemIR::InstId thunk_callee_id) -> SemIR::InstId {
- llvm::ArrayRef<SemIR::InstId> callee_function_params =
- context.inst_blocks().GetOrEmpty(
- context.functions().Get(callee_function_id).call_params_id);
- llvm::ArrayRef<SemIR::InstId> thunk_function_params =
- context.inst_blocks().GetOrEmpty(
- context.functions()
- .Get(GetCalleeFunction(context.sem_ir(), thunk_callee_id)
- .function_id)
- .call_params_id);
- size_t num_params = callee_function_params.size();
- CARBON_CHECK(thunk_function_params.size() == num_params);
- CARBON_CHECK(callee_arg_ids.size() == num_params);
- llvm::SmallVector<SemIR::InstId> thunk_arg_ids;
- thunk_arg_ids.reserve(callee_arg_ids.size());
- for (size_t i = 0; i < callee_function_params.size(); ++i) {
- SemIR::TypeId callee_param_type_id =
- context.insts()
- .GetAs<SemIR::AnyParam>(callee_function_params[i])
- .type_id;
- SemIR::TypeId thunk_param_type_id =
- context.insts()
- .GetAs<SemIR::AnyParam>(thunk_function_params[i])
- .type_id;
- SemIR::InstId arg_id = callee_arg_ids[i];
- if (callee_param_type_id != thunk_param_type_id) {
- CARBON_CHECK(thunk_param_type_id ==
- GetPointerType(context, context.types().GetInstId(
- callee_param_type_id)));
- // TODO: Don't create storage if it's already in a storage (depends on
- // expression category).
- SemIR::InstId temporary_storage_inst_id = AddInstWithCleanup(
- context, loc_id,
- SemIR::TemporaryStorage{.type_id = callee_param_type_id});
- AddInst(context, loc_id,
- SemIR::InitializeFrom{.type_id = callee_param_type_id,
- .src_id = arg_id,
- .dest_id = temporary_storage_inst_id});
- // TODO: Do not use `InitializeFrom` directly. Use the `Initialize`
- // machinery. See
- // https://github.com/carbon-language/carbon-lang/pull/5850/files#r2249030529.
- arg_id = AddInst(context, loc_id,
- SemIR::AddrOf{.type_id = thunk_param_type_id,
- .lvalue_id = temporary_storage_inst_id});
- }
- thunk_arg_ids.push_back(arg_id);
- }
- return PerformCall(context, loc_id, thunk_callee_id, thunk_arg_ids);
- }
- } // namespace Carbon::Check
|