| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- // 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/macros.h"
- #include "clang/AST/ASTContext.h"
- #include "clang/AST/Expr.h"
- #include "clang/Parse/Parser.h"
- #include "clang/Sema/Sema.h"
- #include "common/check.h"
- namespace Carbon::Check {
- auto TryEvaluateMacroToConstant(Context& context, SemIR::LocId loc_id,
- SemIR::NameId name_id,
- clang::MacroInfo* macro_info) -> clang::Expr* {
- auto name_str_opt = context.names().GetAsStringIfIdentifier(name_id);
- CARBON_CHECK(macro_info, "macro info missing");
- if (macro_info->getNumTokens() == 0) {
- context.TODO(loc_id, "Unsupported: macro with 0 replacement tokens");
- return nullptr;
- }
- clang::Sema& sema = context.clang_sema();
- clang::Preprocessor& preprocessor = sema.getPreprocessor();
- auto& parser = context.cpp_context()->parser();
- llvm::SmallVector<clang::Token> tokens(macro_info->tokens().begin(),
- macro_info->tokens().end());
- clang::Token current_token = parser.getCurToken();
- // Add eof token
- clang::Token eof;
- eof.startToken();
- eof.setKind(clang::tok::eof);
- eof.setLocation(current_token.getEndLoc());
- tokens.push_back(eof);
- tokens.push_back(current_token);
- preprocessor.EnterTokenStream(tokens, /*DisableMacroExpansion=*/false,
- /*IsReinject=*/false);
- parser.ConsumeAnyToken(true);
- clang::ExprResult result = parser.ParseConstantExpression();
- clang::Expr* result_expr = result.get();
- bool success =
- !result.isInvalid() && parser.getCurToken().is(clang::tok::eof);
- if (!success) {
- parser.SkipUntil(clang::tok::eof);
- CARBON_DIAGNOSTIC(
- InCppMacroEvaluation, Error,
- "failed to parse macro Cpp.{0} to a valid constant expression",
- std::string);
- context.emitter().Emit(loc_id, InCppMacroEvaluation, (*name_str_opt).str());
- return nullptr;
- }
- result_expr = result_expr->IgnoreParenImpCasts();
- if (isa<clang::StringLiteral>(result_expr) ||
- isa<clang::CharacterLiteral>(result_expr) ||
- isa<clang::CXXNullPtrLiteralExpr>(result_expr)) {
- return result_expr;
- }
- clang::Expr::EvalResult evaluated_result;
- CARBON_CHECK(result_expr->EvaluateAsConstantExpr(evaluated_result,
- sema.getASTContext()));
- clang::APValue ap_value = evaluated_result.Val;
- // TODO: Add support for other types.
- if (ap_value.isLValue()) {
- if (!result_expr->EvaluateAsInt(evaluated_result, sema.getASTContext())) {
- context.TODO(loc_id,
- "Unsupported: macro evaluated to a non-integer LValue");
- return nullptr;
- }
- ap_value = evaluated_result.Val;
- }
- switch (ap_value.getKind()) {
- case clang::APValue::Int:
- if (result_expr->getType()->isBooleanType()) {
- return clang::CXXBoolLiteralExpr::Create(
- sema.getASTContext(), ap_value.getInt().getBoolValue(),
- result_expr->getType(), result_expr->getExprLoc());
- }
- return clang::IntegerLiteral::Create(
- sema.getASTContext(), ap_value.getInt(), result_expr->getType(),
- result_expr->getExprLoc());
- case clang::APValue::Float:
- return clang::FloatingLiteral::Create(
- sema.getASTContext(), ap_value.getFloat(),
- /*isExact=*/true, result_expr->getType(), result_expr->getExprLoc());
- default:
- context.TODO(loc_id,
- "Unsupported: macro evaluated to a constant of type: " +
- result_expr->getType().getAsString());
- return nullptr;
- }
- }
- } // namespace Carbon::Check
|