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)
3690 (ResAttrs.IsArray ? 1 : 0);
3692 if (ActualDim != ExpectedDim) {
3694 diag::err_hlsl_builtin_resource_coordinate_dimension_mismatch)
3705 llvm::function_ref<
bool(
const HLSLAttributedResourceType *ResType)> Check =
3709 const HLSLAttributedResourceType *ResTy =
3713 diag::err_typecheck_expect_hlsl_resource)
3717 if (Check && Check(ResTy)) {
3719 diag::err_invalid_hlsl_resource_type)
3727 QualType BaseType,
unsigned ExpectedCount,
3729 unsigned PassedCount = 1;
3731 PassedCount = VecTy->getNumElements();
3733 if (PassedCount != ExpectedCount) {
3736 S->
Diag(Loc, diag::err_typecheck_convert_incompatible)
3746 bool IncludeArraySlice =
true) {
3749 [](
const HLSLAttributedResourceType *ResType) {
3750 return ResType->getAttrs().ResourceDimension ==
3751 llvm::dxil::ResourceDimension::Unknown;
3757 [](
const HLSLAttributedResourceType *ResType) {
3758 return ResType->getAttrs().ResourceClass !=
3759 llvm::hlsl::ResourceClass::Sampler;
3767 unsigned ExpectedDim =
3769 (IncludeArraySlice && ResourceTy->getAttrs().IsArray ? 1 : 0);
3798 unsigned NextIdx = 3;
3804 diag::err_typecheck_convert_incompatible)
3812 Expr *ComponentArg = TheCall->
getArg(NextIdx);
3816 diag::err_typecheck_convert_incompatible)
3823 std::optional<llvm::APSInt> ComponentOpt =
3826 int64_t ComponentVal = ComponentOpt->getSExtValue();
3827 if (ComponentVal != 0) {
3830 assert(ComponentVal >= 0 && ComponentVal <= 3 &&
3831 "The component is not in the expected range.");
3833 diag::err_hlsl_gathercmp_invalid_component)
3843 const HLSLAttributedResourceType *ResourceTy =
3846 unsigned ExpectedDim =
3855 assert(ResourceTy->hasContainedType() &&
3856 "Expecting a contained type for resource with a dimension "
3858 QualType ReturnType = ResourceTy->getContainedType();
3862 S.
Diag(TheCall->
getBeginLoc(), diag::err_hlsl_samplecmp_requires_float);
3868 ReturnType = VecTy->getElementType();
3881 [](
const HLSLAttributedResourceType *ResType) {
3882 return ResType->getAttrs().ResourceDimension ==
3883 llvm::dxil::ResourceDimension::Unknown;
3891 unsigned ResourceDim =
3893 unsigned LocationDim = ResourceDim + (ResourceTy->getAttrs().IsArray ? 1 : 0);
3901 EltTy = VTy->getElementType();
3916 TheCall->
setType(ResourceTy->getContainedType());
3921 unsigned MinArgs, MaxArgs;
3949 const HLSLAttributedResourceType *ResourceTy =
3951 unsigned ExpectedDim =
3954 unsigned NextIdx = 3;
3963 diag::err_typecheck_convert_incompatible)
3998 diag::err_typecheck_convert_incompatible)
4004 assert(ResourceTy->hasContainedType() &&
4005 "Expecting a contained type for resource with a dimension "
4007 QualType ReturnType = ResourceTy->getContainedType();
4010 S.
Diag(TheCall->
getBeginLoc(), diag::err_hlsl_samplecmp_requires_float);
4023 switch (BuiltinID) {
4024 case Builtin::BI__builtin_hlsl_adduint64: {
4025 if (
SemaRef.checkArgCount(TheCall, 2))
4039 if (NumElementsArg != 2 && NumElementsArg != 4) {
4041 << 1 << 64 << NumElementsArg * 32;
4055 case Builtin::BI__builtin_hlsl_resource_getpointer: {
4056 if (
SemaRef.checkArgCountRange(TheCall, 1, 2) ||
4063 QualType ContainedTy = ResourceTy->getContainedType();
4064 auto ReturnType =
SemaRef.Context.getAddrSpaceQualType(
4067 ReturnType =
SemaRef.Context.getPointerType(ReturnType);
4072 case Builtin::BI__builtin_hlsl_resource_getpointer_typed: {
4073 if (
SemaRef.checkArgCount(TheCall, 3) ||
4080 "expected pointer type for second argument");
4087 diag::err_invalid_use_of_array_type);
4091 auto ReturnType =
SemaRef.Context.getAddrSpaceQualType(
4094 ReturnType =
SemaRef.Context.getPointerType(ReturnType);
4099 case Builtin::BI__builtin_hlsl_resource_load_with_status: {
4100 if (
SemaRef.checkArgCount(TheCall, 3) ||
4103 SemaRef.getASTContext().UnsignedIntTy) ||
4105 SemaRef.getASTContext().UnsignedIntTy) ||
4111 QualType ReturnType = ResourceTy->getContainedType();
4116 case Builtin::BI__builtin_hlsl_resource_load_with_status_typed: {
4117 if (
SemaRef.checkArgCount(TheCall, 4) ||
4120 SemaRef.getASTContext().UnsignedIntTy) ||
4122 SemaRef.getASTContext().UnsignedIntTy) ||
4128 "expected pointer type for second argument");
4135 diag::err_invalid_use_of_array_type);
4141 case Builtin::BI__builtin_hlsl_resource_load_level:
4143 case Builtin::BI__builtin_hlsl_resource_sample:
4145 case Builtin::BI__builtin_hlsl_resource_sample_bias:
4147 case Builtin::BI__builtin_hlsl_resource_sample_grad:
4149 case Builtin::BI__builtin_hlsl_resource_sample_level:
4151 case Builtin::BI__builtin_hlsl_resource_sample_cmp:
4153 case Builtin::BI__builtin_hlsl_resource_sample_cmp_level_zero:
4155 case Builtin::BI__builtin_hlsl_resource_calculate_lod:
4156 case Builtin::BI__builtin_hlsl_resource_calculate_lod_unclamped:
4158 case Builtin::BI__builtin_hlsl_resource_gather:
4160 case Builtin::BI__builtin_hlsl_resource_gather_cmp:
4162 case Builtin::BI__builtin_hlsl_resource_uninitializedhandle: {
4163 assert(TheCall->
getNumArgs() == 1 &&
"expected 1 arg");
4169 case Builtin::BI__builtin_hlsl_resource_handlefrombinding: {
4170 assert(TheCall->
getNumArgs() == 6 &&
"expected 6 args");
4176 case Builtin::BI__builtin_hlsl_resource_handlefromimplicitbinding: {
4177 assert(TheCall->
getNumArgs() == 6 &&
"expected 6 args");
4183 case Builtin::BI__builtin_hlsl_resource_counterhandlefromimplicitbinding: {
4184 assert(TheCall->
getNumArgs() == 3 &&
"expected 3 args");
4187 auto *MainResType = MainHandleTy->
getAs<HLSLAttributedResourceType>();
4188 auto MainAttrs = MainResType->getAttrs();
4189 assert(!MainAttrs.IsCounter &&
"cannot create a counter from a counter");
4190 MainAttrs.IsCounter =
true;
4192 MainResType->getWrappedType(), MainResType->getContainedType(),
4196 TheCall->
setType(CounterHandleTy);
4199 case Builtin::BI__builtin_hlsl_and:
4200 case Builtin::BI__builtin_hlsl_or: {
4201 if (
SemaRef.checkArgCount(TheCall, 2))
4215 case Builtin::BI__builtin_hlsl_all:
4216 case Builtin::BI__builtin_hlsl_any: {
4217 if (
SemaRef.checkArgCount(TheCall, 1))
4223 case Builtin::BI__builtin_hlsl_asdouble: {
4224 if (
SemaRef.checkArgCount(TheCall, 2))
4228 SemaRef.Context.UnsignedIntTy,
4233 SemaRef.Context.UnsignedIntTy,
4242 case Builtin::BI__builtin_hlsl_elementwise_clamp: {
4243 if (
SemaRef.BuiltinElementwiseTernaryMath(
4249 case Builtin::BI__builtin_hlsl_dot: {
4251 if (
SemaRef.BuiltinVectorToScalarMath(TheCall))
4257 case Builtin::BI__builtin_hlsl_elementwise_firstbithigh:
4258 case Builtin::BI__builtin_hlsl_elementwise_firstbitlow: {
4259 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4269 EltTy = VecTy->getElementType();
4270 ResTy =
SemaRef.Context.getExtVectorType(ResTy, VecTy->getNumElements());
4283 case Builtin::BI__builtin_hlsl_select: {
4284 if (
SemaRef.checkArgCount(TheCall, 3))
4292 if (VTy && VTy->getElementType()->isBooleanType() &&
4297 case Builtin::BI__builtin_hlsl_elementwise_saturate:
4298 case Builtin::BI__builtin_hlsl_elementwise_rcp: {
4299 if (
SemaRef.checkArgCount(TheCall, 1))
4305 diag::err_builtin_invalid_arg_type)
4308 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4312 case Builtin::BI__builtin_hlsl_elementwise_degrees:
4313 case Builtin::BI__builtin_hlsl_elementwise_radians:
4314 case Builtin::BI__builtin_hlsl_elementwise_rsqrt:
4315 case Builtin::BI__builtin_hlsl_elementwise_frac:
4316 case Builtin::BI__builtin_hlsl_elementwise_ddx_coarse:
4317 case Builtin::BI__builtin_hlsl_elementwise_ddy_coarse:
4318 case Builtin::BI__builtin_hlsl_elementwise_ddx_fine:
4319 case Builtin::BI__builtin_hlsl_elementwise_ddy_fine: {
4320 if (
SemaRef.checkArgCount(TheCall, 1))
4325 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4329 case Builtin::BI__builtin_hlsl_elementwise_isinf:
4330 case Builtin::BI__builtin_hlsl_elementwise_isnan: {
4331 if (
SemaRef.checkArgCount(TheCall, 1))
4336 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4341 case Builtin::BI__builtin_hlsl_lerp: {
4342 if (
SemaRef.checkArgCount(TheCall, 3))
4349 if (
SemaRef.BuiltinElementwiseTernaryMath(TheCall))
4353 case Builtin::BI__builtin_hlsl_mad: {
4354 if (
SemaRef.BuiltinElementwiseTernaryMath(
4360 case Builtin::BI__builtin_hlsl_mul: {
4361 if (
SemaRef.checkArgCount(TheCall, 2))
4370 if (
const auto *VTy = T->getAs<
VectorType>())
4371 return VTy->getElementType();
4373 return MTy->getElementType();
4377 QualType EltTy0 = getElemType(Ty0);
4386 if (IsVec0 && IsMat1) {
4389 }
else if (IsMat0 && IsVec1) {
4393 assert(IsMat0 && IsMat1);
4403 case Builtin::BI__builtin_hlsl_normalize: {
4404 if (
SemaRef.checkArgCount(TheCall, 1))
4415 case Builtin::BI__builtin_elementwise_fma: {
4416 if (
SemaRef.checkArgCount(TheCall, 3) ||
4431 case Builtin::BI__builtin_hlsl_transpose: {
4432 if (
SemaRef.checkArgCount(TheCall, 1))
4441 << 1 << 3 << 0 << 0 << ArgTy;
4446 MatTy->getElementType(), MatTy->getNumColumns(), MatTy->getNumRows());
4450 case Builtin::BI__builtin_hlsl_elementwise_sign: {
4451 if (
SemaRef.PrepareBuiltinElementwiseMathOneArgCall(TheCall))
4459 case Builtin::BI__builtin_hlsl_step: {
4460 if (
SemaRef.checkArgCount(TheCall, 2))
4472 case Builtin::BI__builtin_hlsl_wave_active_all_equal: {
4473 if (
SemaRef.checkArgCount(TheCall, 1))
4487 unsigned NumElts = VecTy->getNumElements();
4497 case Builtin::BI__builtin_hlsl_wave_active_max:
4498 case Builtin::BI__builtin_hlsl_wave_active_min:
4499 case Builtin::BI__builtin_hlsl_wave_active_sum:
4500 case Builtin::BI__builtin_hlsl_wave_active_product: {
4501 if (
SemaRef.checkArgCount(TheCall, 1))
4514 case Builtin::BI__builtin_hlsl_wave_active_bit_or:
4515 case Builtin::BI__builtin_hlsl_wave_active_bit_xor:
4516 case Builtin::BI__builtin_hlsl_wave_active_bit_and: {
4517 if (
SemaRef.checkArgCount(TheCall, 1))
4532 (VTy && VTy->getElementType()->isIntegerType()))) {
4534 diag::err_builtin_invalid_arg_type)
4535 << ArgTyExpr <<
SemaRef.Context.UnsignedIntTy << 1 << 0 << 0;
4543 case Builtin::BI__builtin_hlsl_interlocked_add: {
4553 diag::err_typecheck_call_too_few_args_at_least)
4558 if (
SemaRef.checkArgCountAtMost(TheCall, 3))
4564 diag::err_builtin_invalid_arg_type)
4592 case Builtin::BI__builtin_elementwise_bitreverse: {
4600 case Builtin::BI__builtin_hlsl_wave_prefix_count_bits: {
4601 if (
SemaRef.checkArgCount(TheCall, 1))
4606 if (!(
ArgType->isScalarType())) {
4608 diag::err_typecheck_expect_any_scalar_or_vector)
4613 if (!(
ArgType->isBooleanType())) {
4615 diag::err_typecheck_expect_any_scalar_or_vector)
4622 case Builtin::BI__builtin_hlsl_wave_read_lane_at: {
4623 if (
SemaRef.checkArgCount(TheCall, 2))
4631 diag::err_typecheck_convert_incompatible)
4632 << ArgTyIndex <<
SemaRef.Context.UnsignedIntTy << 1 << 0 << 0;
4645 case Builtin::BI__builtin_hlsl_wave_get_lane_index: {
4646 if (
SemaRef.checkArgCount(TheCall, 0))
4650 case Builtin::BI__builtin_hlsl_wave_prefix_sum:
4651 case Builtin::BI__builtin_hlsl_wave_prefix_product: {
4652 if (
SemaRef.checkArgCount(TheCall, 1))
4665 case Builtin::BI__builtin_hlsl_quad_read_across_x:
4666 case Builtin::BI__builtin_hlsl_quad_read_across_y:
4667 case Builtin::BI__builtin_hlsl_quad_read_across_diagonal: {
4668 if (
SemaRef.checkArgCount(TheCall, 1))
4680 case Builtin::BI__builtin_hlsl_elementwise_splitdouble: {
4681 if (
SemaRef.checkArgCount(TheCall, 3))
4687 SemaRef.Context.UnsignedIntTy, 1) ||
4689 SemaRef.Context.UnsignedIntTy, 2))
4697 case Builtin::BI__builtin_hlsl_elementwise_clip: {
4698 if (
SemaRef.checkArgCount(TheCall, 1))
4705 case Builtin::BI__builtin_elementwise_acos:
4706 case Builtin::BI__builtin_elementwise_asin:
4707 case Builtin::BI__builtin_elementwise_atan:
4708 case Builtin::BI__builtin_elementwise_atan2:
4709 case Builtin::BI__builtin_elementwise_ceil:
4710 case Builtin::BI__builtin_elementwise_cos:
4711 case Builtin::BI__builtin_elementwise_cosh:
4712 case Builtin::BI__builtin_elementwise_exp:
4713 case Builtin::BI__builtin_elementwise_exp2:
4714 case Builtin::BI__builtin_elementwise_exp10:
4715 case Builtin::BI__builtin_elementwise_floor:
4716 case Builtin::BI__builtin_elementwise_fmod:
4717 case Builtin::BI__builtin_elementwise_log:
4718 case Builtin::BI__builtin_elementwise_log2:
4719 case Builtin::BI__builtin_elementwise_log10:
4720 case Builtin::BI__builtin_elementwise_pow:
4721 case Builtin::BI__builtin_elementwise_roundeven:
4722 case Builtin::BI__builtin_elementwise_sin:
4723 case Builtin::BI__builtin_elementwise_sinh:
4724 case Builtin::BI__builtin_elementwise_sqrt:
4725 case Builtin::BI__builtin_elementwise_tan:
4726 case Builtin::BI__builtin_elementwise_tanh:
4727 case Builtin::BI__builtin_elementwise_trunc: {
4733 case Builtin::BI__builtin_hlsl_buffer_update_counter: {
4734 assert(TheCall->
getNumArgs() == 2 &&
"expected 2 args");
4735 auto checkResTy = [](
const HLSLAttributedResourceType *ResTy) ->
bool {
4736 return !(ResTy->getAttrs().ResourceClass == ResourceClass::UAV &&
4737 ResTy->getAttrs().RawBuffer && ResTy->hasContainedType());
4742 std::optional<llvm::APSInt> Offset =
4744 if (!Offset.has_value() ||
std::abs(Offset->getExtValue()) != 1) {
4746 diag::err_hlsl_expect_arg_const_int_one_or_neg_one)
4752 case Builtin::BI__builtin_hlsl_elementwise_f16tof32: {
4753 if (
SemaRef.checkArgCount(TheCall, 1))
4764 ArgTy = VTy->getElementType();
4767 diag::err_builtin_invalid_arg_type)
4776 case Builtin::BI__builtin_hlsl_elementwise_f32tof16: {
4777 if (
SemaRef.checkArgCount(TheCall, 1))
4792 WorkList.push_back(BaseTy);
4793 while (!WorkList.empty()) {
4794 QualType T = WorkList.pop_back_val();
4795 T = T.getCanonicalType().getUnqualifiedType();
4796 if (
const auto *AT = dyn_cast<ConstantArrayType>(T)) {
4804 for (uint64_t Ct = 0; Ct < AT->
getZExtSize(); ++Ct)
4805 llvm::append_range(List, ElementFields);
4810 if (
const auto *VT = dyn_cast<VectorType>(T)) {
4811 List.insert(List.end(), VT->getNumElements(), VT->getElementType());
4814 if (
const auto *MT = dyn_cast<ConstantMatrixType>(T)) {
4815 List.insert(List.end(), MT->getNumElementsFlattened(),
4816 MT->getElementType());
4819 if (
const auto *RD = T->getAsCXXRecordDecl()) {
4820 if (RD->isStandardLayout())
4821 RD = RD->getStandardLayoutBaseWithFields();
4825 if (RD->
isUnion() || !RD->isAggregate()) {
4831 for (
const auto *FD : RD->
fields())
4832 if (!FD->isUnnamedBitField())
4833 FieldTypes.push_back(FD->
getType());
4835 std::reverse(FieldTypes.begin(), FieldTypes.end());
4836 llvm::append_range(WorkList, FieldTypes);
4840 if (!RD->isStandardLayout()) {
4842 for (
const auto &
Base : RD->bases())
4843 FieldTypes.push_back(
Base.getType());
4844 std::reverse(FieldTypes.begin(), FieldTypes.end());
4845 llvm::append_range(WorkList, FieldTypes);
4880 if (
SemaRef.Context.getTypeSize(QT) / 8 > 16)
4886 int ArraySize = VT->getNumElements();
4891 QualType ElTy = VT->getElementType();
4895 if (
SemaRef.Context.getTypeSize(QT) / 8 > 16)
4911 if (
SemaRef.getASTContext().hasSameType(T1, T2))
4920 return llvm::equal(T1Types, T2Types,
4922 return SemaRef.IsLayoutCompatible(LHS, RHS);
4931 bool HadError =
false;
4933 for (
unsigned i = 0, e =
New->getNumParams(); i != e; ++i) {
4941 const auto *NDAttr = NewParam->
getAttr<HLSLParamModifierAttr>();
4942 unsigned NSpellingIdx = (NDAttr ? NDAttr->getSpellingListIndex() : 0);
4943 const auto *ODAttr = OldParam->
getAttr<HLSLParamModifierAttr>();
4944 unsigned OSpellingIdx = (ODAttr ? ODAttr->getSpellingListIndex() : 0);
4946 if (NSpellingIdx != OSpellingIdx) {
4948 diag::err_hlsl_param_qualifier_mismatch)
4949 << NDAttr << NewParam;
4965 if (
SemaRef.getASTContext().hasSameUnqualifiedType(SrcTy, DestTy))
4980 llvm_unreachable(
"HLSL doesn't support pointers.");
4983 llvm_unreachable(
"HLSL doesn't support complex types.");
4985 llvm_unreachable(
"HLSL doesn't support fixed point types.");
4987 llvm_unreachable(
"Should have returned before this");
4997 llvm_unreachable(
"HLSL doesn't support complex types.");
4999 llvm_unreachable(
"HLSL doesn't support fixed point types.");
5004 llvm_unreachable(
"HLSL doesn't support pointers.");
5006 llvm_unreachable(
"Should have returned before this");
5012 llvm_unreachable(
"HLSL doesn't support pointers.");
5015 llvm_unreachable(
"HLSL doesn't support fixed point types.");
5019 llvm_unreachable(
"HLSL doesn't support complex types.");
5022 llvm_unreachable(
"Unhandled scalar cast");
5043 !(SrcMatTy && SrcMatTy->getNumElementsFlattened() == 1))
5049 SrcTy = SrcMatTy->getElementType();
5054 for (
unsigned I = 0, Size = DestTypes.size(); I < Size; ++I) {
5055 if (DestTypes[I]->isUnionType())
5087 if (SrcTypes.size() < DestTypes.size())
5090 unsigned SrcSize = SrcTypes.size();
5091 unsigned DstSize = DestTypes.size();
5093 for (I = 0; I < DstSize && I < SrcSize; I++) {
5094 if (SrcTypes[I]->isUnionType() || DestTypes[I]->isUnionType())
5102 for (; I < SrcSize; I++) {
5103 if (SrcTypes[I]->isUnionType())
5110 assert(Param->hasAttr<HLSLParamModifierAttr>() &&
5111 "We should not get here without a parameter modifier expression");
5112 const auto *
Attr = Param->getAttr<HLSLParamModifierAttr>();
5119 << Arg << (IsInOut ? 1 : 0);
5125 QualType Ty = Param->getType().getNonLValueExprType(Ctx);
5132 << Arg << (IsInOut ? 1 : 0);
5144 SemaRef.PerformCopyInitialization(Entity, Param->getBeginLoc(), ArgOpV);
5150 auto *OpV =
new (Ctx)
5155 Res =
SemaRef.ActOnBinOp(
SemaRef.getCurScope(), Param->getBeginLoc(),
5156 tok::equal, ArgOpV, OpV);
5172 "Pointer and reference types cannot be inout or out parameters");
5173 Ty =
SemaRef.getASTContext().getLValueReferenceType(Ty);
5189 for (
const auto *FD : RD->
fields()) {
5193 assert(RD->getNumBases() <= 1 &&
5194 "HLSL doesn't support multiple inheritance");
5195 return RD->getNumBases()
5200 if (
const auto *AT = dyn_cast<ArrayType>(Ty)) {
5201 if (
const auto *CAT = dyn_cast<ConstantArrayType>(AT))
5213 bool IsVKPushConstant = IsVulkan && VD->
hasAttr<HLSLVkPushConstantAttr>();
5218 !VD->
hasAttr<HLSLVkConstantIdAttr>() && !IsVKPushConstant &&
5224 if (
Decl->getType().hasAddressSpace())
5227 if (
Decl->getType()->isDependentType())
5239 if (
Decl->
hasAttr<HLSLVkExtBuiltinOutputAttr>()) {
5253 llvm::Triple::Vulkan;
5254 if (IsVulkan &&
Decl->
hasAttr<HLSLVkPushConstantAttr>()) {
5255 if (HasDeclaredAPushConstant)
5261 HasDeclaredAPushConstant =
true;
5288class StructBindingContext {
5291 HLSLResourceBindingAttr *RegBindingsAttrs[4];
5292 unsigned RegBindingOffset[4];
5295 static_assert(
static_cast<unsigned>(RegisterType::SRV) == 0 &&
5296 static_cast<unsigned>(RegisterType::UAV) == 1 &&
5297 static_cast<unsigned>(RegisterType::CBuffer) == 2 &&
5298 static_cast<unsigned>(RegisterType::Sampler) == 3,
5299 "unexpected register type values");
5302 HLSLVkBindingAttr *VkBindingAttr;
5303 unsigned VkBindingOffset;
5308 StructBindingContext(
VarDecl *VD) {
5309 for (
unsigned i = 0; i < 4; ++i) {
5310 RegBindingsAttrs[i] =
nullptr;
5311 RegBindingOffset[i] = 0;
5313 VkBindingAttr =
nullptr;
5314 VkBindingOffset = 0;
5320 if (
auto *RBA = dyn_cast<HLSLResourceBindingAttr>(A)) {
5322 unsigned RegTypeIdx =
static_cast<unsigned>(RegType);
5325 RegBindingsAttrs[RegTypeIdx] = RBA;
5330 if (
auto *VBA = dyn_cast<HLSLVkBindingAttr>(A))
5331 VkBindingAttr = VBA;
5338 Attr *createBindingAttr(SemaHLSL &S, ASTContext &AST,
RegisterType RegType,
5339 unsigned Range,
bool HasCounter) {
5340 assert(
static_cast<unsigned>(RegType) < 4 &&
"unexpected register type");
5342 if (VkBindingAttr) {
5343 unsigned Offset = VkBindingOffset;
5344 VkBindingOffset +=
Range;
5345 return HLSLVkBindingAttr::CreateImplicit(
5346 AST, VkBindingAttr->getBinding() + Offset, VkBindingAttr->getSet(),
5347 VkBindingAttr->getRange());
5350 HLSLResourceBindingAttr *RBA =
5351 RegBindingsAttrs[
static_cast<unsigned>(RegType)];
5352 HLSLResourceBindingAttr *NewAttr =
nullptr;
5354 if (RBA && RBA->hasRegisterSlot()) {
5357 unsigned Offset = RegBindingOffset[
static_cast<unsigned>(RegType)];
5358 RegBindingOffset[
static_cast<unsigned>(RegType)] += Range;
5360 unsigned NewSlotNumber = RBA->getSlotNumber() + Offset;
5361 StringRef NewSlotNumberStr =
5363 NewAttr = HLSLResourceBindingAttr::CreateImplicit(
5364 AST, NewSlotNumberStr, RBA->getSpace(), RBA->getRange());
5365 NewAttr->setBinding(RegType, NewSlotNumber, RBA->getSpaceNumber());
5369 NewAttr = HLSLResourceBindingAttr::CreateImplicit(AST,
"",
"0", {});
5370 NewAttr->setBinding(RegType, std::nullopt,
5371 RBA ? RBA->getSpaceNumber() : 0);
5375 NewAttr->setImplicitCounterBindingOrderID(
5384static void createGlobalResourceDeclForStruct(
5386 QualType ResTy, StructBindingContext &BindingCtx) {
5388 "expected resource type or array of resources");
5399 while (
const auto *AT = dyn_cast<ArrayType>(SingleResTy)) {
5400 const auto *CAT = dyn_cast<ConstantArrayType>(AT);
5405 const HLSLAttributedResourceType *ResHandleTy =
5406 HLSLAttributedResourceType::findHandleTypeOnResource(SingleResTy);
5410 Attr *BindingAttr = BindingCtx.createBindingAttr(
5412 ResDecl->
addAttr(BindingAttr);
5413 ResDecl->
addAttr(InternalLinkageAttr::CreateImplicit(AST));
5422 HLSLAssociatedResourceDeclAttr::CreateImplicit(AST, ResDecl));
5429static void handleArrayOfStructWithResources(
5431 EmbeddedResourceNameBuilder &NameBuilder, StructBindingContext &BindingCtx);
5436static void handleStructWithResources(
Sema &S,
VarDecl *ParentVD,
5438 EmbeddedResourceNameBuilder &NameBuilder,
5439 StructBindingContext &BindingCtx) {
5442 assert(RD->
getNumBases() <= 1 &&
"HLSL doesn't support multiple inheritance");
5449 handleStructWithResources(S, ParentVD, BaseRD, NameBuilder, BindingCtx);
5463 createGlobalResourceDeclForStruct(S, ParentVD, FD->
getLocation(), II,
5466 handleStructWithResources(S, ParentVD, RD, NameBuilder, BindingCtx);
5468 }
else if (
const auto *ArrayTy = dyn_cast<ConstantArrayType>(FDTy)) {
5470 "resource arrays should have been already handled");
5471 handleArrayOfStructWithResources(S, ParentVD, ArrayTy, NameBuilder,
5480handleArrayOfStructWithResources(
Sema &S,
VarDecl *ParentVD,
5482 EmbeddedResourceNameBuilder &NameBuilder,
5483 StructBindingContext &BindingCtx) {
5491 if (!SubCAT && !ElementRD)
5494 for (
unsigned I = 0, E = CAT->
getSize().getZExtValue(); I < E; ++I) {
5497 handleStructWithResources(S, ParentVD, ElementRD, NameBuilder,
5500 handleArrayOfStructWithResources(S, ParentVD, SubCAT, NameBuilder,
5513void SemaHLSL::handleGlobalStructOrArrayOfWithResources(
VarDecl *VD) {
5514 EmbeddedResourceNameBuilder NameBuilder(VD->
getName());
5515 StructBindingContext BindingCtx(VD);
5519 "Expected non-resource struct or array type");
5522 handleStructWithResources(
SemaRef, VD, RD, NameBuilder, BindingCtx);
5526 if (
const auto *CAT = dyn_cast<ConstantArrayType>(VDTy)) {
5527 handleArrayOfStructWithResources(
SemaRef, VD, CAT, NameBuilder, BindingCtx);
5535 if (
SemaRef.RequireCompleteType(
5538 diag::err_typecheck_decl_incomplete_type)) {
5552 DefaultCBufferDecls.push_back(VD);
5557 collectResourceBindingsOnVarDecl(VD);
5559 if (VD->
hasAttr<HLSLVkConstantIdAttr>())
5571 processExplicitBindingsOnDecl(VD);
5609 handleGlobalStructOrArrayOfWithResources(VD);
5613 if (VD->
hasAttr<HLSLGroupSharedAddressSpaceAttr>())
5622 "expected resource record type");
5638 const char *CreateMethodName;
5640 CreateMethodName = HasCounter ?
"__createFromBindingWithImplicitCounter"
5641 :
"__createFromBinding";
5643 CreateMethodName = HasCounter
5644 ?
"__createFromImplicitBindingWithImplicitCounter"
5645 :
"__createFromImplicitBinding";
5650 if (!CreateMethod) {
5655 "create method lookup should always succeed for built-in resource "
5664 Args.push_back(RegSlot);
5672 Args.push_back(OrderId);
5678 Args.push_back(Space);
5682 Args.push_back(RangeSize);
5686 Args.push_back(Index);
5688 StringRef VarName = VD->
getName();
5696 Args.push_back(NameCast);
5704 Args.push_back(CounterId);
5727 SemaRef.CheckCompleteVariableDeclaration(VD);
5733 "expected array of resource records");
5754 lookupMethod(
SemaRef, ResourceDecl,
5755 HasCounter ?
"__createFromBindingWithImplicitCounter"
5756 :
"__createFromBinding",
5760 CreateMethod = lookupMethod(
5762 HasCounter ?
"__createFromImplicitBindingWithImplicitCounter"
5763 :
"__createFromImplicitBinding",
5806std::optional<const DeclBindingInfo *> SemaHLSL::inferGlobalBinding(
Expr *E) {
5807 if (
auto *Ternary = dyn_cast<ConditionalOperator>(E)) {
5808 auto TrueInfo = inferGlobalBinding(Ternary->getTrueExpr());
5809 auto FalseInfo = inferGlobalBinding(Ternary->getFalseExpr());
5810 if (!TrueInfo || !FalseInfo)
5811 return std::nullopt;
5812 if (*TrueInfo != *FalseInfo)
5813 return std::nullopt;
5817 if (
auto *ASE = dyn_cast<ArraySubscriptExpr>(E))
5826 if (
const auto *AttrResType =
5827 HLSLAttributedResourceType::findHandleTypeOnResource(Ty)) {
5829 return Bindings.getDeclBindingInfo(VD, RC);
5836void SemaHLSL::trackLocalResource(
VarDecl *VD,
Expr *E) {
5837 std::optional<const DeclBindingInfo *> ExprBinding = inferGlobalBinding(E);
5840 diag::warn_hlsl_assigning_local_resource_is_not_unique)
5845 if (*ExprBinding ==
nullptr)
5848 auto PrevBinding = Assigns.find(VD);
5849 if (PrevBinding == Assigns.end()) {
5851 Assigns.insert({VD, *ExprBinding});
5856 if (*ExprBinding != PrevBinding->second) {
5858 diag::warn_hlsl_assigning_local_resource_is_not_unique)
5860 SemaRef.Diag(VD->getLocation(), diag::note_var_declared_here) << VD;
5871 "expected LHS to be a resource record or array of resource records");
5872 if (Opc != BO_Assign)
5877 while (
auto *ASE = dyn_cast<ArraySubscriptExpr>(E))
5885 SemaRef.Diag(Loc, diag::err_hlsl_assign_to_global_resource) << VD;
5890 trackLocalResource(VD, RHSExpr);
5907void SemaHLSL::collectResourceBindingsOnVarDecl(
VarDecl *VD) {
5909 "expected global variable that contains HLSL resource");
5912 if (
const HLSLBufferDecl *CBufferOrTBuffer = dyn_cast<HLSLBufferDecl>(VD)) {
5913 Bindings.addDeclBindingInfo(VD, CBufferOrTBuffer->isCBuffer()
5914 ? ResourceClass::CBuffer
5915 : ResourceClass::SRV);
5928 if (
const HLSLAttributedResourceType *AttrResType =
5929 HLSLAttributedResourceType::findHandleTypeOnResource(Ty)) {
5930 Bindings.addDeclBindingInfo(VD, AttrResType->getAttrs().ResourceClass);
5935 if (
const RecordType *RT = dyn_cast<RecordType>(Ty))
5936 collectResourceBindingsOnUserRecordDecl(VD, RT);
5942void SemaHLSL::processExplicitBindingsOnDecl(
VarDecl *VD) {
5945 bool HasBinding =
false;
5946 for (Attr *A : VD->
attrs()) {
5949 if (
auto PA = VD->
getAttr<HLSLVkPushConstantAttr>())
5950 Diag(PA->getLoc(), diag::err_hlsl_attr_incompatible) << A << PA;
5953 HLSLResourceBindingAttr *RBA = dyn_cast<HLSLResourceBindingAttr>(A);
5954 if (!RBA || !RBA->hasRegisterSlot())
5959 assert(RT != RegisterType::I &&
"invalid or obsolete register type should "
5960 "never have an attribute created");
5962 if (RT == RegisterType::C) {
5963 if (Bindings.hasBindingInfoForDecl(VD))
5965 diag::warn_hlsl_user_defined_type_missing_member)
5966 <<
static_cast<int>(RT);
5974 if (DeclBindingInfo *BI = Bindings.getDeclBindingInfo(VD, RC)) {
5979 diag::warn_hlsl_user_defined_type_missing_member)
5980 <<
static_cast<int>(RT);
5988class InitListTransformer {
5992 QualType *DstIt =
nullptr;
5993 Expr **ArgIt =
nullptr;
5999 bool castInitializer(Expr *E) {
6000 assert(DstIt &&
"This should always be something!");
6001 if (DstIt == DestTypes.end()) {
6003 ArgExprs.push_back(E);
6008 DstIt = DestTypes.begin();
6011 Ctx, *DstIt,
false);
6016 ArgExprs.push_back(
Init);
6021 bool buildInitializerListImpl(Expr *E) {
6023 if (
auto *
Init = dyn_cast<InitListExpr>(E)) {
6024 for (
auto *SubInit :
Init->inits())
6025 if (!buildInitializerListImpl(SubInit))
6035 return castInitializer(E);
6049 if (
auto *VecTy = Ty->
getAs<VectorType>()) {
6054 for (uint64_t I = 0; I <
Size; ++I) {
6056 SizeTy, SourceLocation());
6062 if (!castInitializer(ElExpr.
get()))
6067 if (
auto *MTy = Ty->
getAs<ConstantMatrixType>()) {
6068 unsigned Rows = MTy->getNumRows();
6069 unsigned Cols = MTy->getNumColumns();
6070 QualType ElemTy = MTy->getElementType();
6072 for (
unsigned R = 0;
R < Rows; ++
R) {
6073 for (
unsigned C = 0;
C < Cols; ++
C) {
6086 if (!castInitializer(ElExpr.
get()))
6094 if (
auto *ArrTy = dyn_cast<ConstantArrayType>(Ty.
getTypePtr())) {
6098 for (uint64_t I = 0; I <
Size; ++I) {
6100 SizeTy, SourceLocation());
6105 if (!buildInitializerListImpl(ElExpr.
get()))
6112 llvm::SmallVector<CXXRecordDecl *> RecordDecls;
6113 RecordDecls.push_back(RD);
6114 while (RecordDecls.back()->getNumBases()) {
6115 CXXRecordDecl *D = RecordDecls.back();
6117 "HLSL doesn't support multiple inheritance");
6118 RecordDecls.push_back(
6121 while (!RecordDecls.empty()) {
6122 CXXRecordDecl *RD = RecordDecls.pop_back_val();
6123 for (
auto *FD : RD->
fields()) {
6124 if (FD->isUnnamedBitField())
6132 if (!buildInitializerListImpl(Res.
get()))
6140 Expr *generateInitListsImpl(QualType Ty) {
6142 assert(ArgIt != ArgExprs.end() &&
"Something is off in iteration!");
6147 llvm::SmallVector<Expr *>
Inits;
6152 if (
auto *ATy = Ty->
getAs<VectorType>()) {
6153 ElTy = ATy->getElementType();
6154 Size = ATy->getNumElements();
6155 }
else if (
auto *CMTy = Ty->
getAs<ConstantMatrixType>()) {
6156 ElTy = CMTy->getElementType();
6157 Size = CMTy->getNumElementsFlattened();
6160 ElTy = VTy->getElementType();
6161 Size = VTy->getZExtSize();
6163 for (uint64_t I = 0; I <
Size; ++I)
6164 Inits.push_back(generateInitListsImpl(ElTy));
6167 llvm::SmallVector<CXXRecordDecl *> RecordDecls;
6168 RecordDecls.push_back(RD);
6169 while (RecordDecls.back()->getNumBases()) {
6170 CXXRecordDecl *D = RecordDecls.back();
6172 "HLSL doesn't support multiple inheritance");
6173 RecordDecls.push_back(
6176 while (!RecordDecls.empty()) {
6177 CXXRecordDecl *RD = RecordDecls.pop_back_val();
6178 for (
auto *FD : RD->
fields())
6179 if (!FD->isUnnamedBitField())
6184 new (Ctx) InitListExpr(Ctx,
Inits.front()->getBeginLoc(),
Inits,
6185 Inits.back()->getEndLoc(),
false);
6186 NewInit->setType(Ty);
6191 llvm::SmallVector<QualType, 16> DestTypes;
6192 llvm::SmallVector<Expr *, 16> ArgExprs;
6193 InitListTransformer(Sema &SemaRef,
const InitializedEntity &Entity)
6194 : S(SemaRef), Ctx(SemaRef.getASTContext()),
6195 Wrap(Entity.
getType()->isIncompleteArrayType()) {
6196 InitTy = Entity.
getType().getNonReferenceType();
6206 DstIt = DestTypes.begin();
6209 bool buildInitializerList(Expr *E) {
return buildInitializerListImpl(E); }
6211 Expr *generateInitLists() {
6212 assert(!ArgExprs.empty() &&
6213 "Call buildInitializerList to generate argument expressions.");
6214 ArgIt = ArgExprs.begin();
6216 return generateInitListsImpl(InitTy);
6217 llvm::SmallVector<Expr *>
Inits;
6218 while (ArgIt != ArgExprs.end())
6219 Inits.push_back(generateInitListsImpl(InitTy));
6222 new (Ctx) InitListExpr(Ctx,
Inits.front()->getBeginLoc(),
Inits,
6223 Inits.back()->getEndLoc(),
false);
6224 llvm::APInt ArySize(64,
Inits.size());
6226 ArraySizeModifier::Normal, 0));
6238 if (
const ArrayType *AT = dyn_cast<ArrayType>(Ty)) {
6245 if (
const auto *RT = Ty->
getAs<RecordType>()) {
6249 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD)) {
6269 if (
Init->getType()->isScalarType())
6272 InitListTransformer ILT(
SemaRef, Entity);
6274 for (
unsigned I = 0; I <
Init->getNumInits(); ++I) {
6282 Init->setInit(I, E);
6284 if (!ILT.buildInitializerList(E))
6287 size_t ExpectedSize = ILT.DestTypes.size();
6288 size_t ActualSize = ILT.ArgExprs.size();
6289 if (ExpectedSize == 0 && ActualSize == 0)
6296 InitTy =
SemaRef.getASTContext().removeAddrSpaceQualType(InitTy);
6298 SemaRef.Diag(
Init->getBeginLoc(), diag::err_hlsl_incorrect_num_initializers)
6299 << (int)(ExpectedSize < ActualSize) << InitTy
6300 << ExpectedSize << ActualSize;
6310 assert(ExpectedSize > 0 &&
6311 "The expected size of an incomplete array type must be at least 1.");
6313 ((ActualSize + ExpectedSize - 1) / ExpectedSize) * ExpectedSize;
6321 InitTy =
SemaRef.getASTContext().removeAddrSpaceQualType(InitTy);
6322 if (ExpectedSize != ActualSize) {
6323 int TooManyOrFew = ActualSize > ExpectedSize ? 1 : 0;
6324 SemaRef.Diag(
Init->getBeginLoc(), diag::err_hlsl_incorrect_num_initializers)
6325 << TooManyOrFew << InitTy << ExpectedSize << ActualSize;
6332 Init->resizeInits(Ctx, NewInit->getNumInits());
6333 for (
unsigned I = 0; I < NewInit->getNumInits(); ++I)
6334 Init->updateInit(Ctx, I, NewInit->getInit(I));
6342 S.
Diag(OpLoc, diag::err_builtin_matrix_invalid_member)
6352 StringRef AccessorName = CompName->
getName();
6353 assert(!AccessorName.empty() &&
"Matrix Accessor must have a name");
6355 unsigned Rows = MT->getNumRows();
6356 unsigned Cols = MT->getNumColumns();
6357 bool IsZeroBasedAccessor =
false;
6358 unsigned ChunkLen = 0;
6359 if (AccessorName.size() < 2)
6361 "length 4 for zero based: \'_mRC\' or "
6362 "length 3 for one-based: \'_RC\' accessor",
6365 if (AccessorName[0] ==
'_') {
6366 if (AccessorName[1] ==
'm') {
6367 IsZeroBasedAccessor =
true;
6374 S, AccessorName,
"zero based: \'_mRC\' or one-based: \'_RC\' accessor",
6377 if (AccessorName.size() % ChunkLen != 0) {
6378 const llvm::StringRef
Expected = IsZeroBasedAccessor
6379 ?
"zero based: '_mRC' accessor"
6380 :
"one-based: '_RC' accessor";
6385 auto isDigit = [](
char c) {
return c >=
'0' && c <=
'9'; };
6386 auto isZeroBasedIndex = [](
unsigned i) {
return i <= 3; };
6387 auto isOneBasedIndex = [](
unsigned i) {
return i >= 1 && i <= 4; };
6389 bool HasRepeated =
false;
6391 unsigned NumComponents = 0;
6392 const char *Begin = AccessorName.data();
6394 for (
unsigned I = 0, E = AccessorName.size(); I < E; I += ChunkLen) {
6395 const char *Chunk = Begin + I;
6396 char RowChar = 0, ColChar = 0;
6397 if (IsZeroBasedAccessor) {
6399 if (Chunk[0] !=
'_' || Chunk[1] !=
'm') {
6400 char Bad = (Chunk[0] !=
'_') ? Chunk[0] : Chunk[1];
6402 S, StringRef(&Bad, 1),
"\'_m\' prefix",
6409 if (Chunk[0] !=
'_')
6411 S, StringRef(&Chunk[0], 1),
"\'_\' prefix",
6418 bool IsDigitsError =
false;
6420 unsigned BadPos = IsZeroBasedAccessor ? 2 : 1;
6424 IsDigitsError =
true;
6428 unsigned BadPos = IsZeroBasedAccessor ? 3 : 2;
6432 IsDigitsError =
true;
6437 unsigned Row = RowChar -
'0';
6438 unsigned Col = ColChar -
'0';
6440 bool HasIndexingError =
false;
6441 if (IsZeroBasedAccessor) {
6443 if (!isZeroBasedIndex(Row)) {
6444 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6446 HasIndexingError =
true;
6448 if (!isZeroBasedIndex(Col)) {
6449 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6451 HasIndexingError =
true;
6455 if (!isOneBasedIndex(Row)) {
6456 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6458 HasIndexingError =
true;
6460 if (!isOneBasedIndex(Col)) {
6461 S.
Diag(OpLoc, diag::err_hlsl_matrix_element_not_in_bounds)
6463 HasIndexingError =
true;
6470 if (HasIndexingError)
6476 bool HasBoundsError =
false;
6478 Diag(OpLoc, diag::err_hlsl_matrix_index_out_of_bounds)
6480 HasBoundsError =
true;
6483 Diag(OpLoc, diag::err_hlsl_matrix_index_out_of_bounds)
6485 HasBoundsError =
true;
6490 unsigned FlatIndex = Row * Cols + Col;
6491 if (Seen[FlatIndex])
6493 Seen[FlatIndex] =
true;
6496 if (NumComponents == 0 || NumComponents > 4) {
6497 S.
Diag(OpLoc, diag::err_hlsl_matrix_swizzle_invalid_length)
6502 QualType ElemTy = MT->getElementType();
6503 if (NumComponents == 1)
6509 for (Sema::ExtVectorDeclsType::iterator
6513 if ((*I)->getUnderlyingType() == VT)
6524 trackLocalResource(VDecl,
Init);
6526 const HLSLVkConstantIdAttr *ConstIdAttr =
6527 VDecl->
getAttr<HLSLVkConstantIdAttr>();
6534 if (!
Init->isCXX11ConstantExpr(Context, &InitValue)) {
6544 int ConstantID = ConstIdAttr->getId();
6545 llvm::APInt IDVal(Context.getIntWidth(Context.IntTy), ConstantID);
6547 ConstIdAttr->getLocation());
6551 if (
C->getType()->getCanonicalTypeUnqualified() !=
6555 Context.getTrivialTypeSourceInfo(
6556 Init->getType(),
Init->getExprLoc()),
6575 if (!Params || Params->
size() != 1)
6588 if (
auto *TTP = dyn_cast<TemplateTypeParmDecl>(P)) {
6589 if (TTP->hasDefaultArgument()) {
6590 TemplateArgs.
addArgument(TTP->getDefaultArgument());
6593 }
else if (
auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(P)) {
6594 if (NTTP->hasDefaultArgument()) {
6595 TemplateArgs.
addArgument(NTTP->getDefaultArgument());
6598 }
else if (
auto *TTPD = dyn_cast<TemplateTemplateParmDecl>(P)) {
6599 if (TTPD->hasDefaultArgument()) {
6600 TemplateArgs.
addArgument(TTPD->getDefaultArgument());
6607 return SemaRef.CheckTemplateIdType(
6609 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 CheckTextureSamplerAndLocation(Sema &S, CallExpr *TheCall, bool IncludeArraySlice=true)
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 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