Index: clang/test/Headers/lit.local.cfg =================================================================== --- /dev/null +++ clang/test/Headers/lit.local.cfg @@ -0,0 +1,4 @@ +config.substitutions = list(config.substitutions) + +# Enable substituting Clang Sema source directory for TableGen input. +config.substitutions.append(('%clang_src_sema_dir', os.path.join(config.clang_src_dir, 'lib', 'Sema'))) Index: clang/test/Headers/opencl-builtins.cl =================================================================== --- /dev/null +++ clang/test/Headers/opencl-builtins.cl @@ -0,0 +1,19 @@ +// RUN: clang-tblgen -gen-clang-opencl-builtin-tests %clang_src_sema_dir/OpenCLBuiltins.td -o %t.cl +// RUN: %clang_cc1 -include %s %t.cl -triple spir -verify -fsyntax-only -cl-std=CL2.0 -finclude-default-header +// RUN: %clang_cc1 -include %s %t.cl -triple spir -verify -fsyntax-only -cl-std=CL2.0 -fdeclare-opencl-builtins -finclude-default-header + +// Generate an OpenCL source file containing a call to each builtin from +// OpenCLBuiltins.td and then run that generated source file through the +// frontend. +// +// Then test that: +// - The generated file can be parsed using opencl-c.h, giving some confidence +// that OpenCLBuiltins.td does not provide more than what opencl-c.h provides +// (but not vice versa). +// +// - The generated file can be parsed using -fdeclare-opencl-builtins, ensuring +// some internal consistency of declarations in OpenCLBuiltins.td. For +// example, addition of builtin declarations that lead to ambiguity during +// overload resolution will cause this test to fail. + +// expected-no-diagnostics Index: clang/utils/TableGen/ClangOpenCLBuiltinEmitter.cpp =================================================================== --- clang/utils/TableGen/ClangOpenCLBuiltinEmitter.cpp +++ clang/utils/TableGen/ClangOpenCLBuiltinEmitter.cpp @@ -228,6 +228,64 @@ // same entry (). MapVector SignatureListMap; }; + +// OpenCL builtin test generator. This class processes the same TableGen input +// as BuiltinNameEmitter, but generates a .cl file that contains a call to each +// builtin function described in the .td input. +class OpenCLBuiltinTestEmitter { +public: + OpenCLBuiltinTestEmitter(RecordKeeper &Records, raw_ostream &OS) + : Records(Records), OS(OS) {} + + // Entrypoint to generate the functions for testing all OpenCL builtin + // functions. + void emit(); + +private: + struct TypeFlags { + TypeFlags() : IsConst(false), IsVolatile(false), IsPointer(false) {} + bool IsConst : 1; + bool IsVolatile : 1; + bool IsPointer : 1; + StringRef AddrSpace; + }; + + // Return a string representation of the given type, such that it can be + // used as a type in OpenCL C code. + std::string getTypeString(const Record *Type, TypeFlags Flags, + int VectorSize) const; + + // Return the type(s) and vector size(s) for the given type. For + // non-GenericTypes, the resulting vectors will contain 1 element. For + // GenericTypes, the resulting vectors typically contain multiple elements. + void getTypeLists(Record *Type, TypeFlags &Flags, + std::vector &TypeList, + std::vector &VectorList) const; + + // Expand the TableGen Records representing a builtin function signature into + // one or more function signatures. Return them as a vector of a vector of + // strings, with each string containing an OpenCL C type and optional + // qualifiers. + // + // The Records may contain GenericTypes, which expand into multiple + // signatures. Repeated occurrences of GenericType in a signature expand to + // the same types. For example [char, FGenType, FGenType] expands to: + // [char, float, float] + // [char, float2, float2] + // [char, float3, float3] + // ... + void + expandTypesInSignature(const std::vector &Signature, + SmallVectorImpl> &Types); + + // Contains OpenCL builtin functions and related information, stored as + // Record instances. They are coming from the associated TableGen file. + RecordKeeper &Records; + + // The output file. + raw_ostream &OS; +}; + } // namespace void BuiltinNameEmitter::Emit() { @@ -861,7 +919,230 @@ OS << "\n} // OCL2Qual\n"; } +std::string OpenCLBuiltinTestEmitter::getTypeString(const Record *Type, + TypeFlags Flags, + int VectorSize) const { + std::string S; + if (Type->getValueAsBit("IsConst") || Flags.IsConst) { + S += "const "; + } + if (Type->getValueAsBit("IsVolatile") || Flags.IsVolatile) { + S += "volatile "; + } + + auto PrintAddrSpace = [&S](StringRef AddrSpace) { + S += StringSwitch(AddrSpace) + .Case("clang::LangAS::opencl_private", "__private") + .Case("clang::LangAS::opencl_global", "__global") + .Case("clang::LangAS::opencl_constant", "__constant") + .Case("clang::LangAS::opencl_local", "__local") + .Case("clang::LangAS::opencl_generic", "__generic") + .Default("__private"); + S += " "; + }; + if (Flags.IsPointer) { + PrintAddrSpace(Flags.AddrSpace); + } else if (Type->getValueAsBit("IsPointer")) { + PrintAddrSpace(Type->getValueAsString("AddrSpace")); + } + + StringRef Acc = Type->getValueAsString("AccessQualifier"); + if (Acc != "") { + S += StringSwitch(Acc) + .Case("RO", "__read_only ") + .Case("WO", "__write_only ") + .Case("RW", "__read_write "); + } + + S += Type->getValueAsString("Name").str(); + if (VectorSize > 1) { + S += std::to_string(VectorSize); + } + + if (Type->getValueAsBit("IsPointer") || Flags.IsPointer) { + S += " *"; + } + + return S; +} + +void OpenCLBuiltinTestEmitter::getTypeLists( + Record *Type, TypeFlags &Flags, std::vector &TypeList, + std::vector &VectorList) const { + bool isGenType = Type->isSubClassOf("GenericType"); + if (isGenType) { + TypeList = Type->getValueAsDef("TypeList")->getValueAsListOfDefs("List"); + VectorList = + Type->getValueAsDef("VectorList")->getValueAsListOfInts("List"); + return; + } + + if (Type->isSubClassOf("PointerType") || Type->isSubClassOf("ConstType") || + Type->isSubClassOf("VolatileType")) { + StringRef SubTypeName = Type->getValueAsString("Name"); + Record *PossibleGenType = Records.getDef(SubTypeName); + if (PossibleGenType && PossibleGenType->isSubClassOf("GenericType")) { + // When PointerType, ConstType, or VolatileType is applied to a + // GenericType, the flags need to be taken from the subtype, not from the + // GenericType. + Flags.IsPointer = Type->getValueAsBit("IsPointer"); + Flags.IsConst = Type->getValueAsBit("IsConst"); + Flags.IsVolatile = Type->getValueAsBit("IsVolatile"); + Flags.AddrSpace = Type->getValueAsString("AddrSpace"); + getTypeLists(PossibleGenType, Flags, TypeList, VectorList); + return; + } + } + + // Not a GenericType, so just insert the single type. + TypeList.push_back(Type); + VectorList.push_back(Type->getValueAsInt("VecWidth")); +} + +void OpenCLBuiltinTestEmitter::expandTypesInSignature( + const std::vector &Signature, + SmallVectorImpl> &Types) { + // Find out if there are any GenTypes in this signature, and if so, calculate + // into how many signatures they will expand. + unsigned NumSignatures = 1; + SmallVector, 4> ExpandedGenTypes; + for (const auto &Arg : Signature) { + SmallVector ExpandedArg; + std::vector TypeList; + std::vector VectorList; + TypeFlags Flags; + + getTypeLists(Arg, Flags, TypeList, VectorList); + + // Insert the Cartesian product of the types and vector sizes. + for (const auto &Vector : VectorList) { + for (const auto &Type : TypeList) { + ExpandedArg.push_back(getTypeString(Type, Flags, Vector)); + } + } + NumSignatures = std::max(NumSignatures, ExpandedArg.size()); + ExpandedGenTypes.push_back(ExpandedArg); + } + + // Now the total number of signatures is known. Populate the return list with + // all signatures. + for (unsigned I = 0; I < NumSignatures; I++) { + SmallVector Args; + + // Process a single signature. + for (unsigned ArgNum = 0; ArgNum < Signature.size(); ArgNum++) { + // For differently-sized GenTypes in a parameter list, the smaller + // GenTypes just repeat, so index modulo the number of expanded types. + size_t TypeIndex = I % ExpandedGenTypes[ArgNum].size(); + Args.push_back(ExpandedGenTypes[ArgNum][TypeIndex]); + } + Types.push_back(Args); + } +} + +void OpenCLBuiltinTestEmitter::emit() { + emitSourceFileHeader("OpenCL Builtin exhaustive testing", OS); + + // Enable some extensions for testing. + OS << R"( +#pragma OPENCL EXTENSION cl_khr_fp16 : enable +#pragma OPENCL EXTENSION cl_khr_fp64 : enable +#pragma OPENCL EXTENSION cl_khr_int64_base_atomics : enable +#pragma OPENCL EXTENSION cl_khr_int64_extended_atomics : enable +#pragma OPENCL EXTENSION cl_khr_gl_msaa_sharing : enable +#pragma OPENCL EXTENSION cl_khr_mipmap_image_writes : enable +#pragma OPENCL EXTENSION cl_khr_3d_image_writes : enable + +)"; + + // Ensure each test has a unique name by numbering them. + unsigned TestID = 0; + + // Iterate over all builtins. + std::vector Builtins = Records.getAllDerivedDefinitions("Builtin"); + for (const auto *B : Builtins) { + StringRef Name = B->getValueAsString("Name"); + + SmallVector, 4> FTypes; + expandTypesInSignature(B->getValueAsListOfDefs("Signature"), FTypes); + + OS << "// Test " << Name << "\n"; + std::string OptionalEndif; + StringRef Extensions = + B->getValueAsDef("Extension")->getValueAsString("ExtName"); + if (!Extensions.empty()) { + OS << "#if"; + OptionalEndif = "#endif // Extension\n"; + + SmallVector ExtVec; + Extensions.split(ExtVec, " "); + bool isFirst = true; + for (StringRef Ext : ExtVec) { + if (!isFirst) { + OS << " &&"; + } + OS << " defined(" << Ext << ")"; + isFirst = false; + } + OS << "\n"; + } + auto PrintOpenCLVersion = [this](int Version) { + OS << "CL_VERSION_" << (Version / 100) << "_" << ((Version % 100) / 10); + }; + int MinVersion = B->getValueAsDef("MinVersion")->getValueAsInt("ID"); + if (MinVersion != 100) { + // OpenCL 1.0 is the default minimum version. + OS << "#if __OPENCL_C_VERSION__ >= "; + PrintOpenCLVersion(MinVersion); + OS << "\n"; + OptionalEndif = "#endif // MinVersion\n" + OptionalEndif; + } + int MaxVersion = B->getValueAsDef("MaxVersion")->getValueAsInt("ID"); + if (MaxVersion) { + OS << "#if __OPENCL_C_VERSION__ < "; + PrintOpenCLVersion(MaxVersion); + OS << "\n"; + OptionalEndif = "#endif // MaxVersion\n" + OptionalEndif; + } + for (const auto &Signature : FTypes) { + // Emit function declaration. + OS << Signature[0] << " test" << TestID++ << "_" << Name << "("; + if (Signature.size() > 1) { + for (unsigned I = 1; I < Signature.size(); I++) { + if (I != 1) + OS << ", "; + OS << Signature[I] << " arg" << I; + } + } + OS << ") {\n"; + + // Emit function body. + OS << " "; + if (Signature[0] != "void") { + OS << "return "; + } + OS << Name << "("; + for (unsigned I = 1; I < Signature.size(); I++) { + if (I != 1) + OS << ", "; + OS << "arg" << I; + } + OS << ");\n"; + + // End of function body. + OS << "}\n"; + } + OS << OptionalEndif << "\n"; + } +} + void clang::EmitClangOpenCLBuiltins(RecordKeeper &Records, raw_ostream &OS) { BuiltinNameEmitter NameChecker(Records, OS); NameChecker.Emit(); } + +void clang::EmitClangOpenCLBuiltinTests(RecordKeeper &Records, + raw_ostream &OS) { + OpenCLBuiltinTestEmitter TestFileGenerator(Records, OS); + TestFileGenerator.emit(); +} Index: clang/utils/TableGen/TableGen.cpp =================================================================== --- clang/utils/TableGen/TableGen.cpp +++ clang/utils/TableGen/TableGen.cpp @@ -63,6 +63,7 @@ GenClangCommentCommandInfo, GenClangCommentCommandList, GenClangOpenCLBuiltins, + GenClangOpenCLBuiltinTests, GenArmNeon, GenArmFP16, GenArmBF16, @@ -194,6 +195,8 @@ "documentation comments"), clEnumValN(GenClangOpenCLBuiltins, "gen-clang-opencl-builtins", "Generate OpenCL builtin declaration handlers"), + clEnumValN(GenClangOpenCLBuiltinTests, "gen-clang-opencl-builtin-tests", + "Generate OpenCL builtin declaration tests"), clEnumValN(GenArmNeon, "gen-arm-neon", "Generate arm_neon.h for clang"), clEnumValN(GenArmFP16, "gen-arm-fp16", "Generate arm_fp16.h for clang"), clEnumValN(GenArmBF16, "gen-arm-bf16", "Generate arm_bf16.h for clang"), @@ -371,6 +374,9 @@ case GenClangOpenCLBuiltins: EmitClangOpenCLBuiltins(Records, OS); break; + case GenClangOpenCLBuiltinTests: + EmitClangOpenCLBuiltinTests(Records, OS); + break; case GenClangSyntaxNodeList: EmitClangSyntaxNodeList(Records, OS); break; Index: clang/utils/TableGen/TableGenBackends.h =================================================================== --- clang/utils/TableGen/TableGenBackends.h +++ clang/utils/TableGen/TableGenBackends.h @@ -122,6 +122,8 @@ void EmitClangOpenCLBuiltins(llvm::RecordKeeper &Records, llvm::raw_ostream &OS); +void EmitClangOpenCLBuiltinTests(llvm::RecordKeeper &Records, + llvm::raw_ostream &OS); void EmitClangDataCollectors(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);