37#include "llvm/ADT/ArrayRef.h"
38#include "llvm/ADT/STLExtras.h"
39#include "llvm/ADT/SmallVector.h"
40#include "llvm/ADT/StringExtras.h"
41#include "llvm/ADT/StringRef.h"
42#include "llvm/ADT/Twine.h"
43#include "llvm/Frontend/HLSL/HLSLBinding.h"
44#include "llvm/Frontend/HLSL/RootSignatureValidations.h"
45#include "llvm/Support/Casting.h"
46#include "llvm/Support/DXILABI.h"
47#include "llvm/Support/ErrorHandling.h"
48#include "llvm/Support/FormatVariadic.h"
49#include "llvm/TargetParser/Triple.h"
64 case ResourceClass::SRV:
65 return RegisterType::SRV;
66 case ResourceClass::UAV:
67 return RegisterType::UAV;
68 case ResourceClass::CBuffer:
69 return RegisterType::CBuffer;
70 case ResourceClass::Sampler:
71 return RegisterType::Sampler;
73 llvm_unreachable(
"unexpected ResourceClass value");
82 case ResourceClass::SRV:
83 case ResourceClass::UAV:
85 case ResourceClass::CBuffer:
87 case ResourceClass::Sampler:
90 llvm_unreachable(
"unexpected ResourceClass value");
96 assert(RT !=
nullptr);
100 *RT = RegisterType::SRV;
104 *RT = RegisterType::UAV;
108 *RT = RegisterType::CBuffer;
112 *RT = RegisterType::Sampler;
116 *RT = RegisterType::C;
120 *RT = RegisterType::I;
129 case RegisterType::SRV:
131 case RegisterType::UAV:
133 case RegisterType::CBuffer:
135 case RegisterType::Sampler:
137 case RegisterType::C:
139 case RegisterType::I:
142 llvm_unreachable(
"unexpected RegisterType value");
147 case RegisterType::SRV:
148 return ResourceClass::SRV;
149 case RegisterType::UAV:
150 return ResourceClass::UAV;
151 case RegisterType::CBuffer:
152 return ResourceClass::CBuffer;
153 case RegisterType::Sampler:
154 return ResourceClass::Sampler;
155 case RegisterType::C:
156 case RegisterType::I:
160 llvm_unreachable(
"unexpected RegisterType value");
164 const auto *BT = dyn_cast<BuiltinType>(
Type);
168 return Builtin::BI__builtin_get_spirv_spec_constant_int;
171 switch (BT->getKind()) {
172 case BuiltinType::Bool:
173 return Builtin::BI__builtin_get_spirv_spec_constant_bool;
174 case BuiltinType::Short:
175 return Builtin::BI__builtin_get_spirv_spec_constant_short;
176 case BuiltinType::Int:
177 return Builtin::BI__builtin_get_spirv_spec_constant_int;
178 case BuiltinType::LongLong:
179 return Builtin::BI__builtin_get_spirv_spec_constant_longlong;
180 case BuiltinType::UShort:
181 return Builtin::BI__builtin_get_spirv_spec_constant_ushort;
182 case BuiltinType::UInt:
183 return Builtin::BI__builtin_get_spirv_spec_constant_uint;
184 case BuiltinType::ULongLong:
185 return Builtin::BI__builtin_get_spirv_spec_constant_ulonglong;
186 case BuiltinType::Half:
187 return Builtin::BI__builtin_get_spirv_spec_constant_half;
188 case BuiltinType::Float:
189 return Builtin::BI__builtin_get_spirv_spec_constant_float;
190 case BuiltinType::Double:
191 return Builtin::BI__builtin_get_spirv_spec_constant_double;
200 llvm::raw_svector_ostream OS(Buffer);
207 ResourceClass ResClass) {
209 "DeclBindingInfo already added");
215 DeclToBindingListIndex.try_emplace(VD, BindingsList.size());
216 return &BindingsList.emplace_back(VD, ResClass);
220 ResourceClass ResClass) {
221 auto Entry = DeclToBindingListIndex.find(VD);
222 if (Entry != DeclToBindingListIndex.end()) {
223 for (
unsigned Index = Entry->getSecond();
224 Index < BindingsList.size() && BindingsList[Index].Decl == VD;
226 if (BindingsList[Index].ResClass == ResClass)
227 return &BindingsList[Index];
234 return DeclToBindingListIndex.contains(VD);
246 getASTContext(), LexicalParent, CBuffer, KwLoc, Ident, IdentLoc, LBrace);
249 auto RC = CBuffer ? llvm::hlsl::ResourceClass::CBuffer
250 : llvm::hlsl::ResourceClass::SRV;
262 if (T->isArrayType() || T->isStructureType() || T->isConstantMatrixType())
269 assert(Context.getTypeSize(T) <= 64 &&
270 "Scalar bit widths larger than 64 not supported");
273 return Context.getTypeSize(T) / 8;
280 constexpr unsigned CBufferAlign = 16;
281 if (
const auto *RD = T->getAsRecordDecl()) {
283 for (
const FieldDecl *Field : RD->fields()) {
290 unsigned AlignSize = llvm::alignTo(Size, FieldAlign);
291 if ((AlignSize % CBufferAlign) + FieldSize > CBufferAlign) {
292 FieldAlign = CBufferAlign;
295 Size = llvm::alignTo(Size, FieldAlign);
302 unsigned ElementCount = AT->getSize().getZExtValue();
303 if (ElementCount == 0)
306 unsigned ElementSize =
308 unsigned AlignedElementSize = llvm::alignTo(ElementSize, CBufferAlign);
309 return AlignedElementSize * (ElementCount - 1) + ElementSize;
313 unsigned ElementCount = VT->getNumElements();
314 unsigned ElementSize =
316 return ElementSize * ElementCount;
319 return Context.getTypeSize(T) / 8;
330 bool HasPackOffset =
false;
331 bool HasNonPackOffset =
false;
333 VarDecl *Var = dyn_cast<VarDecl>(Field);
336 if (Field->hasAttr<HLSLPackOffsetAttr>()) {
337 PackOffsetVec.emplace_back(Var, Field->
getAttr<HLSLPackOffsetAttr>());
338 HasPackOffset =
true;
340 HasNonPackOffset =
true;
347 if (HasNonPackOffset)
354 std::sort(PackOffsetVec.begin(), PackOffsetVec.end(),
355 [](
const std::pair<VarDecl *, HLSLPackOffsetAttr *> &LHS,
356 const std::pair<VarDecl *, HLSLPackOffsetAttr *> &RHS) {
357 return LHS.second->getOffsetInBytes() <
358 RHS.second->getOffsetInBytes();
360 for (
unsigned i = 0; i < PackOffsetVec.size() - 1; i++) {
361 VarDecl *Var = PackOffsetVec[i].first;
362 HLSLPackOffsetAttr *
Attr = PackOffsetVec[i].second;
364 unsigned Begin =
Attr->getOffsetInBytes();
365 unsigned End = Begin + Size;
366 unsigned NextBegin = PackOffsetVec[i + 1].second->getOffsetInBytes();
367 if (End > NextBegin) {
368 VarDecl *NextVar = PackOffsetVec[i + 1].first;
380 CAT = dyn_cast<ConstantArrayType>(
382 return CAT !=
nullptr;
393static const HLSLAttributedResourceType *
396 "expected array of resource records");
398 while (
const ArrayType *AT = dyn_cast<ArrayType>(Ty))
400 return HLSLAttributedResourceType::findHandleTypeOnResource(Ty);
403static const HLSLAttributedResourceType *
417 return RD->isEmpty();
446 Base.getType()->castAsCXXRecordDecl()))
457 assert(RD ==
nullptr &&
458 "there should be at most 1 record by a given name in a scope");
475 Name.append(NameBaseII->
getName());
482 size_t NameLength = Name.size();
491 Name.append(llvm::Twine(suffix).str());
492 II = &AST.
Idents.
get(Name, tok::TokenKind::identifier);
499 Name.truncate(NameLength);
514 if (
const auto *CAT = dyn_cast<ConstantArrayType>(Ty)) {
516 S, CAT->getElementType()->getUnqualifiedDesugaredType());
521 CAT->getSizeModifier(),
522 CAT->getIndexTypeCVRQualifiers())
571 "struct is already HLSL buffer compatible");
585 LS->
addAttr(PackedAttr::CreateImplicit(AST));
589 if (
unsigned NumBases = StructDecl->
getNumBases()) {
590 assert(NumBases == 1 &&
"HLSL supports only one base type");
640 LS->
addAttr(PackedAttr::CreateImplicit(AST));
645 VarDecl *VD = dyn_cast<VarDecl>(D);
661 "host layout field for $Globals decl failed to be created");
680 HLSLResourceBindingAttr::CreateImplicit(S.
getASTContext(),
"",
"0", {});
681 Attr->setBinding(RT, std::nullopt, 0);
682 Attr->setImplicitBindingOrderID(ImplicitBindingOrderID);
689 BufDecl->setRBraceLoc(RBrace);
706 BufDecl->isCBuffer() ? RegisterType::CBuffer
716 int X,
int Y,
int Z) {
717 if (HLSLNumThreadsAttr *NT = D->
getAttr<HLSLNumThreadsAttr>()) {
718 if (NT->getX() !=
X || NT->getY() != Y || NT->getZ() != Z) {
719 Diag(NT->getLocation(), diag::err_hlsl_attribute_param_mismatch) << AL;
720 Diag(AL.
getLoc(), diag::note_conflicting_attribute);
730 int Min,
int Max,
int Preferred,
731 int SpelledArgsCount) {
732 if (HLSLWaveSizeAttr *WS = D->
getAttr<HLSLWaveSizeAttr>()) {
733 if (WS->getMin() !=
Min || WS->getMax() !=
Max ||
734 WS->getPreferred() != Preferred ||
735 WS->getSpelledArgsCount() != SpelledArgsCount) {
736 Diag(WS->getLocation(), diag::err_hlsl_attribute_param_mismatch) << AL;
737 Diag(AL.
getLoc(), diag::note_conflicting_attribute);
743 Result->setSpelledArgsCount(SpelledArgsCount);
747HLSLVkConstantIdAttr *
753 Diag(AL.
getLoc(), diag::warn_attribute_ignored) << AL;
761 Diag(VD->getLocation(), diag::err_specialization_const);
765 if (!VD->getType().isConstQualified()) {
766 Diag(VD->getLocation(), diag::err_specialization_const);
770 if (HLSLVkConstantIdAttr *CI = D->
getAttr<HLSLVkConstantIdAttr>()) {
771 if (CI->getId() != Id) {
772 Diag(CI->getLocation(), diag::err_hlsl_attribute_param_mismatch) << AL;
773 Diag(AL.
getLoc(), diag::note_conflicting_attribute);
778 HLSLVkConstantIdAttr *
Result =
785 llvm::Triple::EnvironmentType ShaderType) {
786 if (HLSLShaderAttr *NT = D->
getAttr<HLSLShaderAttr>()) {
787 if (NT->getType() != ShaderType) {
788 Diag(NT->getLocation(), diag::err_hlsl_attribute_param_mismatch) << AL;
789 Diag(AL.
getLoc(), diag::note_conflicting_attribute);
793 return HLSLShaderAttr::Create(
getASTContext(), ShaderType, AL);
796HLSLParamModifierAttr *
798 HLSLParamModifierAttr::Spelling Spelling) {
801 if (HLSLParamModifierAttr *PA = D->
getAttr<HLSLParamModifierAttr>()) {
802 if ((PA->isIn() && Spelling == HLSLParamModifierAttr::Keyword_out) ||
803 (PA->isOut() && Spelling == HLSLParamModifierAttr::Keyword_in)) {
804 D->
dropAttr<HLSLParamModifierAttr>();
806 return HLSLParamModifierAttr::Create(
808 HLSLParamModifierAttr::Keyword_inout);
810 Diag(AL.
getLoc(), diag::err_hlsl_duplicate_parameter_modifier) << AL;
811 Diag(PA->getLocation(), diag::note_conflicting_attribute);
837 if (HLSLShaderAttr::isValidShaderType(Env) && Env != llvm::Triple::Library) {
838 if (
const auto *Shader = FD->
getAttr<HLSLShaderAttr>()) {
841 if (Shader->getType() != Env) {
842 Diag(Shader->getLocation(), diag::err_hlsl_entry_shader_attr_mismatch)
854 case llvm::Triple::UnknownEnvironment:
855 case llvm::Triple::Library:
857 case llvm::Triple::RootSignature:
858 llvm_unreachable(
"rootsig environment has no functions");
860 llvm_unreachable(
"Unhandled environment in triple");
866 HLSLAppliedSemanticAttr *Semantic,
871 const auto *ShaderAttr = FD->
getAttr<HLSLShaderAttr>();
872 assert(ShaderAttr &&
"Entry point has no shader attribute");
873 llvm::Triple::EnvironmentType ST = ShaderAttr->getType();
874 auto SemanticName = Semantic->getSemanticName().upper();
879 if (SemanticName ==
"SV_POSITION") {
880 return (ST == llvm::Triple::Vertex && !IsInput) ||
881 (ST == llvm::Triple::Pixel && IsInput);
883 if (SemanticName ==
"SV_VERTEXID")
889bool SemaHLSL::determineActiveSemanticOnScalar(
FunctionDecl *FD,
892 SemanticInfo &ActiveSemantic,
893 SemaHLSL::SemanticContext &SC) {
894 if (ActiveSemantic.Semantic ==
nullptr) {
895 ActiveSemantic.Semantic = D->
getAttr<HLSLParsedSemanticAttr>();
896 if (ActiveSemantic.Semantic)
897 ActiveSemantic.Index = ActiveSemantic.Semantic->getSemanticIndex();
900 if (!ActiveSemantic.Semantic) {
906 HLSLAppliedSemanticAttr(
getASTContext(), *ActiveSemantic.Semantic,
907 ActiveSemantic.Semantic->getAttrName()->getName(),
908 ActiveSemantic.Index.value_or(0));
912 checkSemanticAnnotation(FD, D, A, SC);
913 OutputDecl->addAttr(A);
915 unsigned Location = ActiveSemantic.Index.value_or(0);
918 SC.CurrentIOType & IOType::In)) {
919 bool HasVkLocation =
false;
920 if (
auto *A = D->getAttr<HLSLVkLocationAttr>()) {
921 HasVkLocation = true;
922 Location = A->getLocation();
925 if (SC.UsesExplicitVkLocations.value_or(HasVkLocation) != HasVkLocation) {
926 Diag(D->getLocation(), diag::err_hlsl_semantic_partial_explicit_indexing);
929 SC.UsesExplicitVkLocations = HasVkLocation;
932 const ConstantArrayType *AT = dyn_cast<ConstantArrayType>(D->getType());
933 unsigned ElementCount = AT ? AT->
getZExtSize() : 1;
934 ActiveSemantic.Index = Location + ElementCount;
936 Twine BaseName = Twine(ActiveSemantic.Semantic->getAttrName()->getName());
937 for (
unsigned I = 0; I < ElementCount; ++I) {
938 Twine VariableName = BaseName.concat(Twine(Location + I));
940 auto [_, Inserted] = SC.ActiveSemantics.insert(VariableName.str());
942 Diag(D->getLocation(), diag::err_hlsl_semantic_index_overlap)
943 << VariableName.str();
954 SemanticInfo &ActiveSemantic,
955 SemaHLSL::SemanticContext &SC) {
956 if (ActiveSemantic.Semantic ==
nullptr) {
957 ActiveSemantic.Semantic = D->
getAttr<HLSLParsedSemanticAttr>();
958 if (ActiveSemantic.Semantic)
959 ActiveSemantic.Index = ActiveSemantic.Semantic->getSemanticIndex();
965 const RecordType *RT = dyn_cast<RecordType>(T);
967 return determineActiveSemanticOnScalar(FD, OutputDecl, D, ActiveSemantic,
970 const RecordDecl *RD = RT->getDecl();
971 for (FieldDecl *Field : RD->
fields()) {
972 SemanticInfo Info = ActiveSemantic;
973 if (!determineActiveSemantic(FD, OutputDecl, Field, Info, SC)) {
974 Diag(
Field->getLocation(), diag::note_hlsl_semantic_used_here) <<
Field;
977 if (ActiveSemantic.Semantic)
978 ActiveSemantic = Info;
985 const auto *ShaderAttr = FD->
getAttr<HLSLShaderAttr>();
986 assert(ShaderAttr &&
"Entry point has no shader attribute");
987 llvm::Triple::EnvironmentType ST = ShaderAttr->getType();
991 case llvm::Triple::Pixel:
992 case llvm::Triple::Vertex:
993 case llvm::Triple::Geometry:
994 case llvm::Triple::Hull:
995 case llvm::Triple::Domain:
996 case llvm::Triple::RayGeneration:
997 case llvm::Triple::Intersection:
998 case llvm::Triple::AnyHit:
999 case llvm::Triple::ClosestHit:
1000 case llvm::Triple::Miss:
1001 case llvm::Triple::Callable:
1002 if (
const auto *NT = FD->
getAttr<HLSLNumThreadsAttr>()) {
1003 diagnoseAttrStageMismatch(NT, ST,
1004 {llvm::Triple::Compute,
1005 llvm::Triple::Amplification,
1006 llvm::Triple::Mesh});
1009 if (
const auto *WS = FD->
getAttr<HLSLWaveSizeAttr>()) {
1010 diagnoseAttrStageMismatch(WS, ST,
1011 {llvm::Triple::Compute,
1012 llvm::Triple::Amplification,
1013 llvm::Triple::Mesh});
1018 case llvm::Triple::Compute:
1019 case llvm::Triple::Amplification:
1020 case llvm::Triple::Mesh:
1021 if (!FD->
hasAttr<HLSLNumThreadsAttr>()) {
1023 << llvm::Triple::getEnvironmentTypeName(ST);
1026 if (
const auto *WS = FD->
getAttr<HLSLWaveSizeAttr>()) {
1028 Diag(WS->getLocation(), diag::warn_hlsl_wavesize_unsupported_spirv);
1029 }
else if (Ver < VersionTuple(6, 6)) {
1030 Diag(WS->getLocation(), diag::err_hlsl_attribute_in_wrong_shader_model)
1033 }
else if (WS->getSpelledArgsCount() > 1 && Ver < VersionTuple(6, 8)) {
1036 diag::err_hlsl_attribute_number_arguments_insufficient_shader_model)
1037 << WS << WS->getSpelledArgsCount() <<
"6.8";
1042 case llvm::Triple::RootSignature:
1043 llvm_unreachable(
"rootsig environment has no function entry point");
1045 llvm_unreachable(
"Unhandled environment in triple");
1048 SemaHLSL::SemanticContext InputSC = {};
1049 InputSC.CurrentIOType = IOType::In;
1052 SemanticInfo ActiveSemantic;
1053 ActiveSemantic.Semantic = Param->getAttr<HLSLParsedSemanticAttr>();
1054 if (ActiveSemantic.Semantic)
1055 ActiveSemantic.Index = ActiveSemantic.Semantic->getSemanticIndex();
1058 if (!determineActiveSemantic(FD, Param, Param, ActiveSemantic, InputSC)) {
1059 Diag(Param->getLocation(), diag::note_previous_decl) << Param;
1064 SemanticInfo ActiveSemantic;
1065 SemaHLSL::SemanticContext OutputSC = {};
1066 OutputSC.CurrentIOType = IOType::Out;
1067 ActiveSemantic.Semantic = FD->
getAttr<HLSLParsedSemanticAttr>();
1068 if (ActiveSemantic.Semantic)
1069 ActiveSemantic.Index = ActiveSemantic.Semantic->getSemanticIndex();
1071 determineActiveSemantic(FD, FD, FD, ActiveSemantic, OutputSC);
1074void SemaHLSL::checkSemanticAnnotation(
1076 const HLSLAppliedSemanticAttr *SemanticAttr,
const SemanticContext &SC) {
1077 auto *ShaderAttr = EntryPoint->
getAttr<HLSLShaderAttr>();
1078 assert(ShaderAttr &&
"Entry point has no shader attribute");
1079 llvm::Triple::EnvironmentType ST = ShaderAttr->getType();
1081 auto SemanticName = SemanticAttr->getSemanticName().upper();
1082 if (SemanticName ==
"SV_DISPATCHTHREADID" ||
1083 SemanticName ==
"SV_GROUPINDEX" || SemanticName ==
"SV_GROUPTHREADID" ||
1084 SemanticName ==
"SV_GROUPID") {
1086 if (ST != llvm::Triple::Compute)
1087 diagnoseSemanticStageMismatch(SemanticAttr, ST, SC.CurrentIOType,
1088 {{llvm::Triple::Compute, IOType::In}});
1090 if (SemanticAttr->getSemanticIndex() != 0) {
1091 std::string PrettyName =
1092 "'" + SemanticAttr->getSemanticName().str() +
"'";
1093 Diag(SemanticAttr->getLoc(),
1094 diag::err_hlsl_semantic_indexing_not_supported)
1100 if (SemanticName ==
"SV_POSITION") {
1103 diagnoseSemanticStageMismatch(SemanticAttr, ST, SC.CurrentIOType,
1104 {{llvm::Triple::Vertex, IOType::InOut},
1105 {llvm::Triple::Pixel, IOType::In}});
1108 if (SemanticName ==
"SV_VERTEXID") {
1109 diagnoseSemanticStageMismatch(SemanticAttr, ST, SC.CurrentIOType,
1110 {{llvm::Triple::Vertex, IOType::In}});
1114 if (SemanticName ==
"SV_TARGET") {
1115 diagnoseSemanticStageMismatch(SemanticAttr, ST, SC.CurrentIOType,
1116 {{llvm::Triple::Pixel, IOType::Out}});
1122 if (SemanticAttr->getAttrName()->getName().starts_with_insensitive(
"SV_"))
1123 llvm_unreachable(
"Unknown SemanticAttr");
1126void SemaHLSL::diagnoseAttrStageMismatch(
1127 const Attr *A, llvm::Triple::EnvironmentType Stage,
1128 std::initializer_list<llvm::Triple::EnvironmentType> AllowedStages) {
1129 SmallVector<StringRef, 8> StageStrings;
1130 llvm::transform(AllowedStages, std::back_inserter(StageStrings),
1131 [](llvm::Triple::EnvironmentType ST) {
1133 HLSLShaderAttr::ConvertEnvironmentTypeToStr(ST));
1135 Diag(A->
getLoc(), diag::err_hlsl_attr_unsupported_in_stage)
1136 << A->
getAttrName() << llvm::Triple::getEnvironmentTypeName(Stage)
1137 << (AllowedStages.size() != 1) <<
join(StageStrings,
", ");
1140void SemaHLSL::diagnoseSemanticStageMismatch(
1141 const Attr *A, llvm::Triple::EnvironmentType Stage, IOType CurrentIOType,
1142 std::initializer_list<SemanticStageInfo> Allowed) {
1144 for (
auto &Case : Allowed) {
1145 if (Case.Stage != Stage)
1148 if (CurrentIOType & Case.AllowedIOTypesMask)
1151 SmallVector<std::string, 8> ValidCases;
1153 Allowed, std::back_inserter(ValidCases), [](SemanticStageInfo Case) {
1154 SmallVector<std::string, 2> ValidType;
1155 if (Case.AllowedIOTypesMask & IOType::In)
1156 ValidType.push_back(
"input");
1157 if (Case.AllowedIOTypesMask & IOType::Out)
1158 ValidType.push_back(
"output");
1160 HLSLShaderAttr::ConvertEnvironmentTypeToStr(Case.Stage)) +
1161 " " +
join(ValidType,
"/");
1163 Diag(A->
getLoc(), diag::err_hlsl_semantic_unsupported_iotype_for_stage)
1164 << A->
getAttrName() << (CurrentIOType & IOType::In ?
"input" :
"output")
1165 << llvm::Triple::getEnvironmentTypeName(Case.Stage)
1166 <<
join(ValidCases,
", ");
1170 SmallVector<StringRef, 8> StageStrings;
1172 Allowed, std::back_inserter(StageStrings), [](SemanticStageInfo Case) {
1174 HLSLShaderAttr::ConvertEnvironmentTypeToStr(Case.Stage));
1177 Diag(A->
getLoc(), diag::err_hlsl_attr_unsupported_in_stage)
1178 << A->
getAttrName() << llvm::Triple::getEnvironmentTypeName(Stage)
1179 << (Allowed.size() != 1) <<
join(StageStrings,
", ");
1182template <CastKind Kind>
1185 Ty = VTy->getElementType();
1190template <CastKind Kind>
1202 if (LHSFloat && RHSFloat) {
1230 if (LHSSigned == RHSSigned) {
1231 if (IsCompAssign || IntOrder >= 0)
1239 if (IntOrder != (LHSSigned ? 1 : -1)) {
1240 if (IsCompAssign || RHSSigned)
1248 if (Ctx.getIntWidth(LElTy) != Ctx.getIntWidth(RElTy)) {
1249 if (IsCompAssign || LHSSigned)
1265 QualType ElTy = Ctx.getCorrespondingUnsignedType(LHSSigned ? LElTy : RElTy);
1266 QualType NewTy = Ctx.getExtVectorType(
1276 return CK_FloatingCast;
1278 return CK_IntegralCast;
1280 return CK_IntegralToFloating;
1282 return CK_FloatingToIntegral;
1288 bool IsCompAssign) {
1295 if (!LVecTy && IsCompAssign) {
1297 RHS =
SemaRef.ImpCastExprToType(RHS.
get(), RElTy, CK_HLSLVectorTruncation);
1299 if (Ctx.hasSameUnqualifiedType(LHSType, RHSType))
1301 RHS =
SemaRef.ImpCastExprToType(RHS.
get(), LHSType,
1306 unsigned EndSz = std::numeric_limits<unsigned>::max();
1309 LSz = EndSz = LVecTy->getNumElements();
1312 assert(EndSz != std::numeric_limits<unsigned>::max() &&
1313 "one of the above should have had a value");
1317 if (IsCompAssign && LSz != EndSz) {
1319 diag::err_hlsl_vector_compound_assignment_truncation)
1320 << LHSType << RHSType;
1326 if (!IsCompAssign && LVecTy && LVecTy->getNumElements() > EndSz)
1331 if (!IsCompAssign && !LVecTy)
1335 if (Ctx.hasSameUnqualifiedType(LHSType, RHSType))
1336 return Ctx.getCommonSugaredType(LHSType, RHSType);
1344 LElTy, RElTy, IsCompAssign);
1347 "HLSL Vectors can only contain integer or floating point types");
1349 LElTy, RElTy, IsCompAssign);
1354 assert((Opc == BO_LOr || Opc == BO_LAnd) &&
1355 "Called with non-logical operator");
1357 llvm::raw_svector_ostream OS(Buff);
1359 StringRef NewFnName = Opc == BO_LOr ?
"or" :
"and";
1360 OS << NewFnName <<
"(";
1370std::pair<IdentifierInfo *, bool>
1373 std::string IdStr =
"__hlsl_rootsig_decl_" + std::to_string(Hash);
1380 return {DeclIdent,
Found};
1391 for (
auto &RootSigElement : RootElements)
1392 Elements.push_back(RootSigElement.getElement());
1396 DeclIdent,
SemaRef.getLangOpts().HLSLRootSigVer, Elements);
1398 SignatureDecl->setImplicit();
1404 if (RootSigOverrideIdent) {
1407 if (
SemaRef.LookupQualifiedName(R, DC))
1408 return dyn_cast<HLSLRootSignatureDecl>(R.getFoundDecl());
1416struct PerVisibilityBindingChecker {
1419 std::array<llvm::hlsl::BindingInfoBuilder, 8> Builders;
1423 llvm::dxbc::ShaderVisibility Vis;
1428 PerVisibilityBindingChecker(
SemaHLSL *S) : S(S) {}
1430 void trackBinding(llvm::dxbc::ShaderVisibility
Visibility,
1431 llvm::dxil::ResourceClass RC,
uint32_t Space,
1433 const hlsl::RootSignatureElement *Elem) {
1435 assert(BuilderIndex < Builders.size() &&
1436 "Not enough builders for visibility type");
1437 Builders[BuilderIndex].trackBinding(RC, Space, LowerBound, UpperBound,
1438 static_cast<const void *
>(Elem));
1440 static_assert(llvm::to_underlying(llvm::dxbc::ShaderVisibility::All) == 0,
1441 "'All' visibility must come first");
1442 if (
Visibility == llvm::dxbc::ShaderVisibility::All)
1443 for (
size_t I = 1, E = Builders.size(); I < E; ++I)
1444 Builders[I].trackBinding(RC, Space, LowerBound, UpperBound,
1445 static_cast<const void *
>(Elem));
1447 ElemInfoMap.push_back({Elem,
Visibility,
false});
1450 ElemInfo &
getInfo(
const hlsl::RootSignatureElement *Elem) {
1451 auto It = llvm::lower_bound(
1453 [](
const auto &LHS,
const auto &RHS) {
return LHS.Elem < RHS; });
1454 assert(It->Elem == Elem &&
"Element not in map");
1458 bool checkOverlap() {
1459 llvm::sort(ElemInfoMap, [](
const auto &LHS,
const auto &RHS) {
1460 return LHS.Elem < RHS.Elem;
1463 bool HadOverlap =
false;
1465 using llvm::hlsl::BindingInfoBuilder;
1466 auto ReportOverlap = [
this,
1467 &HadOverlap](
const BindingInfoBuilder &Builder,
1468 const llvm::hlsl::Binding &Reported) {
1472 static_cast<const hlsl::RootSignatureElement *
>(Reported.Cookie);
1473 const llvm::hlsl::Binding &
Previous = Builder.findOverlapping(Reported);
1474 const auto *PrevElem =
1475 static_cast<const hlsl::RootSignatureElement *
>(
Previous.Cookie);
1477 ElemInfo &Info =
getInfo(Elem);
1482 Info.Diagnosed =
true;
1484 ElemInfo &PrevInfo =
getInfo(PrevElem);
1485 llvm::dxbc::ShaderVisibility CommonVis =
1486 Info.Vis == llvm::dxbc::ShaderVisibility::All ? PrevInfo.Vis
1489 this->S->
Diag(Elem->
getLocation(), diag::err_hlsl_resource_range_overlap)
1490 << llvm::to_underlying(Reported.RC) << Reported.LowerBound
1491 << Reported.isUnbounded() << Reported.UpperBound
1496 this->S->
Diag(PrevElem->getLocation(),
1497 diag::note_hlsl_resource_range_here);
1500 for (BindingInfoBuilder &Builder : Builders)
1501 Builder.calculateBindingInfo(ReportOverlap);
1521 bool HadError =
false;
1522 auto ReportError = [
this, &HadError](
SourceLocation Loc, uint32_t LowerBound,
1523 uint32_t UpperBound) {
1525 this->
Diag(Loc, diag::err_hlsl_invalid_rootsig_value)
1526 << LowerBound << UpperBound;
1533 this->
Diag(Loc, diag::err_hlsl_invalid_rootsig_value)
1534 << llvm::formatv(
"{0:f}", LowerBound).sstr<6>()
1535 << llvm::formatv(
"{0:f}", UpperBound).sstr<6>();
1538 auto VerifyRegister = [ReportError](
SourceLocation Loc, uint32_t Register) {
1539 if (!llvm::hlsl::rootsig::verifyRegisterValue(Register))
1540 ReportError(Loc, 0, 0xfffffffe);
1543 auto VerifySpace = [ReportError](
SourceLocation Loc, uint32_t Space) {
1544 if (!llvm::hlsl::rootsig::verifyRegisterSpace(Space))
1545 ReportError(Loc, 0, 0xffffffef);
1548 const uint32_t Version =
1549 llvm::to_underlying(
SemaRef.getLangOpts().HLSLRootSigVer);
1550 const uint32_t VersionEnum = Version - 1;
1551 auto ReportFlagError = [
this, &HadError, VersionEnum](
SourceLocation Loc) {
1553 this->
Diag(Loc, diag::err_hlsl_invalid_rootsig_flag)
1560 const llvm::hlsl::rootsig::RootElement &Elem = RootSigElem.
getElement();
1561 if (
const auto *Descriptor =
1562 std::get_if<llvm::hlsl::rootsig::RootDescriptor>(&Elem)) {
1563 VerifyRegister(Loc, Descriptor->Reg.Number);
1564 VerifySpace(Loc, Descriptor->Space);
1566 if (!llvm::hlsl::rootsig::verifyRootDescriptorFlag(Version,
1568 ReportFlagError(Loc);
1569 }
else if (
const auto *Constants =
1570 std::get_if<llvm::hlsl::rootsig::RootConstants>(&Elem)) {
1571 VerifyRegister(Loc, Constants->Reg.Number);
1572 VerifySpace(Loc, Constants->Space);
1573 }
else if (
const auto *Sampler =
1574 std::get_if<llvm::hlsl::rootsig::StaticSampler>(&Elem)) {
1575 VerifyRegister(Loc, Sampler->Reg.Number);
1576 VerifySpace(Loc, Sampler->Space);
1579 "By construction, parseFloatParam can't produce a NaN from a "
1580 "float_literal token");
1582 if (!llvm::hlsl::rootsig::verifyMaxAnisotropy(Sampler->MaxAnisotropy))
1583 ReportError(Loc, 0, 16);
1584 if (!llvm::hlsl::rootsig::verifyMipLODBias(Sampler->MipLODBias))
1585 ReportFloatError(Loc, -16.f, 15.99f);
1586 }
else if (
const auto *Clause =
1587 std::get_if<llvm::hlsl::rootsig::DescriptorTableClause>(
1589 VerifyRegister(Loc, Clause->Reg.Number);
1590 VerifySpace(Loc, Clause->Space);
1592 if (!llvm::hlsl::rootsig::verifyNumDescriptors(Clause->NumDescriptors)) {
1596 ReportError(Loc, 1, 0xfffffffe);
1599 if (!llvm::hlsl::rootsig::verifyDescriptorRangeFlag(Version, Clause->Type,
1601 ReportFlagError(Loc);
1605 PerVisibilityBindingChecker BindingChecker(
this);
1606 SmallVector<std::pair<
const llvm::hlsl::rootsig::DescriptorTableClause *,
1611 const llvm::hlsl::rootsig::RootElement &Elem = RootSigElem.
getElement();
1612 if (
const auto *Descriptor =
1613 std::get_if<llvm::hlsl::rootsig::RootDescriptor>(&Elem)) {
1614 uint32_t LowerBound(Descriptor->Reg.Number);
1615 uint32_t UpperBound(LowerBound);
1617 BindingChecker.trackBinding(
1618 Descriptor->Visibility,
1619 static_cast<llvm::dxil::ResourceClass
>(Descriptor->Type),
1620 Descriptor->Space, LowerBound, UpperBound, &RootSigElem);
1621 }
else if (
const auto *Constants =
1622 std::get_if<llvm::hlsl::rootsig::RootConstants>(&Elem)) {
1623 uint32_t LowerBound(Constants->Reg.Number);
1624 uint32_t UpperBound(LowerBound);
1626 BindingChecker.trackBinding(
1627 Constants->Visibility, llvm::dxil::ResourceClass::CBuffer,
1628 Constants->Space, LowerBound, UpperBound, &RootSigElem);
1629 }
else if (
const auto *Sampler =
1630 std::get_if<llvm::hlsl::rootsig::StaticSampler>(&Elem)) {
1631 uint32_t LowerBound(Sampler->Reg.Number);
1632 uint32_t UpperBound(LowerBound);
1634 BindingChecker.trackBinding(
1635 Sampler->Visibility, llvm::dxil::ResourceClass::Sampler,
1636 Sampler->Space, LowerBound, UpperBound, &RootSigElem);
1637 }
else if (
const auto *Clause =
1638 std::get_if<llvm::hlsl::rootsig::DescriptorTableClause>(
1641 UnboundClauses.emplace_back(Clause, &RootSigElem);
1642 }
else if (
const auto *Table =
1643 std::get_if<llvm::hlsl::rootsig::DescriptorTable>(&Elem)) {
1644 assert(UnboundClauses.size() == Table->NumClauses &&
1645 "Number of unbound elements must match the number of clauses");
1646 bool HasAnySampler =
false;
1647 bool HasAnyNonSampler =
false;
1648 uint64_t Offset = 0;
1649 bool IsPrevUnbound =
false;
1650 for (
const auto &[Clause, ClauseElem] : UnboundClauses) {
1652 if (Clause->Type == llvm::dxil::ResourceClass::Sampler)
1653 HasAnySampler =
true;
1655 HasAnyNonSampler =
true;
1657 if (HasAnySampler && HasAnyNonSampler)
1658 Diag(Loc, diag::err_hlsl_invalid_mixed_resources);
1663 if (Clause->NumDescriptors == 0)
1667 Clause->Offset == llvm::hlsl::rootsig::DescriptorTableOffsetAppend;
1669 Offset = Clause->Offset;
1671 uint64_t RangeBound = llvm::hlsl::rootsig::computeRangeBound(
1672 Offset, Clause->NumDescriptors);
1674 if (IsPrevUnbound && IsAppending)
1675 Diag(Loc, diag::err_hlsl_appending_onto_unbound);
1676 else if (!llvm::hlsl::rootsig::verifyNoOverflowedOffset(RangeBound))
1677 Diag(Loc, diag::err_hlsl_offset_overflow) << Offset << RangeBound;
1680 Offset = RangeBound + 1;
1681 IsPrevUnbound = Clause->NumDescriptors ==
1682 llvm::hlsl::rootsig::NumDescriptorsUnbounded;
1685 uint32_t LowerBound(Clause->Reg.Number);
1686 uint32_t UpperBound = llvm::hlsl::rootsig::computeRangeBound(
1687 LowerBound, Clause->NumDescriptors);
1689 BindingChecker.trackBinding(
1691 static_cast<llvm::dxil::ResourceClass
>(Clause->Type), Clause->Space,
1692 LowerBound, UpperBound, ClauseElem);
1694 UnboundClauses.clear();
1698 return BindingChecker.checkOverlap();
1703 Diag(AL.
getLoc(), diag::err_attribute_wrong_number_arguments) << AL << 1;
1708 if (
auto *RS = D->
getAttr<RootSignatureAttr>()) {
1709 if (RS->getSignatureIdent() != Ident) {
1710 Diag(AL.
getLoc(), diag::err_disallowed_duplicate_attribute) << RS;
1714 Diag(AL.
getLoc(), diag::warn_duplicate_attribute_exact) << RS;
1720 if (
auto *SignatureDecl =
1721 dyn_cast<HLSLRootSignatureDecl>(R.getFoundDecl())) {
1728 llvm::VersionTuple SMVersion =
1733 uint32_t ZMax = 1024;
1734 uint32_t ThreadMax = 1024;
1735 if (IsDXIL && SMVersion.getMajor() <= 4) {
1738 }
else if (IsDXIL && SMVersion.getMajor() == 5) {
1748 diag::err_hlsl_numthreads_argument_oor)
1757 diag::err_hlsl_numthreads_argument_oor)
1766 diag::err_hlsl_numthreads_argument_oor)
1771 if (
X * Y * Z > ThreadMax) {
1772 Diag(AL.
getLoc(), diag::err_hlsl_numthreads_invalid) << ThreadMax;
1789 if (SpelledArgsCount == 0 || SpelledArgsCount > 3)
1797 if (SpelledArgsCount > 1 &&
1801 uint32_t Preferred = 0;
1802 if (SpelledArgsCount > 2 &&
1806 if (SpelledArgsCount > 2) {
1809 diag::err_attribute_power_of_two_in_range)
1810 << AL << llvm::dxil::MinWaveSize << llvm::dxil::MaxWaveSize
1815 if (Preferred < Min || Preferred >
Max) {
1817 diag::err_attribute_power_of_two_in_range)
1818 << AL <<
Min <<
Max << Preferred;
1821 }
else if (SpelledArgsCount > 1) {
1824 diag::err_attribute_power_of_two_in_range)
1825 << AL << llvm::dxil::MinWaveSize << llvm::dxil::MaxWaveSize <<
Max;
1829 Diag(AL.
getLoc(), diag::err_attribute_argument_invalid) << AL << 1;
1832 Diag(AL.
getLoc(), diag::warn_attr_min_eq_max) << AL;
1837 diag::err_attribute_power_of_two_in_range)
1838 << AL << llvm::dxil::MinWaveSize << llvm::dxil::MaxWaveSize <<
Min;
1843 HLSLWaveSizeAttr *NewAttr =
1880 uint32_t Binding = 0;
1904 if (!T->hasUnsignedIntegerRepresentation() ||
1905 (VT && VT->getNumElements() > 3)) {
1906 Diag(AL.
getLoc(), diag::err_hlsl_attr_invalid_type)
1907 << AL <<
"uint/uint2/uint3";
1916 if (!T->hasFloatingRepresentation() || (VT && VT->getNumElements() > 4)) {
1917 Diag(AL.
getLoc(), diag::err_hlsl_attr_invalid_type)
1918 << AL <<
"float/float1/float2/float3/float4";
1926 std::optional<unsigned> Index) {
1930 QualType ValueType = VD->getType();
1931 if (
auto *FD = dyn_cast<FunctionDecl>(D))
1934 bool IsOutput =
false;
1935 if (HLSLParamModifierAttr *MA = D->
getAttr<HLSLParamModifierAttr>()) {
1942 if (SemanticName ==
"SV_DISPATCHTHREADID") {
1945 Diag(AL.
getLoc(), diag::err_hlsl_semantic_output_not_supported) << AL;
1946 if (Index.has_value())
1947 Diag(AL.
getLoc(), diag::err_hlsl_semantic_indexing_not_supported) << AL;
1952 if (SemanticName ==
"SV_GROUPINDEX") {
1954 Diag(AL.
getLoc(), diag::err_hlsl_semantic_output_not_supported) << AL;
1955 if (Index.has_value())
1956 Diag(AL.
getLoc(), diag::err_hlsl_semantic_indexing_not_supported) << AL;
1961 if (SemanticName ==
"SV_GROUPTHREADID") {
1964 Diag(AL.
getLoc(), diag::err_hlsl_semantic_output_not_supported) << AL;
1965 if (Index.has_value())
1966 Diag(AL.
getLoc(), diag::err_hlsl_semantic_indexing_not_supported) << AL;
1971 if (SemanticName ==
"SV_GROUPID") {
1974 Diag(AL.
getLoc(), diag::err_hlsl_semantic_output_not_supported) << AL;
1975 if (Index.has_value())
1976 Diag(AL.
getLoc(), diag::err_hlsl_semantic_indexing_not_supported) << AL;
1981 if (SemanticName ==
"SV_POSITION") {
1982 const auto *VT = ValueType->getAs<
VectorType>();
1983 if (!ValueType->hasFloatingRepresentation() ||
1984 (VT && VT->getNumElements() > 4))
1985 Diag(AL.
getLoc(), diag::err_hlsl_attr_invalid_type)
1986 << AL <<
"float/float1/float2/float3/float4";
1991 if (SemanticName ==
"SV_VERTEXID") {
1992 uint64_t SizeInBits =
SemaRef.Context.getTypeSize(ValueType);
1993 if (!ValueType->isUnsignedIntegerType() || SizeInBits != 32)
1994 Diag(AL.
getLoc(), diag::err_hlsl_attr_invalid_type) << AL <<
"uint";
1999 if (SemanticName ==
"SV_TARGET") {
2000 const auto *VT = ValueType->getAs<
VectorType>();
2001 if (!ValueType->hasFloatingRepresentation() ||
2002 (VT && VT->getNumElements() > 4))
2003 Diag(AL.
getLoc(), diag::err_hlsl_attr_invalid_type)
2004 << AL <<
"float/float1/float2/float3/float4";
2009 Diag(AL.
getLoc(), diag::err_hlsl_unknown_semantic) << AL;
2013 uint32_t IndexValue(0), ExplicitIndex(0);
2016 assert(0 &&
"HLSLUnparsedSemantic is expected to have 2 int arguments.");
2018 assert(IndexValue > 0 ? ExplicitIndex :
true);
2019 std::optional<unsigned> Index =
2020 ExplicitIndex ? std::optional<unsigned>(IndexValue) : std::nullopt;
2030 Diag(AL.
getLoc(), diag::err_hlsl_attr_invalid_ast_node)
2031 << AL <<
"shader constant in a constant buffer";
2035 uint32_t SubComponent;
2045 bool IsAggregateTy = (T->isArrayType() || T->isStructureType());
2050 if (IsAggregateTy) {
2051 Diag(AL.
getLoc(), diag::err_hlsl_invalid_register_or_packoffset);
2055 if ((Component * 32 + Size) > 128) {
2056 Diag(AL.
getLoc(), diag::err_hlsl_packoffset_cross_reg_boundary);
2061 EltTy = VT->getElementType();
2063 if (Align > 32 && Component == 1) {
2066 Diag(AL.
getLoc(), diag::err_hlsl_packoffset_alignment_mismatch)
2080 if (!
SemaRef.checkStringLiteralArgumentAttr(AL, 0, Str, &ArgLoc))
2083 llvm::Triple::EnvironmentType ShaderType;
2084 if (!HLSLShaderAttr::ConvertStrToEnvironmentType(Str, ShaderType)) {
2085 Diag(AL.
getLoc(), diag::warn_attribute_type_not_supported)
2086 << AL << Str << ArgLoc;
2100 assert(AttrList.size() &&
"expected list of resource attributes");
2107 HLSLAttributedResourceType::Attributes ResAttrs;
2109 bool HasResourceClass =
false;
2110 bool HasResourceDimension =
false;
2111 for (
const Attr *A : AttrList) {
2116 case attr::HLSLResourceClass: {
2118 if (HasResourceClass) {
2120 ? diag::warn_duplicate_attribute_exact
2121 : diag::warn_duplicate_attribute)
2125 ResAttrs.ResourceClass = RC;
2126 HasResourceClass =
true;
2129 case attr::HLSLResourceDimension: {
2130 llvm::dxil::ResourceDimension RD =
2132 if (HasResourceDimension) {
2134 ? diag::warn_duplicate_attribute_exact
2135 : diag::warn_duplicate_attribute)
2139 ResAttrs.ResourceDimension = RD;
2140 HasResourceDimension =
true;
2144 if (ResAttrs.IsROV) {
2148 ResAttrs.IsROV =
true;
2150 case attr::HLSLRawBuffer:
2151 if (ResAttrs.RawBuffer) {
2155 ResAttrs.RawBuffer =
true;
2157 case attr::HLSLIsArray:
2158 if (ResAttrs.IsArray) {
2162 ResAttrs.IsArray =
true;
2164 case attr::HLSLIsCounter:
2165 if (ResAttrs.IsCounter) {
2169 ResAttrs.IsCounter =
true;
2171 case attr::HLSLContainedType: {
2174 if (!ContainedTy.
isNull()) {
2176 ? diag::warn_duplicate_attribute_exact
2177 : diag::warn_duplicate_attribute)
2186 llvm_unreachable(
"unhandled resource attribute type");
2190 if (!HasResourceClass) {
2191 S.
Diag(AttrList.back()->getRange().getEnd(),
2192 diag::err_hlsl_missing_resource_class);
2197 Wrapped, ContainedTy, ResAttrs);
2199 if (LocInfo && ContainedTyInfo) {
2212 if (!T->isHLSLResourceType()) {
2213 Diag(AL.
getLoc(), diag::err_hlsl_attribute_needs_intangible_type)
2228 AttributeCommonInfo::AS_CXX11, 0, false ,
2233 case ParsedAttr::AT_HLSLResourceClass: {
2235 Diag(AL.
getLoc(), diag::err_attribute_argument_type)
2246 if (!HLSLResourceClassAttr::ConvertStrToResourceClass(Identifier, RC)) {
2247 Diag(ArgLoc, diag::warn_attribute_type_not_supported)
2248 <<
"ResourceClass" << Identifier;
2251 A = HLSLResourceClassAttr::Create(
getASTContext(), RC, ACI);
2255 case ParsedAttr::AT_HLSLResourceDimension: {
2256 StringRef Identifier;
2258 if (!
SemaRef.checkStringLiteralArgumentAttr(AL, 0, Identifier, &ArgLoc))
2262 llvm::dxil::ResourceDimension RD;
2263 if (!HLSLResourceDimensionAttr::ConvertStrToResourceDimension(Identifier,
2265 Diag(ArgLoc, diag::warn_attribute_type_not_supported)
2266 <<
"ResourceDimension" << Identifier;
2269 A = HLSLResourceDimensionAttr::Create(
getASTContext(), RD, ACI);
2273 case ParsedAttr::AT_HLSLROV:
2277 case ParsedAttr::AT_HLSLRawBuffer:
2281 case ParsedAttr::AT_HLSLIsCounter:
2285 case ParsedAttr::AT_HLSLIsArray:
2289 case ParsedAttr::AT_HLSLContainedType: {
2291 Diag(AL.
getLoc(), diag::err_attribute_wrong_number_arguments) << AL << 1;
2297 assert(TSI &&
"no type source info for attribute argument");
2299 diag::err_incomplete_type))
2301 A = HLSLContainedTypeAttr::Create(
getASTContext(), TSI, ACI);
2306 llvm_unreachable(
"unhandled HLSL attribute");
2309 HLSLResourcesTypeAttrs.emplace_back(A);
2315 if (!HLSLResourcesTypeAttrs.size())
2321 HLSLResourcesTypeAttrs, QT, &LocInfo)) {
2322 const HLSLAttributedResourceType *RT =
2329 LocsForHLSLAttributedResources.insert(std::pair(RT, LocInfo));
2331 HLSLResourcesTypeAttrs.clear();
2339 auto I = LocsForHLSLAttributedResources.find(RT);
2340 if (I != LocsForHLSLAttributedResources.end()) {
2341 LocInfo = I->second;
2342 LocsForHLSLAttributedResources.erase(I);
2351void SemaHLSL::collectResourceBindingsOnUserRecordDecl(
const VarDecl *VD,
2352 const RecordType *RT) {
2360 "incomplete arrays inside user defined types are not supported");
2369 if (
const HLSLAttributedResourceType *AttrResType =
2370 HLSLAttributedResourceType::findHandleTypeOnResource(Ty)) {
2375 Bindings.addDeclBindingInfo(VD, RC);
2376 }
else if (
const RecordType *RT = dyn_cast<RecordType>(Ty)) {
2382 collectResourceBindingsOnUserRecordDecl(VD, RT);
2394 bool SpecifiedSpace) {
2395 int RegTypeNum =
static_cast<int>(RegType);
2398 if (D->
hasAttr<HLSLGroupSharedAddressSpaceAttr>()) {
2399 S.
Diag(ArgLoc, diag::err_hlsl_binding_type_mismatch) << RegTypeNum;
2404 if (
HLSLBufferDecl *CBufferOrTBuffer = dyn_cast<HLSLBufferDecl>(D)) {
2405 ResourceClass RC = CBufferOrTBuffer->isCBuffer() ? ResourceClass::CBuffer
2406 : ResourceClass::SRV;
2416 assert(
isa<VarDecl>(D) &&
"D is expected to be VarDecl or HLSLBufferDecl");
2420 if (
const HLSLAttributedResourceType *AttrResType =
2421 HLSLAttributedResourceType::findHandleTypeOnResource(
2438 if (SpecifiedSpace && !DeclaredInCOrTBuffer)
2439 S.
Diag(ArgLoc, diag::err_hlsl_space_on_global_constant);
2444 if (RegType == RegisterType::CBuffer)
2445 S.
Diag(ArgLoc, diag::warn_hlsl_deprecated_register_type_b);
2446 else if (RegType != RegisterType::C)
2447 S.
Diag(ArgLoc, diag::err_hlsl_binding_type_mismatch) << RegTypeNum;
2451 if (RegType == RegisterType::C)
2452 S.
Diag(ArgLoc, diag::warn_hlsl_register_type_c_packoffset);
2454 S.
Diag(ArgLoc, diag::err_hlsl_binding_type_mismatch) << RegTypeNum;
2464 S.
Diag(ArgLoc, diag::err_hlsl_binding_type_mismatch) << RegTypeNum;
2472 bool RegisterTypesDetected[5] = {
false};
2473 RegisterTypesDetected[
static_cast<int>(regType)] =
true;
2476 if (HLSLResourceBindingAttr *
attr =
2477 dyn_cast<HLSLResourceBindingAttr>(*it)) {
2480 if (RegisterTypesDetected[
static_cast<int>(otherRegType)]) {
2481 int otherRegTypeNum =
static_cast<int>(otherRegType);
2483 diag::err_hlsl_duplicate_register_annotation)
2487 RegisterTypesDetected[
static_cast<int>(otherRegType)] =
true;
2495 bool SpecifiedSpace) {
2500 "expecting VarDecl or HLSLBufferDecl");
2512 const uint64_t &Limit,
2515 uint64_t ArrayCount = 1) {
2520 if (StartSlot > Limit)
2524 if (
const auto *AT = dyn_cast<ArrayType>(T)) {
2527 if (
const auto *CAT = dyn_cast<ConstantArrayType>(AT))
2528 Count = CAT->
getSize().getZExtValue();
2532 ArrayCount * Count);
2536 if (
auto ResTy = dyn_cast<HLSLAttributedResourceType>(T)) {
2539 if (ResTy->getAttrs().ResourceClass != ResClass)
2543 uint64_t EndSlot = StartSlot + ArrayCount - 1;
2544 if (EndSlot > Limit)
2548 StartSlot = EndSlot + 1;
2553 if (
const auto *RT = dyn_cast<RecordType>(T)) {
2556 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD)) {
2559 ResClass, Ctx, ArrayCount))
2566 ResClass, Ctx, ArrayCount))
2580 const uint64_t Limit = UINT32_MAX;
2581 if (SlotNum > Limit)
2586 if (RegTy == RegisterType::C || RegTy == RegisterType::I)
2589 if (
VarDecl *VD = dyn_cast<VarDecl>(TheDecl)) {
2590 uint64_t BaseSlot = SlotNum;
2598 return (BaseSlot > Limit);
2605 return (SlotNum > Limit);
2608 llvm_unreachable(
"unexpected decl type");
2612 if (
VarDecl *VD = dyn_cast<VarDecl>(TheDecl)) {
2614 if (
const auto *IAT = dyn_cast<IncompleteArrayType>(Ty))
2615 Ty = IAT->getElementType();
2617 diag::err_incomplete_type))
2621 StringRef Slot =
"";
2622 StringRef Space =
"";
2626 Diag(AL.
getLoc(), diag::err_attribute_argument_type)
2636 Diag(AL.
getLoc(), diag::err_attribute_argument_type)
2642 SpaceLoc = Loc->
getLoc();
2645 if (Str.starts_with(
"space")) {
2647 SpaceLoc = Loc->
getLoc();
2656 std::optional<unsigned> SlotNum;
2657 unsigned SpaceNum = 0;
2660 if (!Slot.empty()) {
2662 Diag(SlotLoc, diag::err_hlsl_binding_type_invalid) << Slot.substr(0, 1);
2665 if (RegType == RegisterType::I) {
2666 Diag(SlotLoc, diag::warn_hlsl_deprecated_register_type_i);
2669 const StringRef SlotNumStr = Slot.substr(1);
2674 if (SlotNumStr.getAsInteger(10, N)) {
2675 Diag(SlotLoc, diag::err_hlsl_unsupported_register_number);
2683 Diag(SlotLoc, diag::err_hlsl_register_number_too_large);
2692 if (!Space.starts_with(
"space")) {
2693 Diag(SpaceLoc, diag::err_hlsl_expected_space) << Space;
2696 StringRef SpaceNumStr = Space.substr(5);
2697 if (SpaceNumStr.getAsInteger(10, SpaceNum)) {
2698 Diag(SpaceLoc, diag::err_hlsl_expected_space) << Space;
2703 if (SlotNum.has_value())
2708 HLSLResourceBindingAttr *NewAttr =
2709 HLSLResourceBindingAttr::Create(
getASTContext(), Slot, Space, AL);
2711 NewAttr->setBinding(RegType, SlotNum, SpaceNum);
2737 while (
const auto *AT = Cur->
getAs<AttributedType>()) {
2739 if (K == attr::HLSLRowMajor || K == attr::HLSLColumnMajor) {
2743 Cur = AT->getModifiedType();
2754 ? attr::HLSLRowMajor
2755 : attr::HLSLColumnMajor;
2760 Diag(AL.
getLoc(), diag::err_hlsl_matrix_layout_non_matrix)
2769 if (ExistingKind == AttrK) {
2770 Diag(AL.
getLoc(), diag::warn_duplicate_attribute_exact)
2772 Diag(AL.
getLoc(), diag::note_previous_attribute);
2776 ExistingKind == attr::HLSLRowMajor ?
"row_major" :
"column_major");
2777 Diag(AL.
getLoc(), diag::err_hlsl_matrix_layout_conflict)
2779 Diag(AL.
getLoc(), diag::note_conflicting_attribute);
2784 if (AttrK == attr::HLSLRowMajor)
2785 return ::new (Ctx) HLSLRowMajorAttr(Ctx, AL);
2786 return ::new (Ctx) HLSLColumnMajorAttr(Ctx, AL);
2797 if (K != attr::HLSLRowMajor && K != attr::HLSLColumnMajor)
2799 if (T.isNull() || T->isDependentType())
2804 K == attr::HLSLRowMajor ?
"row_major" :
"column_major");
2805 Diag(Loc, diag::err_hlsl_matrix_layout_non_matrix) << II;
2812 switch (BuiltinID) {
2813 case Builtin::BI__builtin_hlsl_mul:
2814 case Builtin::BI__builtin_hlsl_transpose:
2822 if (!E || DestType.
isNull())
2834 if (!CallMat || CallMat->getNumRows() != DestMat->getNumRows() ||
2835 CallMat->getNumColumns() != DestMat->getNumColumns())
2884 llvm::DenseMap<const FunctionDecl *, unsigned> ScannedDecls;
2888 llvm::Triple::EnvironmentType CurrentShaderEnvironment;
2889 unsigned CurrentShaderStageBit;
2894 bool ReportOnlyShaderStageIssues;
2897 void SetShaderStageContext(llvm::Triple::EnvironmentType ShaderType) {
2898 static_assert(
sizeof(
unsigned) >= 4);
2899 assert(HLSLShaderAttr::isValidShaderType(ShaderType));
2900 assert((
unsigned)(ShaderType - llvm::Triple::Pixel) < 31 &&
2901 "ShaderType is too big for this bitmap");
2904 unsigned bitmapIndex = ShaderType - llvm::Triple::Pixel;
2905 CurrentShaderEnvironment = ShaderType;
2906 CurrentShaderStageBit = (1 << bitmapIndex);
2909 void SetUnknownShaderStageContext() {
2910 CurrentShaderEnvironment = llvm::Triple::UnknownEnvironment;
2911 CurrentShaderStageBit = (1 << 31);
2914 llvm::Triple::EnvironmentType GetCurrentShaderEnvironment()
const {
2915 return CurrentShaderEnvironment;
2918 bool InUnknownShaderStageContext()
const {
2919 return CurrentShaderEnvironment == llvm::Triple::UnknownEnvironment;
2923 void AddToScannedFunctions(
const FunctionDecl *FD) {
2924 unsigned &ScannedStages = ScannedDecls[FD];
2925 ScannedStages |= CurrentShaderStageBit;
2928 unsigned GetScannedStages(
const FunctionDecl *FD) {
return ScannedDecls[FD]; }
2930 bool WasAlreadyScannedInCurrentStage(
const FunctionDecl *FD) {
2931 return WasAlreadyScannedInCurrentStage(GetScannedStages(FD));
2934 bool WasAlreadyScannedInCurrentStage(
unsigned ScannerStages) {
2935 return ScannerStages & CurrentShaderStageBit;
2938 static bool NeverBeenScanned(
unsigned ScannedStages) {
2939 return ScannedStages == 0;
2943 void HandleFunctionOrMethodRef(FunctionDecl *FD, Expr *RefExpr);
2944 void CheckDeclAvailability(NamedDecl *D,
const AvailabilityAttr *AA,
2946 const AvailabilityAttr *FindAvailabilityAttr(
const Decl *D);
2947 bool HasMatchingEnvironmentOrNone(
const AvailabilityAttr *AA);
2950 DiagnoseHLSLAvailability(Sema &SemaRef)
2952 CurrentShaderEnvironment(llvm::Triple::UnknownEnvironment),
2953 CurrentShaderStageBit(0), ReportOnlyShaderStageIssues(
false) {}
2956 void RunOnTranslationUnit(
const TranslationUnitDecl *TU);
2957 void RunOnFunction(
const FunctionDecl *FD);
2959 bool VisitDeclRefExpr(DeclRefExpr *DRE)
override {
2960 FunctionDecl *FD = llvm::dyn_cast<FunctionDecl>(DRE->
getDecl());
2962 HandleFunctionOrMethodRef(FD, DRE);
2966 bool VisitMemberExpr(MemberExpr *ME)
override {
2967 FunctionDecl *FD = llvm::dyn_cast<FunctionDecl>(ME->
getMemberDecl());
2969 HandleFunctionOrMethodRef(FD, ME);
2974void DiagnoseHLSLAvailability::HandleFunctionOrMethodRef(
FunctionDecl *FD,
2977 "expected DeclRefExpr or MemberExpr");
2981 if (FD->
hasBody(FDWithBody)) {
2982 if (!WasAlreadyScannedInCurrentStage(FDWithBody))
2983 DeclsToScan.push_back(FDWithBody);
2988 const AvailabilityAttr *AA = FindAvailabilityAttr(FD);
2990 CheckDeclAvailability(
2994void DiagnoseHLSLAvailability::RunOnTranslationUnit(
3003 DeclContextsToScan.push_back(TU);
3005 while (!DeclContextsToScan.empty()) {
3006 const DeclContext *DC = DeclContextsToScan.pop_back_val();
3007 for (
auto &D : DC->
decls()) {
3014 if (llvm::dyn_cast<NamespaceDecl>(D) || llvm::dyn_cast<ExportDecl>(D)) {
3015 DeclContextsToScan.push_back(llvm::dyn_cast<DeclContext>(D));
3020 const FunctionDecl *FD = llvm::dyn_cast<FunctionDecl>(D);
3025 if (HLSLShaderAttr *ShaderAttr = FD->
getAttr<HLSLShaderAttr>()) {
3026 SetShaderStageContext(ShaderAttr->getType());
3035 for (
const auto *Redecl : FD->
redecls()) {
3036 if (Redecl->isInExportDeclContext()) {
3043 SetUnknownShaderStageContext();
3051void DiagnoseHLSLAvailability::RunOnFunction(
const FunctionDecl *FD) {
3052 assert(DeclsToScan.empty() &&
"DeclsToScan should be empty");
3053 DeclsToScan.push_back(FD);
3055 while (!DeclsToScan.empty()) {
3063 const unsigned ScannedStages = GetScannedStages(FD);
3064 if (WasAlreadyScannedInCurrentStage(ScannedStages))
3067 ReportOnlyShaderStageIssues = !NeverBeenScanned(ScannedStages);
3069 AddToScannedFunctions(FD);
3074bool DiagnoseHLSLAvailability::HasMatchingEnvironmentOrNone(
3075 const AvailabilityAttr *AA) {
3080 llvm::Triple::EnvironmentType CurrentEnv = GetCurrentShaderEnvironment();
3081 if (CurrentEnv == llvm::Triple::UnknownEnvironment)
3084 llvm::Triple::EnvironmentType AttrEnv =
3085 AvailabilityAttr::getEnvironmentType(IIEnvironment->
getName());
3087 return CurrentEnv == AttrEnv;
3090const AvailabilityAttr *
3091DiagnoseHLSLAvailability::FindAvailabilityAttr(
const Decl *D) {
3092 AvailabilityAttr
const *PartialMatch =
nullptr;
3096 for (
const auto *A : D->
attrs()) {
3097 if (
const auto *Avail = dyn_cast<AvailabilityAttr>(A)) {
3098 const AvailabilityAttr *EffectiveAvail = Avail->getEffectiveAttr();
3099 StringRef AttrPlatform = EffectiveAvail->getPlatform()->getName();
3100 StringRef TargetPlatform =
3104 if (AttrPlatform == TargetPlatform) {
3106 if (HasMatchingEnvironmentOrNone(EffectiveAvail))
3108 PartialMatch = Avail;
3112 return PartialMatch;
3117void DiagnoseHLSLAvailability::CheckDeclAvailability(
NamedDecl *D,
3118 const AvailabilityAttr *AA,
3137 if (ReportOnlyShaderStageIssues)
3143 if (InUnknownShaderStageContext())
3148 bool EnvironmentMatches = HasMatchingEnvironmentOrNone(AA);
3149 VersionTuple Introduced = AA->getIntroduced();
3158 llvm::StringRef PlatformName(
3161 llvm::StringRef CurrentEnvStr =
3162 llvm::Triple::getEnvironmentTypeName(GetCurrentShaderEnvironment());
3164 llvm::StringRef AttrEnvStr =
3165 AA->getEnvironment() ? AA->getEnvironment()->getName() :
"";
3166 bool UseEnvironment = !AttrEnvStr.empty();
3168 if (EnvironmentMatches) {
3169 SemaRef.
Diag(
Range.getBegin(), diag::warn_hlsl_availability)
3170 <<
Range << D << PlatformName << Introduced.getAsString()
3171 << UseEnvironment << CurrentEnvStr;
3173 SemaRef.
Diag(
Range.getBegin(), diag::warn_hlsl_availability_unavailable)
3177 SemaRef.
Diag(D->
getLocation(), diag::note_partial_availability_specified_here)
3178 << D << PlatformName << Introduced.getAsString()
3180 << UseEnvironment << AttrEnvStr << CurrentEnvStr;
3187 if (!DefaultCBufferDecls.empty()) {
3190 DefaultCBufferDecls);
3193 SemaRef.getCurLexicalContext()->addDecl(DefaultCBuffer);
3197 for (
const Decl *VD : DefaultCBufferDecls) {
3198 const HLSLResourceBindingAttr *RBA =
3199 VD->
getAttr<HLSLResourceBindingAttr>();
3200 if (RBA && RBA->hasRegisterSlot() &&
3201 RBA->getRegisterType() == HLSLResourceBindingAttr::RegisterType::C) {
3208 SemaRef.Consumer.HandleTopLevelDecl(DG);
3210 diagnoseAvailabilityViolations(TU);
3219 "expected member expr to have resource record type or array of them");
3225 const Expr *NonConstIndexExpr =
nullptr;
3228 if (
const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E)) {
3229 if (!NonConstIndexExpr)
3237 diag::err_hlsl_resource_member_array_access_not_constant);
3241 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
3242 const Expr *IdxExpr = ASE->getIdx();
3244 NonConstIndexExpr = IdxExpr;
3246 }
else if (
const auto *SubME = dyn_cast<MemberExpr>(E)) {
3247 E = SubME->getBase();
3248 }
else if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E)) {
3249 E = ICE->getSubExpr();
3251 llvm_unreachable(
"unexpected expr type in resource member access");
3260 SemaRef.Context.getCanonicalType(
SemaRef.Context.getAddrSpaceQualType(
3263 SemaRef.Context.getLValueReferenceType(AddrSpaceType));
3266 SemaRef.Context.DeclarationNames.getCXXConversionFunctionName(
3270 [[maybe_unused]]
bool LookupSucceeded =
3271 SemaRef.LookupQualifiedName(ConvR, RD);
3272 assert(LookupSucceeded);
3281std::optional<ExprResult>
3284 const HLSLAttributedResourceType *ResTy =
3285 HLSLAttributedResourceType::findHandleTypeOnResource(
3286 BaseType.getTypePtr());
3288 ResTy->getAttrs().ResourceClass != llvm::dxil::ResourceClass::CBuffer)
3289 return std::nullopt;
3291 QualType TemplateType = ResTy->getContainedType();
3295 assert(NamedConversionDecl &&
3296 "Could not find conversion function for ConstantBuffer.");
3297 auto *ConversionDecl =
3300 return SemaRef.BuildCXXMemberCallExpr(BaseExpr.
get(), NamedConversionDecl,
3312 TI.
getTriple().getEnvironment() != llvm::Triple::EnvironmentType::Library)
3315 DiagnoseHLSLAvailability(
SemaRef).RunOnTranslationUnit(TU);
3322 for (
unsigned I = 1, N = TheCall->
getNumArgs(); I < N; ++I) {
3325 S->
Diag(TheCall->
getBeginLoc(), diag::err_vec_builtin_incompatible_vector)
3350 for (
unsigned I = 0; I < TheCall->
getNumArgs(); ++I) {
3365 if (!BaseType->isFloat32Type())
3366 return S->
Diag(Loc, diag::err_builtin_invalid_arg_type)
3367 << ArgOrdinal << 5 << 0
3377 BaseType = VT->getElementType();
3379 BaseType = MT->getElementType();
3381 if (!BaseType->isHalfType() && !BaseType->isFloat32Type())
3382 return S->
Diag(Loc, diag::err_builtin_invalid_arg_type)
3383 << ArgOrdinal << 5 << 0
3397 if (!BaseType->isDoubleType()) {
3400 return S->
Diag(Loc, diag::err_builtin_requires_double_type)
3401 << ArgOrdinal << PassedType;
3408 unsigned ArgIndex) {
3409 auto *Arg = TheCall->
getArg(ArgIndex);
3411 if (Arg->IgnoreCasts()->isModifiableLvalue(S->
Context, &OrigLoc) ==
3414 S->
Diag(OrigLoc, diag::error_hlsl_inout_lvalue) << Arg << 0;
3428 << (ArgIndex + 1) << LValueTy;
3438 if (VecTy->getElementType()->isDoubleType())
3439 return S->
Diag(Loc, diag::err_builtin_invalid_arg_type)
3440 << ArgOrdinal << 1 << 0 << 1
3450 return S->
Diag(Loc, diag::err_builtin_invalid_arg_type)
3451 << ArgOrdinal << 5 << 1
3460 if (VecTy->getElementType()->isUnsignedIntegerType())
3463 return S->
Diag(Loc, diag::err_builtin_invalid_arg_type)
3464 << ArgOrdinal << 4 << 3 << 0
3473 return S->
Diag(Loc, diag::err_builtin_invalid_arg_type)
3474 << ArgOrdinal << 5 << 3
3480 unsigned ArgOrdinal,
unsigned Width) {
3483 ArgTy = VTy->getElementType();
3485 uint64_t ElementBitCount =
3487 if (ElementBitCount != Width) {
3489 diag::err_integer_incorrect_bit_count)
3490 << Width << ElementBitCount;
3507 unsigned ArgIndex) {
3516 diag::err_typecheck_expect_scalar_or_vector)
3517 << ArgType << Scalar;
3524 QualType Scalar,
unsigned ArgIndex) {
3535 if (
const auto *VTy = ArgType->getAs<
VectorType>()) {
3548 diag::err_typecheck_expect_scalar_or_vector_or_matrix)
3549 << ArgType << Scalar;
3554 unsigned ArgIndex) {
3559 if (!(ArgType->isScalarType() ||
3560 (VTy && VTy->getElementType()->isScalarType()))) {
3562 diag::err_typecheck_expect_any_scalar_or_vector)
3572 unsigned ArgIndex) {
3574 assert(ArgIndex < TheCall->getNumArgs());
3582 diag::err_typecheck_expect_any_scalar_or_vector)
3607 diag::err_typecheck_call_different_arg_types)
3626 Arg1ScalarTy = VTy->getElementType();
3630 Arg2ScalarTy = VTy->getElementType();
3633 S->
Diag(Arg1->
getBeginLoc(), diag::err_hlsl_builtin_scalar_vector_mismatch)
3634 << 1 << TheCall->
getCallee() << Arg1Ty << Arg2Ty;
3644 if (Arg1Length > 0 && Arg0Length != Arg1Length) {
3646 diag::err_typecheck_vector_lengths_not_equal)
3652 if (Arg2Length > 0 && Arg0Length != Arg2Length) {
3654 diag::err_typecheck_vector_lengths_not_equal)
3666 assert(TheCall->
getNumArgs() > IndexArgIndex &&
"Index argument missing");
3669 unsigned int ActualDim = 1;
3671 ActualDim = VTy->getNumElements();
3672 IndexTy = VTy->getElementType();
3676 diag::err_typecheck_expect_int)
3682 const HLSLAttributedResourceType *ResTy =
3684 assert(ResTy &&
"Resource argument must be a resource");
3685 HLSLAttributedResourceType::Attributes ResAttrs = ResTy->getAttrs();
3687 unsigned int ExpectedDim = 1;
3688 if (ResAttrs.ResourceDimension != llvm::dxil::ResourceDimension::Unknown)
3691 if (ActualDim != ExpectedDim) {
3693 diag::err_hlsl_builtin_resource_coordinate_dimension_mismatch)
3704 llvm::function_ref<
bool(
const HLSLAttributedResourceType *ResType)> Check =
3708 const HLSLAttributedResourceType *ResTy =
3712 diag::err_typecheck_expect_hlsl_resource)
3716 if (Check && Check(ResTy)) {
3718 diag::err_invalid_hlsl_resource_type)
3726 QualType BaseType,
unsigned ExpectedCount,
3728 unsigned PassedCount = 1;
3730 PassedCount = VecTy->getNumElements();
3732 if (PassedCount != ExpectedCount) {
3735 S->
Diag(Loc, diag::err_typecheck_convert_incompatible)
3747 [](
const HLSLAttributedResourceType *ResType) {
3748 return ResType->getAttrs().ResourceDimension ==
3749 llvm::dxil::ResourceDimension::Unknown;
3755 [](
const HLSLAttributedResourceType *ResType) {
3756 return ResType->getAttrs().ResourceClass !=
3757 llvm::hlsl::ResourceClass::Sampler;
3765 unsigned ExpectedDim =
3793 unsigned NextIdx = 3;
3799 diag::err_typecheck_convert_incompatible)
3807 Expr *ComponentArg = TheCall->
getArg(NextIdx);
3811 diag::err_typecheck_convert_incompatible)
3818 std::optional<llvm::APSInt> ComponentOpt =
3821 int64_t ComponentVal = ComponentOpt->getSExtValue();
3822 if (ComponentVal != 0) {
3825 assert(ComponentVal >= 0 && ComponentVal <= 3 &&
3826 "The component is not in the expected range.");
3828 diag::err_hlsl_gathercmp_invalid_component)
3838 const HLSLAttributedResourceType *ResourceTy =
3841 unsigned ExpectedDim =
3850 assert(ResourceTy->hasContainedType() &&
3851 "Expecting a contained type for resource with a dimension "
3853 QualType ReturnType = ResourceTy->getContainedType();
3857 S.
Diag(TheCall->
getBeginLoc(), diag::err_hlsl_samplecmp_requires_float);
3863 ReturnType = VecTy->getElementType();
3876 [](
const HLSLAttributedResourceType *ResType) {
3877 return ResType->getAttrs().ResourceDimension ==
3878 llvm::dxil::ResourceDimension::Unknown;
3886 unsigned ExpectedDim =
3895 EltTy = VTy->getElementType();
3910 TheCall->
setType(ResourceTy->getContainedType());
3915 unsigned MinArgs, MaxArgs;
3943 const HLSLAttributedResourceType *ResourceTy =
3945 unsigned ExpectedDim =
3948 unsigned NextIdx = 3;
3957 diag::err_typecheck_convert_incompatible)
3992 diag::err_typecheck_convert_incompatible)
3998 assert(ResourceTy->hasContainedType() &&
3999 "Expecting a contained type for resource with a dimension "
4001 QualType ReturnType = ResourceTy->getContainedType();
4004 S.
Diag(TheCall->
getBeginLoc(), diag::err_hlsl_samplecmp_requires_float);
4017 switch (BuiltinID) {
4018 case Builtin::BI__builtin_hlsl_adduint64: {
4019 if (
SemaRef.checkArgCount(TheCall, 2))
4033 if (NumElementsArg != 2 && NumElementsArg != 4) {
4035 << 1 << 64 << NumElementsArg * 32;
4049 case Builtin::BI__builtin_hlsl_resource_getpointer: {
4050 if (
SemaRef.checkArgCountRange(TheCall, 1, 2) ||
4057 QualType ContainedTy = ResourceTy->getContainedType();
4058 auto ReturnType =
SemaRef.Context.getAddrSpaceQualType(
4061 ReturnType =
SemaRef.Context.getPointerType(ReturnType);
4066 case Builtin::BI__builtin_hlsl_resource_getpointer_typed: {
4067 if (
SemaRef.checkArgCount(TheCall, 3) ||
4074 "expected pointer type for second argument");
4081 diag::err_invalid_use_of_array_type);
4085 auto ReturnType =
SemaRef.Context.getAddrSpaceQualType(
4088 ReturnType =
SemaRef.Context.getPointerType(ReturnType);
4093 case Builtin::BI__builtin_hlsl_resource_load_with_status: {
4094 if (
SemaRef.checkArgCount(TheCall, 3) ||
4097 SemaRef.getASTContext().UnsignedIntTy) ||
4099 SemaRef.getASTContext().UnsignedIntTy) ||
4105 QualType ReturnType = ResourceTy->getContainedType();
4110 case Builtin::BI__builtin_hlsl_resource_load_with_status_typed: {
4111 if (
SemaRef.checkArgCount(TheCall, 4) ||
4114 SemaRef.getASTContext().UnsignedIntTy) ||
4116 SemaRef.getASTContext().UnsignedIntTy) ||
4122 "expected pointer type for second argument");
4129 diag::err_invalid_use_of_array_type);
4135 case Builtin::BI__builtin_hlsl_resource_load_level:
4137 case Builtin::BI__builtin_hlsl_resource_sample:
4139 case Builtin::BI__builtin_hlsl_resource_sample_bias:
4141 case Builtin::BI__builtin_hlsl_resource_sample_grad:
4143 case Builtin::BI__builtin_hlsl_resource_sample_level:
4145 case Builtin::BI__builtin_hlsl_resource_sample_cmp:
4147 case Builtin::BI__builtin_hlsl_resource_sample_cmp_level_zero:
4149 case Builtin::BI__builtin_hlsl_resource_calculate_lod:
4150 case Builtin::BI__builtin_hlsl_resource_calculate_lod_unclamped:
4152 case Builtin::BI__builtin_hlsl_resource_gather:
4154 case Builtin::BI__builtin_hlsl_resource_gather_cmp:
4156 case Builtin::BI__builtin_hlsl_resource_uninitializedhandle: {
4157 assert(TheCall->
getNumArgs() == 1 &&
"expected 1 arg");
4163 case Builtin::BI__builtin_hlsl_resource_handlefrombinding: {
4164 assert(TheCall->
getNumArgs() == 6 &&
"expected 6 args");
4170 case Builtin::BI__builtin_hlsl_resource_handlefromimplicitbinding: {
4171 assert(TheCall->
getNumArgs() == 6 &&
"expected 6 args");
4177 case Builtin::BI__builtin_hlsl_resource_counterhandlefromimplicitbinding: {
4178 assert(TheCall->
getNumArgs() == 3 &&
"expected 3 args");
4181 auto *MainResType = MainHandleTy->
getAs<HLSLAttributedResourceType>();
4182 auto MainAttrs = MainResType->getAttrs();
4183 assert(!MainAttrs.IsCounter &&
"cannot create a counter from a counter");
4184 MainAttrs.IsCounter =
true;
4186 MainResType->getWrappedType(), MainResType->getContainedType(),
4190 TheCall->
setType(CounterHandleTy);
4193 case Builtin::BI__builtin_hlsl_and:
4194 case Builtin::BI__builtin_hlsl_or: {
4195 if (
SemaRef.checkArgCount(TheCall, 2))
4209 case Builtin::BI__builtin_hlsl_all:
4210 case Builtin::BI__builtin_hlsl_any: {
4211 if (
SemaRef.checkArgCount(TheCall, 1))
4217 case Builtin::BI__builtin_hlsl_asdouble: {
4218 if (
SemaRef.checkArgCount(TheCall, 2))
4222 SemaRef.Context.UnsignedIntTy,
4227 SemaRef.Context.UnsignedIntTy,
4236 case Builtin::BI__builtin_hlsl_elementwise_clamp: {
4237 if (
SemaRef.BuiltinElementwiseTernaryMath(
4243 case Builtin::BI__builtin_hlsl_dot: {
4245 if (
SemaRef.BuiltinVectorToScalarMath(TheCall))
4251 case Builtin::BI__builtin_hlsl_elementwise_firstbithigh:
4252 case Builtin::BI__builtin_hlsl_elementwise_firstbitlow: {
4253 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4263 EltTy = VecTy->getElementType();
4264 ResTy =
SemaRef.Context.getExtVectorType(ResTy, VecTy->getNumElements());
4277 case Builtin::BI__builtin_hlsl_select: {
4278 if (
SemaRef.checkArgCount(TheCall, 3))
4286 if (VTy && VTy->getElementType()->isBooleanType() &&
4291 case Builtin::BI__builtin_hlsl_elementwise_saturate:
4292 case Builtin::BI__builtin_hlsl_elementwise_rcp: {
4293 if (
SemaRef.checkArgCount(TheCall, 1))
4299 diag::err_builtin_invalid_arg_type)
4302 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4306 case Builtin::BI__builtin_hlsl_elementwise_degrees:
4307 case Builtin::BI__builtin_hlsl_elementwise_radians:
4308 case Builtin::BI__builtin_hlsl_elementwise_rsqrt:
4309 case Builtin::BI__builtin_hlsl_elementwise_frac:
4310 case Builtin::BI__builtin_hlsl_elementwise_ddx_coarse:
4311 case Builtin::BI__builtin_hlsl_elementwise_ddy_coarse:
4312 case Builtin::BI__builtin_hlsl_elementwise_ddx_fine:
4313 case Builtin::BI__builtin_hlsl_elementwise_ddy_fine: {
4314 if (
SemaRef.checkArgCount(TheCall, 1))
4319 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4323 case Builtin::BI__builtin_hlsl_elementwise_isinf:
4324 case Builtin::BI__builtin_hlsl_elementwise_isnan: {
4325 if (
SemaRef.checkArgCount(TheCall, 1))
4330 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4335 case Builtin::BI__builtin_hlsl_lerp: {
4336 if (
SemaRef.checkArgCount(TheCall, 3))
4343 if (
SemaRef.BuiltinElementwiseTernaryMath(TheCall))
4347 case Builtin::BI__builtin_hlsl_mad: {
4348 if (
SemaRef.BuiltinElementwiseTernaryMath(
4354 case Builtin::BI__builtin_hlsl_mul: {
4355 if (
SemaRef.checkArgCount(TheCall, 2))
4364 if (
const auto *VTy = T->getAs<
VectorType>())
4365 return VTy->getElementType();
4367 return MTy->getElementType();
4371 QualType EltTy0 = getElemType(Ty0);
4380 if (IsVec0 && IsMat1) {
4383 }
else if (IsMat0 && IsVec1) {
4387 assert(IsMat0 && IsMat1);
4397 case Builtin::BI__builtin_hlsl_normalize: {
4398 if (
SemaRef.checkArgCount(TheCall, 1))
4409 case Builtin::BI__builtin_elementwise_fma: {
4410 if (
SemaRef.checkArgCount(TheCall, 3) ||
4425 case Builtin::BI__builtin_hlsl_transpose: {
4426 if (
SemaRef.checkArgCount(TheCall, 1))
4435 << 1 << 3 << 0 << 0 << ArgTy;
4440 MatTy->getElementType(), MatTy->getNumColumns(), MatTy->getNumRows());
4444 case Builtin::BI__builtin_hlsl_elementwise_sign: {
4445 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4453 case Builtin::BI__builtin_hlsl_step: {
4454 if (
SemaRef.checkArgCount(TheCall, 2))
4466 case Builtin::BI__builtin_hlsl_wave_active_all_equal: {
4467 if (
SemaRef.checkArgCount(TheCall, 1))
4481 unsigned NumElts = VecTy->getNumElements();
4491 case Builtin::BI__builtin_hlsl_wave_active_max:
4492 case Builtin::BI__builtin_hlsl_wave_active_min:
4493 case Builtin::BI__builtin_hlsl_wave_active_sum:
4494 case Builtin::BI__builtin_hlsl_wave_active_product: {
4495 if (
SemaRef.checkArgCount(TheCall, 1))
4508 case Builtin::BI__builtin_hlsl_wave_active_bit_or:
4509 case Builtin::BI__builtin_hlsl_wave_active_bit_xor:
4510 case Builtin::BI__builtin_hlsl_wave_active_bit_and: {
4511 if (
SemaRef.checkArgCount(TheCall, 1))
4526 (VTy && VTy->getElementType()->isIntegerType()))) {
4528 diag::err_builtin_invalid_arg_type)
4529 << ArgTyExpr <<
SemaRef.Context.UnsignedIntTy << 1 << 0 << 0;
4537 case Builtin::BI__builtin_hlsl_interlocked_add: {
4547 diag::err_typecheck_call_too_few_args_at_least)
4552 if (
SemaRef.checkArgCountAtMost(TheCall, 3))
4558 diag::err_builtin_invalid_arg_type)
4586 case Builtin::BI__builtin_elementwise_bitreverse: {
4594 case Builtin::BI__builtin_hlsl_wave_prefix_count_bits: {
4595 if (
SemaRef.checkArgCount(TheCall, 1))
4600 if (!(
ArgType->isScalarType())) {
4602 diag::err_typecheck_expect_any_scalar_or_vector)
4607 if (!(
ArgType->isBooleanType())) {
4609 diag::err_typecheck_expect_any_scalar_or_vector)
4616 case Builtin::BI__builtin_hlsl_wave_read_lane_at: {
4617 if (
SemaRef.checkArgCount(TheCall, 2))
4625 diag::err_typecheck_convert_incompatible)
4626 << ArgTyIndex <<
SemaRef.Context.UnsignedIntTy << 1 << 0 << 0;
4639 case Builtin::BI__builtin_hlsl_wave_get_lane_index: {
4640 if (
SemaRef.checkArgCount(TheCall, 0))
4644 case Builtin::BI__builtin_hlsl_wave_prefix_sum:
4645 case Builtin::BI__builtin_hlsl_wave_prefix_product: {
4646 if (
SemaRef.checkArgCount(TheCall, 1))
4659 case Builtin::BI__builtin_hlsl_quad_read_across_x:
4660 case Builtin::BI__builtin_hlsl_quad_read_across_y:
4661 case Builtin::BI__builtin_hlsl_quad_read_across_diagonal: {
4662 if (
SemaRef.checkArgCount(TheCall, 1))
4674 case Builtin::BI__builtin_hlsl_elementwise_splitdouble: {
4675 if (
SemaRef.checkArgCount(TheCall, 3))
4681 SemaRef.Context.UnsignedIntTy, 1) ||
4683 SemaRef.Context.UnsignedIntTy, 2))
4691 case Builtin::BI__builtin_hlsl_elementwise_clip: {
4692 if (
SemaRef.checkArgCount(TheCall, 1))
4699 case Builtin::BI__builtin_elementwise_acos:
4700 case Builtin::BI__builtin_elementwise_asin:
4701 case Builtin::BI__builtin_elementwise_atan:
4702 case Builtin::BI__builtin_elementwise_atan2:
4703 case Builtin::BI__builtin_elementwise_ceil:
4704 case Builtin::BI__builtin_elementwise_cos:
4705 case Builtin::BI__builtin_elementwise_cosh:
4706 case Builtin::BI__builtin_elementwise_exp:
4707 case Builtin::BI__builtin_elementwise_exp2:
4708 case Builtin::BI__builtin_elementwise_exp10:
4709 case Builtin::BI__builtin_elementwise_floor:
4710 case Builtin::BI__builtin_elementwise_fmod:
4711 case Builtin::BI__builtin_elementwise_log:
4712 case Builtin::BI__builtin_elementwise_log2:
4713 case Builtin::BI__builtin_elementwise_log10:
4714 case Builtin::BI__builtin_elementwise_pow:
4715 case Builtin::BI__builtin_elementwise_roundeven:
4716 case Builtin::BI__builtin_elementwise_sin:
4717 case Builtin::BI__builtin_elementwise_sinh:
4718 case Builtin::BI__builtin_elementwise_sqrt:
4719 case Builtin::BI__builtin_elementwise_tan:
4720 case Builtin::BI__builtin_elementwise_tanh:
4721 case Builtin::BI__builtin_elementwise_trunc: {
4727 case Builtin::BI__builtin_hlsl_buffer_update_counter: {
4728 assert(TheCall->
getNumArgs() == 2 &&
"expected 2 args");
4729 auto checkResTy = [](
const HLSLAttributedResourceType *ResTy) ->
bool {
4730 return !(ResTy->getAttrs().ResourceClass == ResourceClass::UAV &&
4731 ResTy->getAttrs().RawBuffer && ResTy->hasContainedType());
4736 std::optional<llvm::APSInt> Offset =
4738 if (!Offset.has_value() ||
std::abs(Offset->getExtValue()) != 1) {
4740 diag::err_hlsl_expect_arg_const_int_one_or_neg_one)
4746 case Builtin::BI__builtin_hlsl_elementwise_f16tof32: {
4747 if (
SemaRef.checkArgCount(TheCall, 1))
4758 ArgTy = VTy->getElementType();
4761 diag::err_builtin_invalid_arg_type)
4770 case Builtin::BI__builtin_hlsl_elementwise_f32tof16: {
4771 if (
SemaRef.checkArgCount(TheCall, 1))
4786 WorkList.push_back(BaseTy);
4787 while (!WorkList.empty()) {
4788 QualType T = WorkList.pop_back_val();
4789 T = T.getCanonicalType().getUnqualifiedType();
4790 if (
const auto *AT = dyn_cast<ConstantArrayType>(T)) {
4798 for (uint64_t Ct = 0; Ct < AT->
getZExtSize(); ++Ct)
4799 llvm::append_range(List, ElementFields);
4804 if (
const auto *VT = dyn_cast<VectorType>(T)) {
4805 List.insert(List.end(), VT->getNumElements(), VT->getElementType());
4808 if (
const auto *MT = dyn_cast<ConstantMatrixType>(T)) {
4809 List.insert(List.end(), MT->getNumElementsFlattened(),
4810 MT->getElementType());
4813 if (
const auto *RD = T->getAsCXXRecordDecl()) {
4814 if (RD->isStandardLayout())
4815 RD = RD->getStandardLayoutBaseWithFields();
4819 if (RD->
isUnion() || !RD->isAggregate()) {
4825 for (
const auto *FD : RD->
fields())
4826 if (!FD->isUnnamedBitField())
4827 FieldTypes.push_back(FD->
getType());
4829 std::reverse(FieldTypes.begin(), FieldTypes.end());
4830 llvm::append_range(WorkList, FieldTypes);
4834 if (!RD->isStandardLayout()) {
4836 for (
const auto &
Base : RD->bases())
4837 FieldTypes.push_back(
Base.getType());
4838 std::reverse(FieldTypes.begin(), FieldTypes.end());
4839 llvm::append_range(WorkList, FieldTypes);
4874 if (
SemaRef.Context.getTypeSize(QT) / 8 > 16)
4880 int ArraySize = VT->getNumElements();
4885 QualType ElTy = VT->getElementType();
4889 if (
SemaRef.Context.getTypeSize(QT) / 8 > 16)
4905 if (
SemaRef.getASTContext().hasSameType(T1, T2))
4914 return llvm::equal(T1Types, T2Types,
4916 return SemaRef.IsLayoutCompatible(LHS, RHS);
4925 bool HadError =
false;
4927 for (
unsigned i = 0, e =
New->getNumParams(); i != e; ++i) {
4935 const auto *NDAttr = NewParam->
getAttr<HLSLParamModifierAttr>();
4936 unsigned NSpellingIdx = (NDAttr ? NDAttr->getSpellingListIndex() : 0);
4937 const auto *ODAttr = OldParam->
getAttr<HLSLParamModifierAttr>();
4938 unsigned OSpellingIdx = (ODAttr ? ODAttr->getSpellingListIndex() : 0);
4940 if (NSpellingIdx != OSpellingIdx) {
4942 diag::err_hlsl_param_qualifier_mismatch)
4943 << NDAttr << NewParam;
4959 if (
SemaRef.getASTContext().hasSameUnqualifiedType(SrcTy, DestTy))
4974 llvm_unreachable(
"HLSL doesn't support pointers.");
4977 llvm_unreachable(
"HLSL doesn't support complex types.");
4979 llvm_unreachable(
"HLSL doesn't support fixed point types.");
4981 llvm_unreachable(
"Should have returned before this");
4991 llvm_unreachable(
"HLSL doesn't support complex types.");
4993 llvm_unreachable(
"HLSL doesn't support fixed point types.");
4998 llvm_unreachable(
"HLSL doesn't support pointers.");
5000 llvm_unreachable(
"Should have returned before this");
5006 llvm_unreachable(
"HLSL doesn't support pointers.");
5009 llvm_unreachable(
"HLSL doesn't support fixed point types.");
5013 llvm_unreachable(
"HLSL doesn't support complex types.");
5016 llvm_unreachable(
"Unhandled scalar cast");
5037 !(SrcMatTy && SrcMatTy->getNumElementsFlattened() == 1))
5043 SrcTy = SrcMatTy->getElementType();
5048 for (
unsigned I = 0, Size = DestTypes.size(); I < Size; ++I) {
5049 if (DestTypes[I]->isUnionType())
5081 if (SrcTypes.size() < DestTypes.size())
5084 unsigned SrcSize = SrcTypes.size();
5085 unsigned DstSize = DestTypes.size();
5087 for (I = 0; I < DstSize && I < SrcSize; I++) {
5088 if (SrcTypes[I]->isUnionType() || DestTypes[I]->isUnionType())
5096 for (; I < SrcSize; I++) {
5097 if (SrcTypes[I]->isUnionType())
5104 assert(Param->hasAttr<HLSLParamModifierAttr>() &&
5105 "We should not get here without a parameter modifier expression");
5106 const auto *
Attr = Param->getAttr<HLSLParamModifierAttr>();
5113 << Arg << (IsInOut ? 1 : 0);
5119 QualType Ty = Param->getType().getNonLValueExprType(Ctx);
5126 << Arg << (IsInOut ? 1 : 0);
5138 SemaRef.PerformCopyInitialization(Entity, Param->getBeginLoc(), ArgOpV);
5144 auto *OpV =
new (Ctx)
5149 Res =
SemaRef.ActOnBinOp(
SemaRef.getCurScope(), Param->getBeginLoc(),
5150 tok::equal, ArgOpV, OpV);
5166 "Pointer and reference types cannot be inout or out parameters");
5167 Ty =
SemaRef.getASTContext().getLValueReferenceType(Ty);
5183 for (
const auto *FD : RD->
fields()) {
5187 assert(RD->getNumBases() <= 1 &&
5188 "HLSL doesn't support multiple inheritance");
5189 return RD->getNumBases()
5194 if (
const auto *AT = dyn_cast<ArrayType>(Ty)) {
5195 if (
const auto *CAT = dyn_cast<ConstantArrayType>(AT))
5207 bool IsVKPushConstant = IsVulkan && VD->
hasAttr<HLSLVkPushConstantAttr>();
5212 !VD->
hasAttr<HLSLVkConstantIdAttr>() && !IsVKPushConstant &&
5218 if (
Decl->getType().hasAddressSpace())
5221 if (
Decl->getType()->isDependentType())
5233 if (
Decl->
hasAttr<HLSLVkExtBuiltinOutputAttr>()) {
5247 llvm::Triple::Vulkan;
5248 if (IsVulkan &&
Decl->
hasAttr<HLSLVkPushConstantAttr>()) {
5249 if (HasDeclaredAPushConstant)
5255 HasDeclaredAPushConstant =
true;
5282class StructBindingContext {
5285 HLSLResourceBindingAttr *RegBindingsAttrs[4];
5286 unsigned RegBindingOffset[4];
5289 static_assert(
static_cast<unsigned>(RegisterType::SRV) == 0 &&
5290 static_cast<unsigned>(RegisterType::UAV) == 1 &&
5291 static_cast<unsigned>(RegisterType::CBuffer) == 2 &&
5292 static_cast<unsigned>(RegisterType::Sampler) == 3,
5293 "unexpected register type values");
5296 HLSLVkBindingAttr *VkBindingAttr;
5297 unsigned VkBindingOffset;
5302 StructBindingContext(
VarDecl *VD) {
5303 for (
unsigned i = 0; i < 4; ++i) {
5304 RegBindingsAttrs[i] =
nullptr;
5305 RegBindingOffset[i] = 0;
5307 VkBindingAttr =
nullptr;
5308 VkBindingOffset = 0;
5314 if (
auto *RBA = dyn_cast<HLSLResourceBindingAttr>(A)) {
5316 unsigned RegTypeIdx =
static_cast<unsigned>(RegType);
5319 RegBindingsAttrs[RegTypeIdx] = RBA;
5324 if (
auto *VBA = dyn_cast<HLSLVkBindingAttr>(A))
5325 VkBindingAttr = VBA;
5332 Attr *createBindingAttr(SemaHLSL &S, ASTContext &AST,
RegisterType RegType,
5333 unsigned Range,
bool HasCounter) {
5334 assert(
static_cast<unsigned>(RegType) < 4 &&
"unexpected register type");
5336 if (VkBindingAttr) {
5337 unsigned Offset = VkBindingOffset;
5338 VkBindingOffset +=
Range;
5339 return HLSLVkBindingAttr::CreateImplicit(
5340 AST, VkBindingAttr->getBinding() + Offset, VkBindingAttr->getSet(),
5341 VkBindingAttr->getRange());
5344 HLSLResourceBindingAttr *RBA =
5345 RegBindingsAttrs[
static_cast<unsigned>(RegType)];
5346 HLSLResourceBindingAttr *NewAttr =
nullptr;
5348 if (RBA && RBA->hasRegisterSlot()) {
5351 unsigned Offset = RegBindingOffset[
static_cast<unsigned>(RegType)];
5352 RegBindingOffset[
static_cast<unsigned>(RegType)] += Range;
5354 unsigned NewSlotNumber = RBA->getSlotNumber() + Offset;
5355 StringRef NewSlotNumberStr =
5357 NewAttr = HLSLResourceBindingAttr::CreateImplicit(
5358 AST, NewSlotNumberStr, RBA->getSpace(), RBA->getRange());
5359 NewAttr->setBinding(RegType, NewSlotNumber, RBA->getSpaceNumber());
5363 NewAttr = HLSLResourceBindingAttr::CreateImplicit(AST,
"",
"0", {});
5364 NewAttr->setBinding(RegType, std::nullopt,
5365 RBA ? RBA->getSpaceNumber() : 0);
5369 NewAttr->setImplicitCounterBindingOrderID(
5378static void createGlobalResourceDeclForStruct(
5380 QualType ResTy, StructBindingContext &BindingCtx) {
5382 "expected resource type or array of resources");
5393 while (
const auto *AT = dyn_cast<ArrayType>(SingleResTy)) {
5394 const auto *CAT = dyn_cast<ConstantArrayType>(AT);
5399 const HLSLAttributedResourceType *ResHandleTy =
5400 HLSLAttributedResourceType::findHandleTypeOnResource(SingleResTy);
5404 Attr *BindingAttr = BindingCtx.createBindingAttr(
5406 ResDecl->
addAttr(BindingAttr);
5407 ResDecl->
addAttr(InternalLinkageAttr::CreateImplicit(AST));
5416 HLSLAssociatedResourceDeclAttr::CreateImplicit(AST, ResDecl));
5423static void handleArrayOfStructWithResources(
5425 EmbeddedResourceNameBuilder &NameBuilder, StructBindingContext &BindingCtx);
5430static void handleStructWithResources(
Sema &S,
VarDecl *ParentVD,
5432 EmbeddedResourceNameBuilder &NameBuilder,
5433 StructBindingContext &BindingCtx) {
5436 assert(RD->
getNumBases() <= 1 &&
"HLSL doesn't support multiple inheritance");
5443 handleStructWithResources(S, ParentVD, BaseRD, NameBuilder, BindingCtx);
5457 createGlobalResourceDeclForStruct(S, ParentVD, FD->
getLocation(), II,
5460 handleStructWithResources(S, ParentVD, RD, NameBuilder, BindingCtx);
5462 }
else if (
const auto *ArrayTy = dyn_cast<ConstantArrayType>(FDTy)) {
5464 "resource arrays should have been already handled");
5465 handleArrayOfStructWithResources(S, ParentVD, ArrayTy, NameBuilder,
5474handleArrayOfStructWithResources(
Sema &S,
VarDecl *ParentVD,
5476 EmbeddedResourceNameBuilder &NameBuilder,
5477 StructBindingContext &BindingCtx) {
5485 if (!SubCAT && !ElementRD)
5488 for (
unsigned I = 0, E = CAT->
getSize().getZExtValue(); I < E; ++I) {
5491 handleStructWithResources(S, ParentVD, ElementRD, NameBuilder,
5494 handleArrayOfStructWithResources(S, ParentVD, SubCAT, NameBuilder,
5507void SemaHLSL::handleGlobalStructOrArrayOfWithResources(
VarDecl *VD) {
5508 EmbeddedResourceNameBuilder NameBuilder(VD->
getName());
5509 StructBindingContext BindingCtx(VD);
5513 "Expected non-resource struct or array type");
5516 handleStructWithResources(
SemaRef, VD, RD, NameBuilder, BindingCtx);
5520 if (
const auto *CAT = dyn_cast<ConstantArrayType>(VDTy)) {
5521 handleArrayOfStructWithResources(
SemaRef, VD, CAT, NameBuilder, BindingCtx);
5529 if (
SemaRef.RequireCompleteType(
5532 diag::err_typecheck_decl_incomplete_type)) {
5546 DefaultCBufferDecls.push_back(VD);
5551 collectResourceBindingsOnVarDecl(VD);
5553 if (VD->
hasAttr<HLSLVkConstantIdAttr>())
5565 processExplicitBindingsOnDecl(VD);
5603 handleGlobalStructOrArrayOfWithResources(VD);
5607 if (VD->
hasAttr<HLSLGroupSharedAddressSpaceAttr>())
5616 "expected resource record type");
5632 const char *CreateMethodName;
5634 CreateMethodName = HasCounter ?
"__createFromBindingWithImplicitCounter"
5635 :
"__createFromBinding";
5637 CreateMethodName = HasCounter
5638 ?
"__createFromImplicitBindingWithImplicitCounter"
5639 :
"__createFromImplicitBinding";
5644 if (!CreateMethod) {
5649 "create method lookup should always succeed for built-in resource "
5658 Args.push_back(RegSlot);
5666 Args.push_back(OrderId);
5672 Args.push_back(Space);
5676 Args.push_back(RangeSize);
5680 Args.push_back(Index);
5682 StringRef VarName = VD->
getName();
5690 Args.push_back(NameCast);
5698 Args.push_back(CounterId);
5721 SemaRef.CheckCompleteVariableDeclaration(VD);
5727 "expected array of resource records");
5748 lookupMethod(
SemaRef, ResourceDecl,
5749 HasCounter ?
"__createFromBindingWithImplicitCounter"
5750 :
"__createFromBinding",
5754 CreateMethod = lookupMethod(
5756 HasCounter ?
"__createFromImplicitBindingWithImplicitCounter"
5757 :
"__createFromImplicitBinding",
5800std::optional<const DeclBindingInfo *> SemaHLSL::inferGlobalBinding(
Expr *E) {
5801 if (
auto *Ternary = dyn_cast<ConditionalOperator>(E)) {
5802 auto TrueInfo = inferGlobalBinding(Ternary->getTrueExpr());
5803 auto FalseInfo = inferGlobalBinding(Ternary->getFalseExpr());
5804 if (!TrueInfo || !FalseInfo)
5805 return std::nullopt;
5806 if (*TrueInfo != *FalseInfo)
5807 return std::nullopt;
5811 if (
auto *ASE = dyn_cast<ArraySubscriptExpr>(E))
5820 if (
const auto *AttrResType =
5821 HLSLAttributedResourceType::findHandleTypeOnResource(Ty)) {
5823 return Bindings.getDeclBindingInfo(VD, RC);
5830void SemaHLSL::trackLocalResource(
VarDecl *VD,
Expr *E) {
5831 std::optional<const DeclBindingInfo *> ExprBinding = inferGlobalBinding(E);
5834 diag::warn_hlsl_assigning_local_resource_is_not_unique)
5839 if (*ExprBinding ==
nullptr)
5842 auto PrevBinding = Assigns.find(VD);
5843 if (PrevBinding == Assigns.end()) {
5845 Assigns.insert({VD, *ExprBinding});
5850 if (*ExprBinding != PrevBinding->second) {
5852 diag::warn_hlsl_assigning_local_resource_is_not_unique)
5854 SemaRef.Diag(VD->getLocation(), diag::note_var_declared_here) << VD;
5865 "expected LHS to be a resource record or array of resource records");
5866 if (Opc != BO_Assign)
5871 while (
auto *ASE = dyn_cast<ArraySubscriptExpr>(E))
5879 SemaRef.Diag(Loc, diag::err_hlsl_assign_to_global_resource) << VD;
5884 trackLocalResource(VD, RHSExpr);
5901void SemaHLSL::collectResourceBindingsOnVarDecl(
VarDecl *VD) {
5903 "expected global variable that contains HLSL resource");
5906 if (
const HLSLBufferDecl *CBufferOrTBuffer = dyn_cast<HLSLBufferDecl>(VD)) {
5907 Bindings.addDeclBindingInfo(VD, CBufferOrTBuffer->isCBuffer()
5908 ? ResourceClass::CBuffer
5909 : ResourceClass::SRV);
5922 if (
const HLSLAttributedResourceType *AttrResType =
5923 HLSLAttributedResourceType::findHandleTypeOnResource(Ty)) {
5924 Bindings.addDeclBindingInfo(VD, AttrResType->getAttrs().ResourceClass);
5929 if (
const RecordType *RT = dyn_cast<RecordType>(Ty))
5930 collectResourceBindingsOnUserRecordDecl(VD, RT);
5936void SemaHLSL::processExplicitBindingsOnDecl(
VarDecl *VD) {
5939 bool HasBinding =
false;
5940 for (Attr *A : VD->
attrs()) {
5943 if (
auto PA = VD->
getAttr<HLSLVkPushConstantAttr>())
5944 Diag(PA->getLoc(), diag::err_hlsl_attr_incompatible) << A << PA;
5947 HLSLResourceBindingAttr *RBA = dyn_cast<HLSLResourceBindingAttr>(A);
5948 if (!RBA || !RBA->hasRegisterSlot())
5953 assert(RT != RegisterType::I &&
"invalid or obsolete register type should "
5954 "never have an attribute created");
5956 if (RT == RegisterType::C) {
5957 if (Bindings.hasBindingInfoForDecl(VD))
5959 diag::warn_hlsl_user_defined_type_missing_member)
5960 <<
static_cast<int>(RT);
5968 if (DeclBindingInfo *BI = Bindings.getDeclBindingInfo(VD, RC)) {
5973 diag::warn_hlsl_user_defined_type_missing_member)
5974 <<
static_cast<int>(RT);
5982class InitListTransformer {
5986 QualType *DstIt =
nullptr;
5987 Expr **ArgIt =
nullptr;
5993 bool castInitializer(Expr *E) {
5994 assert(DstIt &&
"This should always be something!");
5995 if (DstIt == DestTypes.end()) {
5997 ArgExprs.push_back(E);
6002 DstIt = DestTypes.begin();
6005 Ctx, *DstIt,
false);
6010 ArgExprs.push_back(
Init);
6015 bool buildInitializerListImpl(Expr *E) {
6017 if (
auto *
Init = dyn_cast<InitListExpr>(E)) {
6018 for (
auto *SubInit :
Init->inits())
6019 if (!buildInitializerListImpl(SubInit))
6029 return castInitializer(E);
6043 if (
auto *VecTy = Ty->
getAs<VectorType>()) {
6048 for (uint64_t I = 0; I <
Size; ++I) {
6050 SizeTy, SourceLocation());
6056 if (!castInitializer(ElExpr.
get()))
6061 if (
auto *MTy = Ty->
getAs<ConstantMatrixType>()) {
6062 unsigned Rows = MTy->getNumRows();
6063 unsigned Cols = MTy->getNumColumns();
6064 QualType ElemTy = MTy->getElementType();
6066 for (
unsigned R = 0;
R < Rows; ++
R) {
6067 for (
unsigned C = 0;
C < Cols; ++
C) {
6080 if (!castInitializer(ElExpr.
get()))
6088 if (
auto *ArrTy = dyn_cast<ConstantArrayType>(Ty.
getTypePtr())) {
6092 for (uint64_t I = 0; I <
Size; ++I) {
6094 SizeTy, SourceLocation());
6099 if (!buildInitializerListImpl(ElExpr.
get()))
6106 llvm::SmallVector<CXXRecordDecl *> RecordDecls;
6107 RecordDecls.push_back(RD);
6108 while (RecordDecls.back()->getNumBases()) {
6109 CXXRecordDecl *D = RecordDecls.back();
6111 "HLSL doesn't support multiple inheritance");
6112 RecordDecls.push_back(
6115 while (!RecordDecls.empty()) {
6116 CXXRecordDecl *RD = RecordDecls.pop_back_val();
6117 for (
auto *FD : RD->
fields()) {
6118 if (FD->isUnnamedBitField())
6126 if (!buildInitializerListImpl(Res.
get()))
6134 Expr *generateInitListsImpl(QualType Ty) {
6136 assert(ArgIt != ArgExprs.end() &&
"Something is off in iteration!");
6141 llvm::SmallVector<Expr *>
Inits;
6146 if (
auto *ATy = Ty->
getAs<VectorType>()) {
6147 ElTy = ATy->getElementType();
6148 Size = ATy->getNumElements();
6149 }
else if (
auto *CMTy = Ty->
getAs<ConstantMatrixType>()) {
6150 ElTy = CMTy->getElementType();
6151 Size = CMTy->getNumElementsFlattened();
6154 ElTy = VTy->getElementType();
6155 Size = VTy->getZExtSize();
6157 for (uint64_t I = 0; I <
Size; ++I)
6158 Inits.push_back(generateInitListsImpl(ElTy));
6161 llvm::SmallVector<CXXRecordDecl *> RecordDecls;
6162 RecordDecls.push_back(RD);
6163 while (RecordDecls.back()->getNumBases()) {
6164 CXXRecordDecl *D = RecordDecls.back();
6166 "HLSL doesn't support multiple inheritance");
6167 RecordDecls.push_back(
6170 while (!RecordDecls.empty()) {
6171 CXXRecordDecl *RD = RecordDecls.pop_back_val();
6172 for (
auto *FD : RD->
fields())
6173 if (!FD->isUnnamedBitField())
6178 new (Ctx) InitListExpr(Ctx,
Inits.front()->getBeginLoc(),
Inits,
6179 Inits.back()->getEndLoc(),
false);
6180 NewInit->setType(Ty);
6185 llvm::SmallVector<QualType, 16> DestTypes;
6186 llvm::SmallVector<Expr *, 16> ArgExprs;
6187 InitListTransformer(Sema &SemaRef,
const InitializedEntity &Entity)
6188 : S(SemaRef), Ctx(SemaRef.getASTContext()),
6189 Wrap(Entity.
getType()->isIncompleteArrayType()) {
6190 InitTy = Entity.
getType().getNonReferenceType();
6200 DstIt = DestTypes.begin();
6203 bool buildInitializerList(Expr *E) {
return buildInitializerListImpl(E); }
6205 Expr *generateInitLists() {
6206 assert(!ArgExprs.empty() &&
6207 "Call buildInitializerList to generate argument expressions.");
6208 ArgIt = ArgExprs.begin();
6210 return generateInitListsImpl(InitTy);
6211 llvm::SmallVector<Expr *>
Inits;
6212 while (ArgIt != ArgExprs.end())
6213 Inits.push_back(generateInitListsImpl(InitTy));
6216 new (Ctx) InitListExpr(Ctx,
Inits.front()->getBeginLoc(),
Inits,
6217 Inits.back()->getEndLoc(),
false);
6218 llvm::APInt ArySize(64,
Inits.size());
6220 ArraySizeModifier::Normal, 0));
6232 if (
const ArrayType *AT = dyn_cast<ArrayType>(Ty)) {
6239 if (
const auto *RT = Ty->
getAs<RecordType>()) {
6243 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD)) {
6263 if (
Init->getType()->isScalarType())
6266 InitListTransformer ILT(
SemaRef, Entity);
6268 for (
unsigned I = 0; I <
Init->getNumInits(); ++I) {
6276 Init->setInit(I, E);
6278 if (!ILT.buildInitializerList(E))
6281 size_t ExpectedSize = ILT.DestTypes.size();
6282 size_t ActualSize = ILT.ArgExprs.size();
6283 if (ExpectedSize == 0 && ActualSize == 0)
6290 InitTy =
SemaRef.getASTContext().removeAddrSpaceQualType(InitTy);
6292 SemaRef.Diag(
Init->getBeginLoc(), diag::err_hlsl_incorrect_num_initializers)
6293 << (int)(ExpectedSize < ActualSize) << InitTy
6294 << ExpectedSize << ActualSize;
6304 assert(ExpectedSize > 0 &&
6305 "The expected size of an incomplete array type must be at least 1.");
6307 ((ActualSize + ExpectedSize - 1) / ExpectedSize) * ExpectedSize;
6315 InitTy =
SemaRef.getASTContext().removeAddrSpaceQualType(InitTy);
6316 if (ExpectedSize != ActualSize) {
6317 int TooManyOrFew = ActualSize > ExpectedSize ? 1 : 0;
6318 SemaRef.Diag(
Init->getBeginLoc(), diag::err_hlsl_incorrect_num_initializers)
6319 << TooManyOrFew << InitTy << ExpectedSize << ActualSize;
6326 Init->resizeInits(Ctx, NewInit->getNumInits());
6327 for (
unsigned I = 0; I < NewInit->getNumInits(); ++I)
6328 Init->updateInit(Ctx, I, NewInit->getInit(I));
6336 S.
Diag(OpLoc, diag::err_builtin_matrix_invalid_member)
6346 StringRef AccessorName = CompName->
getName();
6347 assert(!AccessorName.empty() &&
"Matrix Accessor must have a name");
6349 unsigned Rows = MT->getNumRows();
6350 unsigned Cols = MT->getNumColumns();
6351 bool IsZeroBasedAccessor =
false;
6352 unsigned ChunkLen = 0;
6353 if (AccessorName.size() < 2)
6355 "length 4 for zero based: \'_mRC\' or "
6356 "length 3 for one-based: \'_RC\' accessor",
6359 if (AccessorName[0] ==
'_') {
6360 if (AccessorName[1] ==
'm') {
6361 IsZeroBasedAccessor =
true;
6368 S, AccessorName,
"zero based: \'_mRC\' or one-based: \'_RC\' accessor",
6371 if (AccessorName.size() % ChunkLen != 0) {
6372 const llvm::StringRef
Expected = IsZeroBasedAccessor
6373 ?
"zero based: '_mRC' accessor"
6374 :
"one-based: '_RC' accessor";
6379 auto isDigit = [](
char c) {
return c >=
'0' && c <=
'9'; };
6380 auto isZeroBasedIndex = [](
unsigned i) {
return i <= 3; };
6381 auto isOneBasedIndex = [](
unsigned i) {
return i >= 1 && i <= 4; };
6383 bool HasRepeated =
false;
6385 unsigned NumComponents = 0;
6386 const char *Begin = AccessorName.data();
6388 for (
unsigned I = 0, E = AccessorName.size(); I < E; I += ChunkLen) {
6389 const char *Chunk = Begin + I;
6390 char RowChar = 0, ColChar = 0;
6391 if (IsZeroBasedAccessor) {
6393 if (Chunk[0] !=
'_' || Chunk[1] !=
'm') {
6394 char Bad = (Chunk[0] !=
'_') ? Chunk[0] : Chunk[1];
6396 S, StringRef(&Bad, 1),
"\'_m\' prefix",
6403 if (Chunk[0] !=
'_')
6405 S, StringRef(&Chunk[0], 1),
"\'_\' prefix",
6412 bool IsDigitsError =
false;
6414 unsigned BadPos = IsZeroBasedAccessor ? 2 : 1;
6418 IsDigitsError =
true;
6422 unsigned BadPos = IsZeroBasedAccessor ? 3 : 2;
6426 IsDigitsError =
true;
6431 unsigned Row = RowChar -
'0';
6432 unsigned Col = ColChar -
'0';
6434 bool HasIndexingError =
false;
6435 if (IsZeroBasedAccessor) {
6437 if (!isZeroBasedIndex(Row)) {
6438 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6440 HasIndexingError =
true;
6442 if (!isZeroBasedIndex(Col)) {
6443 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6445 HasIndexingError =
true;
6449 if (!isOneBasedIndex(Row)) {
6450 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6452 HasIndexingError =
true;
6454 if (!isOneBasedIndex(Col)) {
6455 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6457 HasIndexingError =
true;
6464 if (HasIndexingError)
6470 bool HasBoundsError =
false;
6472 Diag(OpLoc, diag::err_hlsl_matrix_index_out_of_bounds)
6474 HasBoundsError =
true;
6477 Diag(OpLoc, diag::err_hlsl_matrix_index_out_of_bounds)
6479 HasBoundsError =
true;
6484 unsigned FlatIndex = Row * Cols + Col;
6485 if (Seen[FlatIndex])
6487 Seen[FlatIndex] =
true;
6490 if (NumComponents == 0 || NumComponents > 4) {
6491 S.
Diag(OpLoc, diag::err_hlsl_matrix_swizzle_invalid_length)
6496 QualType ElemTy = MT->getElementType();
6497 if (NumComponents == 1)
6503 for (Sema::ExtVectorDeclsType::iterator
6507 if ((*I)->getUnderlyingType() == VT)
6518 trackLocalResource(VDecl,
Init);
6520 const HLSLVkConstantIdAttr *ConstIdAttr =
6521 VDecl->
getAttr<HLSLVkConstantIdAttr>();
6528 if (!
Init->isCXX11ConstantExpr(Context, &InitValue)) {
6538 int ConstantID = ConstIdAttr->getId();
6539 llvm::APInt IDVal(Context.getIntWidth(Context.IntTy), ConstantID);
6541 ConstIdAttr->getLocation());
6545 if (
C->getType()->getCanonicalTypeUnqualified() !=
6549 Context.getTrivialTypeSourceInfo(
6550 Init->getType(),
Init->getExprLoc()),
6569 if (!Params || Params->
size() != 1)
6582 if (
auto *TTP = dyn_cast<TemplateTypeParmDecl>(P)) {
6583 if (TTP->hasDefaultArgument()) {
6584 TemplateArgs.
addArgument(TTP->getDefaultArgument());
6587 }
else if (
auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(P)) {
6588 if (NTTP->hasDefaultArgument()) {
6589 TemplateArgs.
addArgument(NTTP->getDefaultArgument());
6592 }
else if (
auto *TTPD = dyn_cast<TemplateTemplateParmDecl>(P)) {
6593 if (TTPD->hasDefaultArgument()) {
6594 TemplateArgs.
addArgument(TTPD->getDefaultArgument());
6601 return SemaRef.CheckTemplateIdType(
6603 TemplateArgs,
nullptr,
false);
Defines the clang::ASTContext interface.
Defines enum values for all the target-independent builtin functions.
llvm::dxil::ResourceClass ResourceClass
Defines the C++ Decl subclasses, other than those for templates (found in DeclTemplate....
Defines the clang::IdentifierInfo, clang::IdentifierTable, and clang::Selector interfaces.
Forward-declares and imports various common LLVM datatypes that clang wants to use unqualified.
llvm::SmallVector< std::pair< const MemRegion *, SVal >, 4 > Bindings
static bool CheckArgTypeMatches(Sema *S, Expr *Arg, QualType ExpectedType)
static void BuildFlattenedTypeList(QualType BaseTy, llvm::SmallVectorImpl< QualType > &List)
static bool CheckUnsignedIntRepresentation(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
static bool containsIncompleteArrayType(QualType Ty)
static QualType handleIntegerVectorBinOpConversion(Sema &SemaRef, ExprResult &LHS, ExprResult &RHS, QualType LHSType, QualType RHSType, QualType LElTy, QualType RElTy, bool IsCompAssign)
static bool convertToRegisterType(StringRef Slot, RegisterType *RT)
static StringRef createRegisterString(ASTContext &AST, RegisterType RegType, unsigned N)
static bool CheckWaveActive(Sema *S, CallExpr *TheCall)
static void createHostLayoutStructForBuffer(Sema &S, HLSLBufferDecl *BufDecl)
static void castVector(Sema &S, ExprResult &E, QualType &Ty, unsigned Sz)
static QualType ReportMatrixInvalidMember(Sema &S, StringRef Name, StringRef Expected, SourceLocation OpLoc, SourceLocation CompLoc)
static bool CheckBoolSelect(Sema *S, CallExpr *TheCall)
static unsigned calculateLegacyCbufferFieldAlign(const ASTContext &Context, QualType T)
static bool isZeroSizedArray(const ConstantArrayType *CAT)
static bool DiagnoseHLSLRegisterAttribute(Sema &S, SourceLocation &ArgLoc, Decl *D, RegisterType RegType, bool SpecifiedSpace)
static bool hasConstantBufferLayout(QualType QT)
static FieldDecl * createFieldForHostLayoutStruct(Sema &S, const Type *Ty, IdentifierInfo *II, CXXRecordDecl *LayoutStruct)
static bool CheckUnsignedIntVecRepresentation(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
static bool isInvalidConstantBufferLeafElementType(const Type *Ty)
static bool CheckCalculateLodBuiltin(Sema &S, CallExpr *TheCall)
static Builtin::ID getSpecConstBuiltinId(const Type *Type)
static bool CheckFloatingOrIntRepresentation(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
static const Type * createHostLayoutType(Sema &S, const Type *Ty)
static bool CheckAnyScalarOrVector(Sema *S, CallExpr *TheCall, unsigned ArgIndex)
static const HLSLAttributedResourceType * getResourceArrayHandleType(QualType QT)
static IdentifierInfo * getHostLayoutStructName(Sema &S, NamedDecl *BaseDecl, bool MustBeUnique)
static bool CheckArgAddrSpaceOneOf(Sema *S, CallExpr *TheCall, unsigned ArgIndex, ArrayRef< LangAS > AllowedSpaces)
static void addImplicitBindingAttrToDecl(Sema &S, Decl *D, RegisterType RT, uint32_t ImplicitBindingOrderID)
static void SetElementTypeAsReturnType(Sema *S, CallExpr *TheCall, QualType ReturnType)
static unsigned calculateLegacyCbufferSize(const ASTContext &Context, QualType T)
static bool CheckLoadLevelBuiltin(Sema &S, CallExpr *TheCall)
static RegisterType getRegisterType(ResourceClass RC)
static bool ValidateRegisterNumber(uint64_t SlotNum, Decl *TheDecl, ASTContext &Ctx, RegisterType RegTy)
static bool isVkPipelineBuiltin(const ASTContext &AstContext, FunctionDecl *FD, HLSLAppliedSemanticAttr *Semantic, bool IsInput)
static bool CheckVectorElementCount(Sema *S, QualType PassedType, QualType BaseType, unsigned ExpectedCount, SourceLocation Loc)
static bool CheckModifiableLValue(Sema *S, CallExpr *TheCall, unsigned ArgIndex)
static QualType castElement(Sema &S, ExprResult &E, QualType Ty)
static char getRegisterTypeChar(RegisterType RT)
static bool CheckNotBoolScalarOrVector(Sema *S, CallExpr *TheCall, unsigned ArgIndex)
static bool isMatrixOrArrayOfMatrix(const ASTContext &Ctx, QualType QT)
static bool findExistingMatrixLayoutMarker(QualType T, attr::Kind &ExistingKind)
Walks the existing AttributedType sugar of T looking for a previously applied HLSLRowMajor/HLSLColumn...
static CXXRecordDecl * findRecordDeclInContext(IdentifierInfo *II, DeclContext *DC)
static bool CheckWavePrefix(Sema *S, CallExpr *TheCall)
static bool CheckExpectedBitWidth(Sema *S, CallExpr *TheCall, unsigned ArgOrdinal, unsigned Width)
static LangAS getLangASFromResourceClass(ResourceClass RC)
static bool CheckVectorSelect(Sema *S, CallExpr *TheCall)
static QualType handleFloatVectorBinOpConversion(Sema &SemaRef, ExprResult &LHS, ExprResult &RHS, QualType LHSType, QualType RHSType, QualType LElTy, QualType RElTy, bool IsCompAssign)
static const Type * getHostLayoutFieldType(QualType QT)
static ResourceClass getResourceClass(RegisterType RT)
static CXXRecordDecl * createHostLayoutStruct(Sema &S, CXXRecordDecl *StructDecl)
static bool CheckScalarOrVector(Sema *S, CallExpr *TheCall, QualType Scalar, unsigned ArgIndex)
static bool isLayoutAdaptingMatrixBuiltin(unsigned BuiltinID)
static bool CheckSamplingBuiltin(Sema &S, CallExpr *TheCall, SampleKind Kind)
static bool CheckScalarOrVectorOrMatrix(Sema *S, CallExpr *TheCall, QualType Scalar, unsigned ArgIndex)
static bool CheckFloatRepresentation(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
static bool CheckAnyDoubleRepresentation(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
static bool requiresImplicitBufferLayoutStructure(const CXXRecordDecl *RD)
static bool CheckResourceHandle(Sema *S, CallExpr *TheCall, unsigned ArgIndex, llvm::function_ref< bool(const HLSLAttributedResourceType *ResType)> Check=nullptr)
static void validatePackoffset(Sema &S, HLSLBufferDecl *BufDecl)
static bool IsDefaultBufferConstantDecl(const ASTContext &Ctx, VarDecl *VD)
HLSLResourceBindingAttr::RegisterType RegisterType
static CastKind getScalarCastKind(ASTContext &Ctx, QualType DestTy, QualType SrcTy)
static bool CheckGatherBuiltin(Sema &S, CallExpr *TheCall, bool IsCmp)
static bool isValidWaveSizeValue(unsigned Value)
static bool isResourceRecordTypeOrArrayOf(QualType Ty)
static bool AccumulateHLSLResourceSlots(QualType Ty, uint64_t &StartSlot, const uint64_t &Limit, const ResourceClass ResClass, ASTContext &Ctx, uint64_t ArrayCount=1)
static bool CheckNoDoubleVectors(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
static bool ValidateMultipleRegisterAnnotations(Sema &S, Decl *TheDecl, RegisterType regType)
static bool CheckTextureSamplerAndLocation(Sema &S, CallExpr *TheCall)
static bool DiagnoseLocalRegisterBinding(Sema &S, SourceLocation &ArgLoc, Decl *D, RegisterType RegType, bool SpecifiedSpace)
static bool CheckIndexType(Sema *S, CallExpr *TheCall, unsigned IndexArgIndex)
This file declares semantic analysis for HLSL constructs.
Defines the clang::SourceLocation class and associated facilities.
Defines various enumerations that describe declaration and type specifiers.
C Language Family Type Representation.
Defines the clang::TypeLoc interface and its subclasses.
C Language Family Type Representation.
static const TypeInfo & getInfo(unsigned id)
return(__x > > __y)|(__x<<(32 - __y))
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
virtual bool HandleTopLevelDecl(DeclGroupRef D)
HandleTopLevelDecl - Handle the specified top-level declaration.
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
unsigned getIntWidth(QualType T) const
int getIntegerTypeOrder(QualType LHS, QualType RHS) const
Return the highest ranked integer type, see C99 6.3.1.8p1.
QualType getPointerType(QualType T) const
Return the uniqued reference to the type for a pointer to the specified type.
const IncompleteArrayType * getAsIncompleteArrayType(QualType T) const
QualType getConstantArrayType(QualType EltTy, const llvm::APInt &ArySize, const Expr *SizeExpr, ArraySizeModifier ASM, unsigned IndexTypeQuals) const
Return the unique reference to the type for a constant array of the specified element type.
QualType getBaseElementType(const ArrayType *VAT) const
Return the innermost element type of an array type.
int getFloatingTypeOrder(QualType LHS, QualType RHS) const
Compare the rank of the two specified floating point types, ignoring the domain of the type (i....
TypeSourceInfo * getTrivialTypeSourceInfo(QualType T, SourceLocation Loc=SourceLocation()) const
Allocate a TypeSourceInfo where all locations have been initialized to a given location,...
QualType getStringLiteralArrayType(QualType EltTy, unsigned Length) const
Return a type for a constant array for a string literal of the specified element type and length.
uint64_t getTypeSize(QualType T) const
Return the size of the specified (complete) type T, in bits.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
CanQualType UnsignedIntTy
QualType getTypedefType(ElaboratedTypeKeyword Keyword, NestedNameSpecifier Qualifier, const TypedefNameDecl *Decl, QualType UnderlyingType=QualType(), std::optional< bool > TypeMatchesDeclOrNone=std::nullopt) const
Return the unique reference to the type for the specified typedef-name decl.
llvm::StringRef backupStr(llvm::StringRef S) const
QualType getSizeType() const
Return the unique type for "size_t" (C99 7.17), defined in <stddef.h>.
QualType getExtVectorType(QualType VectorType, unsigned NumElts) const
Return the unique reference to an extended vector type of the specified element type and size.
const TargetInfo & getTargetInfo() const
QualType getHLSLAttributedResourceType(QualType Wrapped, QualType Contained, const HLSLAttributedResourceType::Attributes &Attrs)
QualType getAddrSpaceQualType(QualType T, LangAS AddressSpace) const
Return the uniqued reference to the type for an address space qualified type with the specified type ...
CanQualType getCanonicalTagType(const TagDecl *TD) const
static bool hasSameUnqualifiedType(QualType T1, QualType T2)
Determine whether the given types are equivalent after cvr-qualifiers have been removed.
QualType getConstantMatrixType(QualType ElementType, unsigned NumRows, unsigned NumColumns) const
Return the unique reference to the matrix type of the specified element type and size.
unsigned getTypeAlign(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in bits.
Represents an array type, per C99 6.7.5.2 - Array Declarators.
QualType getElementType() const
Attr - This represents one attribute.
attr::Kind getKind() const
SourceLocation getLocation() const
SourceLocation getScopeLoc() const
SourceRange getRange() const
const IdentifierInfo * getScopeName() const
SourceLocation getLoc() const
const IdentifierInfo * getAttrName() const
Represents a base class of a C++ class.
QualType getType() const
Retrieves the type of the base class.
Represents a static or instance method of a struct/union/class.
Represents a C++ struct/union/class.
bool isHLSLIntangible() const
Returns true if the class contains HLSL intangible type, either as a field or in base class.
static CXXRecordDecl * Create(const ASTContext &C, TagKind TK, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, IdentifierInfo *Id, CXXRecordDecl *PrevDecl=nullptr)
void setBases(CXXBaseSpecifier const *const *Bases, unsigned NumBases)
Sets the base classes of this struct or class.
base_class_iterator bases_end()
void completeDefinition() override
Indicates that the definition of this class is now complete.
unsigned getNumBases() const
Retrieves the number of base classes of this class.
bool isHLSLBuiltinRecord() const
Returns true if the class is a built-in HLSL record.
base_class_iterator bases_begin()
bool isEmpty() const
Determine whether this is an empty class in the sense of (C++11 [meta.unary.prop]).
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Expr * getArg(unsigned Arg)
getArg - Return the specified argument.
SourceLocation getBeginLoc() const
static CallExpr * Create(const ASTContext &Ctx, Expr *Fn, ArrayRef< Expr * > Args, QualType Ty, ExprValueKind VK, SourceLocation RParenLoc, FPOptionsOverride FPFeatures, unsigned MinNumArgs=0, ADLCallKind UsesADL=NotADL)
Create a call expression.
FunctionDecl * getDirectCallee()
If the callee is a FunctionDecl, return it. Otherwise return null.
unsigned getNumArgs() const
getNumArgs - Return the number of actual arguments to this call.
SourceLocation getEndLoc() const
static CanQual< Type > CreateUnsafe(QualType Other)
QualType withConst() const
Retrieves a version of this type with const applied.
const T * getTypePtr() const
Retrieve the underlying type pointer, which refers to a canonical type.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
Represents the canonical version of C arrays with a specified constant size.
bool isZeroSize() const
Return true if the size is zero.
llvm::APInt getSize() const
Return the constant array size as an APInt.
uint64_t getZExtSize() const
Return the size zero-extended as a uint64_t.
Represents a concrete matrix type with constant number of rows and columns.
unsigned getNumColumns() const
Returns the number of columns in the matrix.
static DeclAccessPair make(NamedDecl *D, AccessSpecifier AS)
DeclContext - This is used only as base class of specific decl types that can act as declaration cont...
lookup_result lookup(DeclarationName Name) const
lookup - Find the declarations (if any) with the given Name in this context.
bool isTranslationUnit() const
void addDecl(Decl *D)
Add the declaration D into this context.
decl_range decls() const
decls_begin/decls_end - Iterate over the declarations stored in this context.
DeclContext * getNonTransparentContext()
A reference to a declared variable, function, enum, etc.
static DeclRefExpr * Create(const ASTContext &Context, NestedNameSpecifierLoc QualifierLoc, SourceLocation TemplateKWLoc, ValueDecl *D, bool RefersToEnclosingVariableOrCapture, SourceLocation NameLoc, QualType T, ExprValueKind VK, NamedDecl *FoundD=nullptr, const TemplateArgumentListInfo *TemplateArgs=nullptr, NonOdrUseReason NOUR=NOUR_None)
Decl - This represents one declaration (or definition), e.g.
ASTContext & getASTContext() const LLVM_READONLY
attr_iterator attr_end() const
bool isImplicit() const
isImplicit - Indicates whether the declaration was implicitly generated by the implementation.
void setInvalidDecl(bool Invalid=true)
setInvalidDecl - Indicates the Decl had a semantic error.
bool isInExportDeclContext() const
Whether this declaration was exported in a lexical context.
attr_iterator attr_begin() const
DeclContext * getNonTransparentDeclContext()
Return the non transparent context.
SourceLocation getLocation() const
void setImplicit(bool I=true)
DeclContext * getDeclContext()
AccessSpecifier getAccess() const
SourceLocation getBeginLoc() const LLVM_READONLY
The name of a declaration.
Represents a ValueDecl that came out of a declarator.
SourceLocation getBeginLoc() const LLVM_READONLY
This represents one expression.
bool isIntegerConstantExpr(const ASTContext &Ctx) const
ExprValueKind getValueKind() const
getValueKind - The value kind that this expression produces.
Expr * IgnoreParenImpCasts() LLVM_READONLY
Skip past any parentheses and implicit casts which might surround this expression until reaching a fi...
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
std::optional< llvm::APSInt > getIntegerConstantExpr(const ASTContext &Ctx) const
isIntegerConstantExpr - Return the value if this expression is a valid integer constant expression.
bool isLValue() const
isLValue - True if this expression is an "l-value" according to the rules of the current language.
ExprObjectKind getObjectKind() const
getObjectKind - The object kind that this expression produces.
Expr * IgnoreCasts() LLVM_READONLY
Skip past any casts which might surround this expression until reaching a fixed point.
bool HasSideEffects(const ASTContext &Ctx, bool IncludePossibleEffects=true) const
HasSideEffects - This routine returns true for all those expressions which have any effect other than...
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
ExtVectorType - Extended vector type.
Represents difference between two FPOptions values.
Represents a member of a struct/union/class.
static FieldDecl * Create(const ASTContext &C, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, const IdentifierInfo *Id, QualType T, TypeSourceInfo *TInfo, Expr *BW, bool Mutable, InClassInitStyle InitStyle)
static FixItHint CreateReplacement(CharSourceRange RemoveRange, StringRef Code)
Create a code modification hint that replaces the given source range with the given code string.
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
bool isThisDeclarationADefinition() const
Returns whether this specific declaration of the function is also a definition that does not contain ...
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
bool isTemplateInstantiation() const
Determines if the given function was instantiated from a function template.
redecl_range redecls() const
Returns an iterator range for all the redeclarations of the same decl.
unsigned getNumParams() const
Return the number of parameters this function must have based on its FunctionType.
DeclarationNameInfo getNameInfo() const
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
bool isDefined(const FunctionDecl *&Definition, bool CheckForPendingFriendDefinition=false) const
Returns true if the function has a definition that does not need to be instantiated.
HLSLBufferDecl - Represent a cbuffer or tbuffer declaration.
static HLSLBufferDecl * Create(ASTContext &C, DeclContext *LexicalParent, bool CBuffer, SourceLocation KwLoc, IdentifierInfo *ID, SourceLocation IDLoc, SourceLocation LBrace)
void addLayoutStruct(CXXRecordDecl *LS)
void setHasValidPackoffset(bool PO)
static HLSLBufferDecl * CreateDefaultCBuffer(ASTContext &C, DeclContext *LexicalParent, ArrayRef< Decl * > DefaultCBufferDecls)
buffer_decl_range buffer_decls() const
static HLSLOutArgExpr * Create(const ASTContext &C, QualType Ty, OpaqueValueExpr *Base, OpaqueValueExpr *OpV, Expr *WB, bool IsInOut)
static HLSLRootSignatureDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation Loc, IdentifierInfo *ID, llvm::dxbc::RootSignatureVersion Version, ArrayRef< llvm::hlsl::rootsig::RootElement > RootElements)
One of these records is kept for each identifier that is lexed.
StringRef getName() const
Return the actual identifier string.
A simple pair of identifier info and location.
SourceLocation getLoc() const
IdentifierInfo * getIdentifierInfo() const
IdentifierInfo & get(StringRef Name)
Return the identifier token info for the specified named identifier.
ImplicitCastExpr - Allows us to explicitly represent implicit type conversions, which have no direct ...
static ImplicitCastExpr * Create(const ASTContext &Context, QualType T, CastKind Kind, Expr *Operand, const CXXCastPath *BasePath, ExprValueKind Cat, FPOptionsOverride FPO)
Describes an C or C++ initializer list.
Describes an entity that is being initialized.
QualType getType() const
Retrieve type being initialized.
static InitializedEntity InitializeParameter(ASTContext &Context, ParmVarDecl *Parm)
Create the initialization entity for a parameter.
static IntegerLiteral * Create(const ASTContext &C, const llvm::APInt &V, QualType type, SourceLocation l)
Returns a new integer literal with value 'V' and type 'type'.
iterator begin(Source *source, bool LocalOnly=false)
Represents the results of name lookup.
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
Represents a matrix type, as defined in the Matrix Types clang extensions.
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
ValueDecl * getMemberDecl() const
Retrieve the member declaration to which this expression refers.
This represents a decl that may have a name.
NamedDecl * getUnderlyingDecl()
Looks through UsingDecls and ObjCCompatibleAliasDecls for the underlying named decl.
IdentifierInfo * getIdentifier() const
Get the identifier that names this declaration, if there is one.
StringRef getName() const
Get the name of identifier for this declaration as a StringRef.
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
A C++ nested-name-specifier augmented with source location information.
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
Represents a parameter to a function.
ParsedAttr - Represents a syntactic attribute.
unsigned getSemanticSpelling() const
If the parsed attribute has a semantic equivalent, and it would have a semantic Spelling enumeration ...
unsigned getMinArgs() const
bool checkExactlyNumArgs(class Sema &S, unsigned Num) const
Check if the attribute has exactly as many args as Num.
IdentifierLoc * getArgAsIdent(unsigned Arg) const
bool hasParsedType() const
void setInvalid(bool b=true) const
const ParsedType & getTypeArg() const
unsigned getNumArgs() const
getNumArgs - Return the number of actual arguments to this attribute.
bool isArgIdent(unsigned Arg) const
Expr * getArgAsExpr(unsigned Arg) const
AttributeCommonInfo::Kind getKind() const
A (possibly-)qualified type.
void addRestrict()
Add the restrict qualifier to this QualType.
QualType getNonLValueExprType(const ASTContext &Context) const
Determine the type of a (typically non-lvalue) expression with the specified result type.
QualType getDesugaredType(const ASTContext &Context) const
Return the specified type with any "sugar" removed from the type.
bool isNull() const
Return true if this QualType doesn't point to a type yet.
const Type * getTypePtr() const
Retrieves a pointer to the underlying (unqualified) type.
LangAS getAddressSpace() const
Return the address space of this type.
QualType getNonReferenceType() const
If Type is a reference type (e.g., const int&), returns the type that the reference refers to ("const...
QualType getCanonicalType() const
QualType getUnqualifiedType() const
Retrieve the unqualified variant of the given type, removing as little sugar as possible.
bool hasAddressSpace() const
Check if this type has any address space qualifier.
Represents a struct/union/class.
field_range fields() const
RecordDecl * getDefinitionOrSelf() const
bool hasBindingInfoForDecl(const VarDecl *VD) const
DeclBindingInfo * getDeclBindingInfo(const VarDecl *VD, ResourceClass ResClass)
DeclBindingInfo * addDeclBindingInfo(const VarDecl *VD, ResourceClass ResClass)
Scope - A scope is a transient data structure that is used while parsing the program.
ASTContext & getASTContext() const
SemaDiagnosticBuilder Diag(SourceLocation Loc, unsigned DiagID)
Emit a diagnostic.
ExprResult ActOnOutParamExpr(ParmVarDecl *Param, Expr *Arg)
HLSLRootSignatureDecl * lookupRootSignatureOverrideDecl(DeclContext *DC) const
bool CanPerformElementwiseCast(Expr *Src, QualType DestType)
void handleWaveSizeAttr(Decl *D, const ParsedAttr &AL)
void handleVkLocationAttr(Decl *D, const ParsedAttr &AL)
HLSLAttributedResourceLocInfo TakeLocForHLSLAttribute(const HLSLAttributedResourceType *RT)
void handleSemanticAttr(Decl *D, const ParsedAttr &AL)
bool CanPerformScalarCast(QualType SrcTy, QualType DestTy)
QualType ProcessResourceTypeAttributes(QualType Wrapped)
void handleShaderAttr(Decl *D, const ParsedAttr &AL)
uint32_t getNextImplicitBindingOrderID()
void CheckEntryPoint(FunctionDecl *FD)
void handleVkExtBuiltinOutputAttr(Decl *D, const ParsedAttr &AL)
void emitLogicalOperatorFixIt(Expr *LHS, Expr *RHS, BinaryOperatorKind Opc)
void propagateContextualMatrixLayout(Expr *E, QualType DestType)
T * createSemanticAttr(const AttributeCommonInfo &ACI, std::optional< unsigned > Location)
bool initGlobalResourceDecl(VarDecl *VD)
void ActOnEndOfTranslationUnit(TranslationUnitDecl *TU)
bool initGlobalResourceArrayDecl(VarDecl *VD)
HLSLVkConstantIdAttr * mergeVkConstantIdAttr(Decl *D, const AttributeCommonInfo &AL, int Id)
HLSLNumThreadsAttr * mergeNumThreadsAttr(Decl *D, const AttributeCommonInfo &AL, int X, int Y, int Z)
void deduceAddressSpace(VarDecl *Decl)
std::pair< IdentifierInfo *, bool > ActOnStartRootSignatureDecl(StringRef Signature)
Computes the unique Root Signature identifier from the given signature, then lookup if there is a pre...
void handlePackOffsetAttr(Decl *D, const ParsedAttr &AL)
Attr * buildMatrixLayoutTypeAttr(QualType T, const ParsedAttr &AL)
std::optional< ExprResult > tryPerformConstantBufferConversion(ExprResult &BaseExpr)
bool diagnosePositionType(QualType T, const ParsedAttr &AL)
bool handleInitialization(VarDecl *VDecl, Expr *&Init)
bool diagnoseInputIDType(QualType T, const ParsedAttr &AL)
void handleParamModifierAttr(Decl *D, const ParsedAttr &AL)
bool CheckResourceBinOp(BinaryOperatorKind Opc, Expr *LHSExpr, Expr *RHSExpr, SourceLocation Loc)
bool CanPerformAggregateSplatCast(Expr *Src, QualType DestType)
bool ActOnResourceMemberAccessExpr(MemberExpr *ME)
bool IsScalarizedLayoutCompatible(QualType T1, QualType T2) const
QualType ActOnTemplateShorthand(TemplateDecl *Template, SourceLocation NameLoc)
void diagnoseSystemSemanticAttr(Decl *D, const ParsedAttr &AL, std::optional< unsigned > Index)
void handleRootSignatureAttr(Decl *D, const ParsedAttr &AL)
bool CheckCompatibleParameterABI(FunctionDecl *New, FunctionDecl *Old)
QualType handleVectorBinOpConversion(ExprResult &LHS, ExprResult &RHS, QualType LHSType, QualType RHSType, bool IsCompAssign)
QualType checkMatrixComponent(Sema &S, QualType baseType, ExprValueKind &VK, SourceLocation OpLoc, const IdentifierInfo *CompName, SourceLocation CompLoc)
bool IsConstantBufferElementCompatible(QualType T1)
void handleResourceBindingAttr(Decl *D, const ParsedAttr &AL)
bool IsTypedResourceElementCompatible(QualType T1)
bool transformInitList(const InitializedEntity &Entity, InitListExpr *Init)
void handleNumThreadsAttr(Decl *D, const ParsedAttr &AL)
bool ActOnUninitializedVarDecl(VarDecl *D)
void handleVkExtBuiltinInputAttr(Decl *D, const ParsedAttr &AL)
bool canHaveOverloadedBinOp(QualType Ty, BinaryOperatorKind Opc)
void ActOnTopLevelFunction(FunctionDecl *FD)
bool handleResourceTypeAttr(QualType T, const ParsedAttr &AL)
void handleVkPushConstantAttr(Decl *D, const ParsedAttr &AL)
HLSLShaderAttr * mergeShaderAttr(Decl *D, const AttributeCommonInfo &AL, llvm::Triple::EnvironmentType ShaderType)
NamedDecl * getConstantBufferConversionFunction(QualType Type, CXXRecordDecl *RD)
void ActOnFinishBuffer(Decl *Dcl, SourceLocation RBrace)
void handleVkBindingAttr(Decl *D, const ParsedAttr &AL)
HLSLParamModifierAttr * mergeParamModifierAttr(Decl *D, const AttributeCommonInfo &AL, HLSLParamModifierAttr::Spelling Spelling)
QualType getInoutParameterType(QualType Ty)
void handleVkConstantIdAttr(Decl *D, const ParsedAttr &AL)
Decl * ActOnStartBuffer(Scope *BufferScope, bool CBuffer, SourceLocation KwLoc, IdentifierInfo *Ident, SourceLocation IdentLoc, SourceLocation LBrace)
bool diagnoseMatrixLayoutInstantiation(attr::Kind K, QualType T, SourceLocation Loc)
HLSLWaveSizeAttr * mergeWaveSizeAttr(Decl *D, const AttributeCommonInfo &AL, int Min, int Max, int Preferred, int SpelledArgsCount)
bool handleRootSignatureElements(ArrayRef< hlsl::RootSignatureElement > Elements)
void ActOnFinishRootSignatureDecl(SourceLocation Loc, IdentifierInfo *DeclIdent, ArrayRef< hlsl::RootSignatureElement > Elements)
Creates the Root Signature decl of the parsed Root Signature elements onto the AST and push it onto c...
void ActOnVariableDeclarator(VarDecl *VD)
bool CheckBuiltinFunctionCall(unsigned BuiltinID, CallExpr *TheCall)
Sema - This implements semantic analysis and AST building for C.
@ LookupOrdinaryName
Ordinary name lookup, which finds ordinary names (functions, variables, typedefs, etc....
@ LookupMemberName
Member name lookup, which finds the names of class/struct/union members.
ExtVectorDeclsType ExtVectorDecls
ExtVectorDecls - This is a list all the extended vector types.
ASTContext & getASTContext() const
ExprResult ImpCastExprToType(Expr *E, QualType Type, CastKind CK, ExprValueKind VK=VK_PRValue, const CXXCastPath *BasePath=nullptr, CheckedConversionKind CCK=CheckedConversionKind::Implicit)
ImpCastExprToType - If Expr is not of type 'Type', insert an implicit cast.
const LangOptions & getLangOpts() const
ExprResult TemporaryMaterializationConversion(Expr *E)
If E is a prvalue denoting an unmaterialized temporary, materialize it as an xvalue.
ExprResult BuildFieldReferenceExpr(Expr *BaseExpr, bool IsArrow, SourceLocation OpLoc, const CXXScopeSpec &SS, FieldDecl *Field, DeclAccessPair FoundDecl, const DeclarationNameInfo &MemberNameInfo)
bool checkArgCountRange(CallExpr *Call, unsigned MinArgCount, unsigned MaxArgCount)
Checks that a call expression's argument count is in the desired range.
ExternalSemaSource * getExternalSource() const
bool checkArgCount(CallExpr *Call, unsigned DesiredArgCount)
Checks that a call expression's argument count is the desired number.
ExprResult CreateBuiltinArraySubscriptExpr(Expr *Base, SourceLocation LLoc, Expr *Idx, SourceLocation RLoc)
bool LookupQualifiedName(LookupResult &R, DeclContext *LookupCtx, bool InUnqualifiedLookup=false)
Perform qualified name lookup into a given context.
ExprResult PerformCopyInitialization(const InitializedEntity &Entity, SourceLocation EqualLoc, ExprResult Init, bool TopLevelOfInitList=false, bool AllowExplicit=false)
ExprResult CreateBuiltinMatrixSubscriptExpr(Expr *Base, Expr *RowIdx, Expr *ColumnIdx, SourceLocation RBLoc)
Encodes a location in the source.
SourceLocation getLocWithOffset(IntTy Offset) const
Return a source location with the specified offset from this SourceLocation.
A trivial tuple used to represent a source range.
SourceLocation getEnd() const
SourceLocation getEndLoc() const LLVM_READONLY
void printPretty(raw_ostream &OS, PrinterHelper *Helper, const PrintingPolicy &Policy, unsigned Indentation=0, StringRef NewlineSymbol="\n", const ASTContext *Context=nullptr) const
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
SourceLocation getBeginLoc() const LLVM_READONLY
StringLiteral - This represents a string literal expression, e.g.
static StringLiteral * Create(const ASTContext &Ctx, StringRef Str, StringLiteralKind Kind, bool Pascal, QualType Ty, ArrayRef< SourceLocation > Locs)
This is the "fully general" constructor that allows representation of strings formed from one or more...
void startDefinition()
Starts the definition of this tag declaration.
Exposes information about the current target.
TargetOptions & getTargetOpts() const
Retrieve the target options.
const llvm::Triple & getTriple() const
Returns the target triple of the primary target.
StringRef getPlatformName() const
Retrieve the name of the platform as it is used in the availability attribute.
VersionTuple getPlatformMinVersion() const
Retrieve the minimum desired version of the platform, to which the program should be compiled.
std::string HLSLEntry
The entry point name for HLSL shader being compiled as specified by -E.
A convenient class for passing around template argument information.
void addArgument(const TemplateArgumentLoc &Loc)
The base class of all kinds of template declarations (e.g., class, function, etc.).
Stores a list of template parameters for a TemplateDecl and its derived classes.
The top declaration context.
SourceLocation getBeginLoc() const
Get the begin source location.
A container of type source information.
TypeLoc getTypeLoc() const
Return the TypeLoc wrapper for the type source info.
The base class of the type hierarchy.
bool isBooleanType() const
bool isIncompleteArrayType() const
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
bool isConstantArrayType() const
bool hasIntegerRepresentation() const
Determine whether this type has an integer representation of some sort, e.g., it is an integer type o...
CXXRecordDecl * castAsCXXRecordDecl() const
bool isArithmeticType() const
bool isConstantMatrixType() const
bool isHLSLBuiltinIntangibleType() const
bool isPointerType() const
CanQualType getCanonicalTypeUnqualified() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
bool isHLSLIntangibleType() const
bool isEnumeralType() const
bool isScalarType() const
bool isIntegralType(const ASTContext &Ctx) const
Determine whether this type is an integral type.
const Type * getArrayElementTypeNoTypeQual() const
If this is an array type, return the element type of the array, potentially with type qualifiers miss...
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool hasUnsignedIntegerRepresentation() const
Determine whether this type has an unsigned integer representation of some sort, e....
bool isDependentType() const
Whether this type is a dependent type, meaning that its definition somehow depends on a template para...
bool isAggregateType() const
Determines whether the type is a C++ aggregate type or C aggregate or union type.
ScalarTypeKind getScalarTypeKind() const
Given that this is a scalar type, classify it.
bool hasSignedIntegerRepresentation() const
Determine whether this type has an signed integer representation of some sort, e.g....
bool isMatrixType() const
bool isHLSLResourceRecord() const
bool hasFloatingRepresentation() const
Determine whether this type has a floating-point representation of some sort, e.g....
bool isVectorType() const
bool isRealFloatingType() const
Floating point categories.
bool isHLSLAttributedResourceType() const
bool isFloatingType() const
const T * getAs() const
Member-template getAs<specific type>'.
const Type * getUnqualifiedDesugaredType() const
Return the specified type with any "sugar" removed from the type, removing any typedefs,...
bool isRecordType() const
bool isHLSLResourceRecordArray() const
void setType(QualType newType)
Represents a variable declaration or definition.
static VarDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, const IdentifierInfo *Id, QualType T, TypeSourceInfo *TInfo, StorageClass S)
void setInitStyle(InitializationStyle Style)
@ CallInit
Call-style initialization (C++98)
void setStorageClass(StorageClass SC)
bool hasGlobalStorage() const
Returns true for all variables that do not have local storage.
StorageClass getStorageClass() const
Returns the storage class as written in the source.
Represents a GCC generic vector type.
unsigned getNumElements() const
QualType getElementType() const
void pushName(llvm::StringRef N)
void pushArrayIndex(uint64_t Index)
void pushBaseName(llvm::StringRef N)
IdentifierInfo * getNameAsIdentifier(ASTContext &AST) const
Defines the clang::TargetInfo interface.
uint32_t getResourceDimensions(llvm::dxil::ResourceDimension Dim)
bool hasCounterHandle(const CXXRecordDecl *RD)
llvm::ImmutableSet< T > join(llvm::ImmutableSet< T > A, llvm::ImmutableSet< T > B, typename llvm::ImmutableSet< T >::Factory &F)
Computes the union of two ImmutableSets.
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
if(T->getSizeExpr()) TRY_TO(TraverseStmt(const_cast< Expr * >(T -> getSizeExpr())))
static bool CheckFloatOrHalfRepresentation(Sema *S, SourceLocation Loc, int ArgOrdinal, clang::QualType PassedType)
@ ICIS_NoInit
No in-class initializer.
@ TemplateName
The identifier is a template name. FIXME: Add an annotation for that.
@ OK_Ordinary
An ordinary object is located at an address in memory.
static bool CheckAllArgTypesAreCorrect(Sema *S, CallExpr *TheCall, llvm::ArrayRef< llvm::function_ref< bool(Sema *, SourceLocation, int, QualType)> > Checks)
@ AANT_ArgumentIdentifier
@ Result
The result type of a method or function.
@ Ordinary
This parameter uses ordinary ABI rules for its type.
llvm::Expected< QualType > ExpectedType
@ Template
We are parsing a template declaration.
LLVM_READONLY bool isDigit(unsigned char c)
Return true if this character is an ASCII digit: [0-9].
static bool CheckAllArgsHaveSameType(Sema *S, CallExpr *TheCall)
@ Type
The name was classified as a type.
LangAS
Defines the address space values used by the address space qualifier of QualType.
bool CreateHLSLAttributedResourceType(Sema &S, QualType Wrapped, ArrayRef< const Attr * > AttrList, QualType &ResType, HLSLAttributedResourceLocInfo *LocInfo=nullptr)
CastKind
CastKind - The kind of operation required for a conversion.
ExprValueKind
The categorization of expression values, currently following the C++11 scheme.
@ VK_PRValue
A pr-value expression (in the C++11 taxonomy) produces a temporary value.
@ VK_LValue
An l-value expression is a reference to an object with independent storage.
DynamicRecursiveASTVisitorBase< false > DynamicRecursiveASTVisitor
U cast(CodeGen::Address addr)
@ None
No keyword precedes the qualified type name.
ActionResult< Expr * > ExprResult
Visibility
Describes the different kinds of visibility that a declaration may have.
hash_code hash_value(const clang::dependencies::ModuleID &ID)
__DEVICE__ bool isnan(float __x)
__DEVICE__ _Tp abs(const std::complex< _Tp > &__c)
__packed_splat4 __packed_splat2 __packed_splat8 __packed_splat4 __packed_splat2 __packed_splat4 __packed_splat2 __packed_splat8 __packed_splat4 uint32_t
TypeSourceInfo * ContainedTyInfo
Describes how types, statements, expressions, and declarations should be printed.
unsigned getImplicitOrderID() const
void setCounterImplicitOrderID(unsigned Value) const
bool hasCounterImplicitOrderID() const
unsigned getSpace() const
bool hasImplicitOrderID() const
void setImplicitOrderID(unsigned Value) const
const SourceLocation & getLocation() const
const llvm::hlsl::rootsig::RootElement & getElement() const