Files
llvm-project/clang/lib/AST/ByteCode/Primitives.h
Timm Baeder a0e42c2c47 [clang][bytecode] Add new IntegralType for function addresses (#194206)
We used to use just `::Address` for functions, which later caused
problems because we casted the pointer to `ValueDecl*` and passed it to
`Program::getOrCreateGlobal()`, which doesn't work of course.
2026-04-26 08:29:14 +02:00

84 lines
2.2 KiB
C++

//===------ Primitives.h - Types for the constexpr VM -----------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
//
// Utilities and helper functions for all primitive types:
// - Integral
// - Floating
// - Boolean
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_CLANG_AST_INTERP_PRIMITIVES_H
#define LLVM_CLANG_AST_INTERP_PRIMITIVES_H
#include "clang/AST/ComparisonCategories.h"
namespace clang {
namespace interp {
enum class IntegralKind : uint8_t {
/// Just a number, nothing else.
Number = 0,
/// A pointer to a ValueDecl.
Address,
/// A pointer to an interp::Block.
BlockAddress,
/// A pointer to a AddrLabelExpr.
LabelAddress,
/// A pointer to a FunctionDecl.
FunctionAddress,
/// Difference between two AddrLabelExpr.
AddrLabelDiff
};
/// Helper to compare two comparable types.
template <typename T> ComparisonCategoryResult Compare(const T &X, const T &Y) {
if (X < Y)
return ComparisonCategoryResult::Less;
if (X > Y)
return ComparisonCategoryResult::Greater;
return ComparisonCategoryResult::Equal;
}
template <typename T> inline bool CheckAddUB(T A, T B, T &R) {
if constexpr (std::is_signed_v<T>) {
return llvm::AddOverflow<T>(A, B, R);
} else {
R = A + B;
return false;
}
}
template <typename T> inline bool CheckSubUB(T A, T B, T &R) {
if constexpr (std::is_signed_v<T>) {
return llvm::SubOverflow<T>(A, B, R);
} else {
R = A - B;
return false;
}
}
template <typename T> inline bool CheckMulUB(T A, T B, T &R) {
if constexpr (std::is_signed_v<T>) {
return llvm::MulOverflow<T>(A, B, R);
} else if constexpr (sizeof(T) < sizeof(int)) {
// Silly integer promotion rules will convert both A and B to int,
// even it T is unsigned. Prevent that by manually casting to uint first.
R = static_cast<T>(static_cast<unsigned>(A) * static_cast<unsigned>(B));
return false;
} else {
R = A * B;
return false;
}
}
} // namespace interp
} // namespace clang
#endif