clang 20.0.0 (based on r547379) from build 12806354. Bug: http://b/379133546 Test: N/A Change-Id: I2eb8938af55d809de674be63cb30cf27e801862b Upstream-Commit: ad834e67b1105d15ef907f6255d4c96e8e733f57
279 lines
11 KiB
C++
279 lines
11 KiB
C++
//===-IslExprBuilder.h - Helper to generate code for isl AST expressions --===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef POLLY_ISL_EXPR_BUILDER_H
|
|
#define POLLY_ISL_EXPR_BUILDER_H
|
|
|
|
#include "polly/CodeGen/IRBuilder.h"
|
|
#include "polly/Support/ScopHelper.h"
|
|
#include "isl/isl-noexceptions.h"
|
|
|
|
namespace llvm {
|
|
// Provide PointerLikeTypeTraits for isl_id.
|
|
template <> struct PointerLikeTypeTraits<isl_id *> {
|
|
|
|
public:
|
|
static inline const void *getAsVoidPointer(isl_id *P) { return (void *)P; }
|
|
static inline const Region *getFromVoidPointer(void *P) {
|
|
return (Region *)P;
|
|
}
|
|
static constexpr int NumLowBitsAvailable = 0;
|
|
};
|
|
} // namespace llvm
|
|
|
|
namespace polly {
|
|
class ScopArrayInfo;
|
|
|
|
/// LLVM-IR generator for isl_ast_expr[essions]
|
|
///
|
|
/// This generator generates LLVM-IR that performs the computation described by
|
|
/// an isl_ast_expr[ession].
|
|
///
|
|
/// Example:
|
|
///
|
|
/// An isl_ast_expr[ession] can look like this:
|
|
///
|
|
/// (N + M) + 10
|
|
///
|
|
/// The IslExprBuilder could create the following LLVM-IR:
|
|
///
|
|
/// %tmp1 = add nsw i64 %N
|
|
/// %tmp2 = add nsw i64 %tmp1, %M
|
|
/// %tmp3 = add nsw i64 %tmp2, 10
|
|
///
|
|
/// The implementation of this class is mostly a mapping from isl_ast_expr
|
|
/// constructs to the corresponding LLVM-IR constructs.
|
|
///
|
|
/// The following decisions may need some explanation:
|
|
///
|
|
/// 1) Which data-type to choose
|
|
///
|
|
/// isl_ast_expr[essions] are untyped expressions that assume arbitrary
|
|
/// precision integer computations. LLVM-IR instead has fixed size integers.
|
|
/// When lowering to LLVM-IR we need to chose both the size of the data type and
|
|
/// the sign of the operations we use.
|
|
///
|
|
/// At the moment, we hardcode i64 bit signed computations. Our experience has
|
|
/// shown that 64 bit are generally large enough for the loop bounds that appear
|
|
/// in the wild. Signed computations are needed, as loop bounds may become
|
|
/// negative.
|
|
///
|
|
/// It is possible to track overflows that occurred in the generated IR. See the
|
|
/// description of @see OverflowState for more information.
|
|
///
|
|
/// FIXME: Hardcoding sizes can cause issues:
|
|
///
|
|
/// - On embedded systems and especially for high-level-synthesis 64 bit
|
|
/// computations are very costly.
|
|
///
|
|
/// The right approach is to compute the minimal necessary bitwidth and
|
|
/// signedness for each subexpression during in the isl AST generation and
|
|
/// to use this information in our IslAstGenerator. Preliminary patches are
|
|
/// available, but have not been committed yet.
|
|
///
|
|
class IslExprBuilder final {
|
|
public:
|
|
/// A map from isl_ids to llvm::Values.
|
|
typedef llvm::MapVector<isl_id *, llvm::AssertingVH<llvm::Value>> IDToValueTy;
|
|
|
|
typedef llvm::MapVector<isl_id *, const ScopArrayInfo *> IDToScopArrayInfoTy;
|
|
|
|
/// A map from isl_ids to ScopArrayInfo objects.
|
|
///
|
|
/// This map is used to obtain ScopArrayInfo objects for isl_ids which do not
|
|
/// carry a ScopArrayInfo object in their user pointer. This is useful if the
|
|
/// construction of ScopArrayInfo objects happens only after references (e.g.
|
|
/// in an AST) to an isl_id are generated and the user pointer of the isl_id
|
|
/// can not be changed any more.
|
|
///
|
|
/// This is useful for external users who just use the IslExprBuilder for
|
|
/// code generation.
|
|
IDToScopArrayInfoTy *IDToSAI = nullptr;
|
|
|
|
/// Set the isl_id to ScopArrayInfo map.
|
|
///
|
|
/// @param NewIDToSAI The new isl_id to ScopArrayInfo map to use.
|
|
void setIDToSAI(IDToScopArrayInfoTy *NewIDToSAI) { IDToSAI = NewIDToSAI; }
|
|
|
|
/// Construct an IslExprBuilder.
|
|
///
|
|
/// @param Builder The IRBuilder used to construct the
|
|
/// isl_ast_expr[ession]. The insert location of this
|
|
/// IRBuilder defines WHERE the corresponding LLVM-IR
|
|
/// is generated.
|
|
/// @param IDToValue The isl_ast_expr[ession] may reference parameters or
|
|
/// variables (identified by an isl_id). The IDTOValue map
|
|
/// specifies the LLVM-IR Values that correspond to these
|
|
/// parameters and variables.
|
|
/// @param GlobalMap A mapping from llvm::Values used in the original scop
|
|
/// region to a new set of llvm::Values.
|
|
/// @param DL DataLayout for the current Module.
|
|
/// @param SE ScalarEvolution analysis for the current function.
|
|
/// @param DT DominatorTree analysis for the current function.
|
|
/// @param LI LoopInfo analysis for the current function.
|
|
/// @param StartBlock The first basic block after the RTC.
|
|
IslExprBuilder(Scop &S, PollyIRBuilder &Builder, IDToValueTy &IDToValue,
|
|
ValueMapT &GlobalMap, const llvm::DataLayout &DL,
|
|
llvm::ScalarEvolution &SE, llvm::DominatorTree &DT,
|
|
llvm::LoopInfo &LI, llvm::BasicBlock *StartBlock);
|
|
|
|
/// Change the function that code is emitted into.
|
|
void switchGeneratedFunc(llvm::Function *GenFn, llvm::DominatorTree *GenDT,
|
|
llvm::LoopInfo *GenLI, llvm::ScalarEvolution *GenSE);
|
|
|
|
/// Create LLVM-IR for an isl_ast_expr[ession].
|
|
///
|
|
/// @param Expr The ast expression for which we generate LLVM-IR.
|
|
///
|
|
/// @return The llvm::Value* containing the result of the computation.
|
|
llvm::Value *create(__isl_take isl_ast_expr *Expr);
|
|
|
|
/// Return the largest of two types.
|
|
///
|
|
/// @param T1 The first type.
|
|
/// @param T2 The second type.
|
|
///
|
|
/// @return The largest of the two types.
|
|
llvm::Type *getWidestType(llvm::Type *T1, llvm::Type *T2);
|
|
|
|
/// Return the type with which this expression should be computed.
|
|
///
|
|
/// The type needs to be large enough to hold all possible input and all
|
|
/// possible output values.
|
|
///
|
|
/// @param Expr The expression for which to find the type.
|
|
/// @return The type with which the expression should be computed.
|
|
llvm::IntegerType *getType(__isl_keep isl_ast_expr *Expr);
|
|
|
|
/// Change if runtime overflows are tracked or not.
|
|
///
|
|
/// @param Enable Flag to enable/disable the tracking.
|
|
///
|
|
/// Note that this will reset the tracking state and that tracking is only
|
|
/// allowed if the last tracked expression dominates the current insert point.
|
|
void setTrackOverflow(bool Enable);
|
|
|
|
/// Return the current overflow status or nullptr if it is not tracked.
|
|
///
|
|
/// @return A nullptr if tracking is disabled or otherwise an i1 that has the
|
|
/// value of "0" if and only if no overflow happened since tracking
|
|
/// was enabled.
|
|
llvm::Value *getOverflowState() const;
|
|
|
|
/// Create LLVM-IR that computes the memory location of an access expression.
|
|
///
|
|
/// For a given isl_ast_expr[ession] of type isl_ast_op_access this function
|
|
/// creates IR that computes the address the access expression refers to.
|
|
///
|
|
/// @param Expr The ast expression of type isl_ast_op_access
|
|
/// for which we generate LLVM-IR.
|
|
///
|
|
/// @return A pair of the llvm::Value* containing the result of the
|
|
/// computation and the llvm::Type* it points to.
|
|
std::pair<llvm::Value *, llvm::Type *>
|
|
createAccessAddress(__isl_take isl_ast_expr *Expr);
|
|
|
|
/// Check if an @p Expr contains integer constants larger than 64 bit.
|
|
///
|
|
/// @param Expr The expression to check.
|
|
///
|
|
/// @return True if the ast expression is larger than 64 bit.
|
|
bool hasLargeInts(isl::ast_expr Expr);
|
|
|
|
private:
|
|
Scop &S;
|
|
|
|
/// Flag that will be set if an overflow occurred at runtime.
|
|
///
|
|
/// Note that this flag is by default a nullptr and if it is a nullptr
|
|
/// we will not record overflows but simply perform the computations.
|
|
/// The intended usage is as follows:
|
|
/// - If overflows in [an] expression[s] should be tracked, call
|
|
/// the setTrackOverflow(true) function.
|
|
/// - Use create(...) for all expressions that should be checked.
|
|
/// - Call getOverflowState() to get the value representing the current
|
|
/// state of the overflow flag.
|
|
/// - To stop tracking call setTrackOverflow(false).
|
|
llvm::Value *OverflowState;
|
|
|
|
PollyIRBuilder &Builder;
|
|
IDToValueTy &IDToValue;
|
|
ValueMapT &GlobalMap;
|
|
|
|
const llvm::DataLayout &DL;
|
|
llvm::ScalarEvolution &SE;
|
|
llvm::BasicBlock *StartBlock;
|
|
|
|
/// Relates to the region where the code is emitted into.
|
|
/// @{
|
|
llvm::DominatorTree *GenDT;
|
|
llvm::LoopInfo *GenLI;
|
|
llvm::ScalarEvolution *GenSE;
|
|
/// @}
|
|
|
|
llvm::Value *createOp(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpUnary(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpAccess(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpBin(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpNAry(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpSelect(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpICmp(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpBoolean(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpBooleanConditional(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createId(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createInt(__isl_take isl_ast_expr *Expr);
|
|
llvm::Value *createOpAddressOf(__isl_take isl_ast_expr *Expr);
|
|
|
|
/// Create a binary operation @p Opc and track overflows if requested.
|
|
///
|
|
/// @param OpC The binary operation that should be performed [Add/Sub/Mul].
|
|
/// @param LHS The left operand.
|
|
/// @param RHS The right operand.
|
|
/// @param Name The (base) name of the new IR operations.
|
|
///
|
|
/// @return A value that represents the result of the binary operation.
|
|
llvm::Value *createBinOp(llvm::BinaryOperator::BinaryOps Opc,
|
|
llvm::Value *LHS, llvm::Value *RHS,
|
|
const llvm::Twine &Name);
|
|
|
|
/// Create an addition and track overflows if requested.
|
|
///
|
|
/// @param LHS The left operand.
|
|
/// @param RHS The right operand.
|
|
/// @param Name The (base) name of the new IR operations.
|
|
///
|
|
/// @return A value that represents the result of the addition.
|
|
llvm::Value *createAdd(llvm::Value *LHS, llvm::Value *RHS,
|
|
const llvm::Twine &Name = "");
|
|
|
|
/// Create a subtraction and track overflows if requested.
|
|
///
|
|
/// @param LHS The left operand.
|
|
/// @param RHS The right operand.
|
|
/// @param Name The (base) name of the new IR operations.
|
|
///
|
|
/// @return A value that represents the result of the subtraction.
|
|
llvm::Value *createSub(llvm::Value *LHS, llvm::Value *RHS,
|
|
const llvm::Twine &Name = "");
|
|
|
|
/// Create a multiplication and track overflows if requested.
|
|
///
|
|
/// @param LHS The left operand.
|
|
/// @param RHS The right operand.
|
|
/// @param Name The (base) name of the new IR operations.
|
|
///
|
|
/// @return A value that represents the result of the multiplication.
|
|
llvm::Value *createMul(llvm::Value *LHS, llvm::Value *RHS,
|
|
const llvm::Twine &Name = "");
|
|
};
|
|
} // namespace polly
|
|
|
|
#endif
|