2017-08-04 06:30:46 +08:00
|
|
|
//==- AMDGPUArgumentrUsageInfo.h - Function Arg Usage Info -------*- C++ -*-==//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2017-08-04 06:30:46 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_LIB_TARGET_AMDGPU_AMDGPUARGUMENTUSAGEINFO_H
|
|
|
|
#define LLVM_LIB_TARGET_AMDGPU_AMDGPUARGUMENTUSAGEINFO_H
|
|
|
|
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2019-07-01 21:44:46 +08:00
|
|
|
#include "llvm/CodeGen/Register.h"
|
2017-08-04 06:30:46 +08:00
|
|
|
#include "llvm/Pass.h"
|
2020-07-06 01:17:02 +08:00
|
|
|
#include "llvm/Support/LowLevelTypeImpl.h"
|
2017-08-04 06:30:46 +08:00
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
class Function;
|
|
|
|
class raw_ostream;
|
|
|
|
class TargetRegisterClass;
|
|
|
|
class TargetRegisterInfo;
|
|
|
|
|
|
|
|
struct ArgDescriptor {
|
|
|
|
private:
|
|
|
|
friend struct AMDGPUFunctionArgInfo;
|
2017-08-04 18:53:07 +08:00
|
|
|
friend class AMDGPUArgumentUsageInfo;
|
2017-08-04 06:30:46 +08:00
|
|
|
|
|
|
|
union {
|
2020-07-21 01:25:07 +08:00
|
|
|
MCRegister Reg;
|
2017-08-04 06:30:46 +08:00
|
|
|
unsigned StackOffset;
|
|
|
|
};
|
|
|
|
|
2019-06-28 09:52:13 +08:00
|
|
|
// Bitmask to locate argument within the register.
|
|
|
|
unsigned Mask;
|
|
|
|
|
2017-08-04 06:30:46 +08:00
|
|
|
bool IsStack : 1;
|
|
|
|
bool IsSet : 1;
|
|
|
|
|
|
|
|
public:
|
2020-03-12 04:13:52 +08:00
|
|
|
constexpr ArgDescriptor(unsigned Val = 0, unsigned Mask = ~0u,
|
2019-06-28 09:52:13 +08:00
|
|
|
bool IsStack = false, bool IsSet = false)
|
2019-07-01 21:44:46 +08:00
|
|
|
: Reg(Val), Mask(Mask), IsStack(IsStack), IsSet(IsSet) {}
|
2019-06-28 09:52:13 +08:00
|
|
|
|
2020-03-12 04:13:52 +08:00
|
|
|
static constexpr ArgDescriptor createRegister(Register Reg,
|
|
|
|
unsigned Mask = ~0u) {
|
2019-06-28 09:52:13 +08:00
|
|
|
return ArgDescriptor(Reg, Mask, false, true);
|
|
|
|
}
|
|
|
|
|
2020-03-12 04:13:52 +08:00
|
|
|
static constexpr ArgDescriptor createStack(unsigned Offset,
|
|
|
|
unsigned Mask = ~0u) {
|
2019-09-06 07:07:10 +08:00
|
|
|
return ArgDescriptor(Offset, Mask, true, true);
|
2017-08-04 06:30:46 +08:00
|
|
|
}
|
|
|
|
|
2020-03-12 04:13:52 +08:00
|
|
|
static constexpr ArgDescriptor createArg(const ArgDescriptor &Arg,
|
|
|
|
unsigned Mask) {
|
2019-07-01 21:44:46 +08:00
|
|
|
return ArgDescriptor(Arg.Reg, Mask, Arg.IsStack, Arg.IsSet);
|
2017-08-04 06:30:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool isSet() const {
|
|
|
|
return IsSet;
|
|
|
|
}
|
|
|
|
|
|
|
|
explicit operator bool() const {
|
|
|
|
return isSet();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isRegister() const {
|
|
|
|
return !IsStack;
|
|
|
|
}
|
|
|
|
|
2020-07-21 01:25:07 +08:00
|
|
|
MCRegister getRegister() const {
|
2017-08-04 06:30:46 +08:00
|
|
|
assert(!IsStack);
|
2019-07-01 21:44:46 +08:00
|
|
|
return Reg;
|
2017-08-04 06:30:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned getStackOffset() const {
|
|
|
|
assert(IsStack);
|
|
|
|
return StackOffset;
|
|
|
|
}
|
|
|
|
|
2019-06-28 09:52:13 +08:00
|
|
|
unsigned getMask() const {
|
|
|
|
return Mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isMasked() const {
|
|
|
|
return Mask != ~0u;
|
|
|
|
}
|
|
|
|
|
2017-08-04 06:30:46 +08:00
|
|
|
void print(raw_ostream &OS, const TargetRegisterInfo *TRI = nullptr) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline raw_ostream &operator<<(raw_ostream &OS, const ArgDescriptor &Arg) {
|
|
|
|
Arg.print(OS);
|
|
|
|
return OS;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct AMDGPUFunctionArgInfo {
|
|
|
|
enum PreloadedValue {
|
|
|
|
// SGPRS:
|
|
|
|
PRIVATE_SEGMENT_BUFFER = 0,
|
|
|
|
DISPATCH_PTR = 1,
|
|
|
|
QUEUE_PTR = 2,
|
|
|
|
KERNARG_SEGMENT_PTR = 3,
|
|
|
|
DISPATCH_ID = 4,
|
|
|
|
FLAT_SCRATCH_INIT = 5,
|
|
|
|
WORKGROUP_ID_X = 10,
|
|
|
|
WORKGROUP_ID_Y = 11,
|
|
|
|
WORKGROUP_ID_Z = 12,
|
|
|
|
PRIVATE_SEGMENT_WAVE_BYTE_OFFSET = 14,
|
|
|
|
IMPLICIT_BUFFER_PTR = 15,
|
2017-08-04 07:12:44 +08:00
|
|
|
IMPLICIT_ARG_PTR = 16,
|
2017-08-04 06:30:46 +08:00
|
|
|
|
|
|
|
// VGPRS:
|
2017-08-04 07:12:44 +08:00
|
|
|
WORKITEM_ID_X = 17,
|
|
|
|
WORKITEM_ID_Y = 18,
|
|
|
|
WORKITEM_ID_Z = 19,
|
|
|
|
FIRST_VGPR_VALUE = WORKITEM_ID_X
|
2017-08-04 06:30:46 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
// Kernel input registers setup for the HSA ABI in allocation order.
|
|
|
|
|
|
|
|
// User SGPRs in kernels
|
|
|
|
// XXX - Can these require argument spills?
|
|
|
|
ArgDescriptor PrivateSegmentBuffer;
|
|
|
|
ArgDescriptor DispatchPtr;
|
|
|
|
ArgDescriptor QueuePtr;
|
|
|
|
ArgDescriptor KernargSegmentPtr;
|
|
|
|
ArgDescriptor DispatchID;
|
|
|
|
ArgDescriptor FlatScratchInit;
|
|
|
|
ArgDescriptor PrivateSegmentSize;
|
|
|
|
|
|
|
|
// System SGPRs in kernels.
|
|
|
|
ArgDescriptor WorkGroupIDX;
|
|
|
|
ArgDescriptor WorkGroupIDY;
|
|
|
|
ArgDescriptor WorkGroupIDZ;
|
|
|
|
ArgDescriptor WorkGroupInfo;
|
|
|
|
ArgDescriptor PrivateSegmentWaveByteOffset;
|
|
|
|
|
2017-08-04 07:12:44 +08:00
|
|
|
// Pointer with offset from kernargsegmentptr to where special ABI arguments
|
|
|
|
// are passed to callable functions.
|
|
|
|
ArgDescriptor ImplicitArgPtr;
|
|
|
|
|
2017-08-04 06:30:46 +08:00
|
|
|
// Input registers for non-HSA ABI
|
2020-03-12 04:13:52 +08:00
|
|
|
ArgDescriptor ImplicitBufferPtr;
|
2017-08-04 06:30:46 +08:00
|
|
|
|
|
|
|
// VGPRs inputs. These are always v0, v1 and v2 for entry functions.
|
|
|
|
ArgDescriptor WorkItemIDX;
|
|
|
|
ArgDescriptor WorkItemIDY;
|
|
|
|
ArgDescriptor WorkItemIDZ;
|
|
|
|
|
2020-07-06 01:17:02 +08:00
|
|
|
std::tuple<const ArgDescriptor *, const TargetRegisterClass *, LLT>
|
2017-08-04 06:30:46 +08:00
|
|
|
getPreloadedValue(PreloadedValue Value) const;
|
2020-03-12 04:13:52 +08:00
|
|
|
|
|
|
|
static constexpr AMDGPUFunctionArgInfo fixedABILayout();
|
2017-08-04 06:30:46 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
class AMDGPUArgumentUsageInfo : public ImmutablePass {
|
|
|
|
private:
|
|
|
|
DenseMap<const Function *, AMDGPUFunctionArgInfo> ArgInfoMap;
|
|
|
|
|
|
|
|
public:
|
|
|
|
static char ID;
|
|
|
|
|
2020-03-17 09:07:46 +08:00
|
|
|
static const AMDGPUFunctionArgInfo ExternFunctionInfo;
|
|
|
|
static const AMDGPUFunctionArgInfo FixedABIFunctionInfo;
|
|
|
|
|
2017-08-04 06:30:46 +08:00
|
|
|
AMDGPUArgumentUsageInfo() : ImmutablePass(ID) { }
|
|
|
|
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
|
|
|
AU.setPreservesAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool doInitialization(Module &M) override;
|
|
|
|
bool doFinalization(Module &M) override;
|
|
|
|
|
|
|
|
void print(raw_ostream &OS, const Module *M = nullptr) const override;
|
|
|
|
|
|
|
|
void setFuncArgInfo(const Function &F, const AMDGPUFunctionArgInfo &ArgInfo) {
|
|
|
|
ArgInfoMap[&F] = ArgInfo;
|
|
|
|
}
|
|
|
|
|
2020-03-12 04:13:52 +08:00
|
|
|
const AMDGPUFunctionArgInfo &lookupFuncArgInfo(const Function &F) const;
|
2017-08-04 06:30:46 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace llvm
|
|
|
|
|
|
|
|
#endif
|