// Copyright(c) 2015-2016, NVIDIA CORPORATION. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include #include #include #include #include #include #include #include #include #include #include #include #include #include const std::string exceptionHeader( "#if defined(_MSC_VER) && (_MSC_VER == 1800)\n" "# define noexcept _NOEXCEPT\n" "#endif\n" "\n" " class ErrorCategoryImpl : public std::error_category\n" " {\n" " public:\n" " virtual const char* name() const noexcept override { return \"vk::Result\"; }\n" " virtual std::string message(int ev) const override { return to_string(static_cast(ev)); }\n" " };\n" "\n" "#if defined(_MSC_VER) && (_MSC_VER == 1800)\n" "# undef noexcept\n" "#endif\n" "\n" " VULKAN_HPP_INLINE const std::error_category& errorCategory()\n" " {\n" " static ErrorCategoryImpl instance;\n" " return instance;\n" " }\n" "\n" " VULKAN_HPP_INLINE std::error_code make_error_code(Result e)\n" " {\n" " return std::error_code(static_cast(e), errorCategory());\n" " }\n" "\n" " VULKAN_HPP_INLINE std::error_condition make_error_condition(Result e)\n" " {\n" " return std::error_condition(static_cast(e), errorCategory());\n" " }\n" "\n" ); const std::string flagsHeader( " template struct FlagTraits\n" " {\n" " enum { allFlags = 0 };\n" " };\n" "\n" " template \n" " class Flags\n" " {\n" " public:\n" " Flags()\n" " : m_mask(0)\n" " {\n" " }\n" "\n" " Flags(BitType bit)\n" " : m_mask(static_cast(bit))\n" " {\n" " }\n" "\n" " Flags(Flags const& rhs)\n" " : m_mask(rhs.m_mask)\n" " {\n" " }\n" "\n" " Flags & operator=(Flags const& rhs)\n" " {\n" " m_mask = rhs.m_mask;\n" " return *this;\n" " }\n" "\n" " Flags & operator|=(Flags const& rhs)\n" " {\n" " m_mask |= rhs.m_mask;\n" " return *this;\n" " }\n" "\n" " Flags & operator&=(Flags const& rhs)\n" " {\n" " m_mask &= rhs.m_mask;\n" " return *this;\n" " }\n" "\n" " Flags & operator^=(Flags const& rhs)\n" " {\n" " m_mask ^= rhs.m_mask;\n" " return *this;\n" " }\n" "\n" " Flags operator|(Flags const& rhs) const\n" " {\n" " Flags result(*this);\n" " result |= rhs;\n" " return result;\n" " }\n" "\n" " Flags operator&(Flags const& rhs) const\n" " {\n" " Flags result(*this);\n" " result &= rhs;\n" " return result;\n" " }\n" "\n" " Flags operator^(Flags const& rhs) const\n" " {\n" " Flags result(*this);\n" " result ^= rhs;\n" " return result;\n" " }\n" "\n" " bool operator!() const\n" " {\n" " return !m_mask;\n" " }\n" "\n" " Flags operator~() const\n" " {\n" " Flags result(*this);\n" " result.m_mask ^= FlagTraits::allFlags;\n" " return result;\n" " }\n" "\n" " bool operator==(Flags const& rhs) const\n" " {\n" " return m_mask == rhs.m_mask;\n" " }\n" "\n" " bool operator!=(Flags const& rhs) const\n" " {\n" " return m_mask != rhs.m_mask;\n" " }\n" "\n" " explicit operator bool() const\n" " {\n" " return !!m_mask;\n" " }\n" "\n" " explicit operator MaskType() const\n" " {\n" " return m_mask;\n" " }\n" "\n" " private:\n" " MaskType m_mask;\n" " };\n" " \n" " template \n" " Flags operator|(BitType bit, Flags const& flags)\n" " {\n" " return flags | bit;\n" " }\n" " \n" " template \n" " Flags operator&(BitType bit, Flags const& flags)\n" " {\n" " return flags & bit;\n" " }\n" " \n" " template \n" " Flags operator^(BitType bit, Flags const& flags)\n" " {\n" " return flags ^ bit;\n" " }\n" "\n" ); std::string const optionalClassHeader = R"( template class Optional { public: Optional(RefType & reference) { m_ptr = &reference; } Optional(RefType * ptr) { m_ptr = ptr; } Optional(std::nullptr_t) { m_ptr = nullptr; } operator RefType*() const { return m_ptr; } RefType const* operator->() const { return m_ptr; } explicit operator bool() const { return !!m_ptr; } private: RefType *m_ptr; }; )"; std::string const arrayProxyHeader = ( "#ifndef VULKAN_HPP_DISABLE_ENHANCED_MODE\n" " template \n" " class ArrayProxy\n" " {\n" " public:\n" " ArrayProxy(std::nullptr_t)\n" " : m_count(0)\n" " , m_ptr(nullptr)\n" " {}\n" "\n" " ArrayProxy(T & ptr)\n" " : m_count(1)\n" " , m_ptr(&ptr)\n" " {}\n" "\n" " ArrayProxy(uint32_t count, T * ptr)\n" " : m_count(count)\n" " , m_ptr(ptr)\n" " {}\n" "\n" " template \n" " ArrayProxy(std::array::type, N> & data)\n" " : m_count(N)\n" " , m_ptr(data.data())\n" " {}\n" "\n" " template \n" " ArrayProxy(std::array::type, N> const& data)\n" " : m_count(N)\n" " , m_ptr(data.data())\n" " {}\n" "\n" " template ::type>>\n" " ArrayProxy(std::vector::type, Allocator> & data)\n" " : m_count(static_cast(data.size()))\n" " , m_ptr(data.data())\n" " {}\n" "\n" " template ::type>>\n" " ArrayProxy(std::vector::type, Allocator> const& data)\n" " : m_count(static_cast(data.size()))\n" " , m_ptr(data.data())\n" " {}\n" "\n" " ArrayProxy(std::initializer_list const& data)\n" " : m_count(static_cast(data.end() - data.begin()))\n" " , m_ptr(data.begin())\n" " {}\n" "\n" " const T * begin() const\n" " {\n" " return m_ptr;\n" " }\n" "\n" " const T * end() const\n" " {\n" " return m_ptr + m_count;\n" " }\n" "\n" " const T & front() const\n" " {\n" " assert(m_count && m_ptr);\n" " return *m_ptr;\n" " }\n" "\n" " const T & back() const\n" " {\n" " assert(m_count && m_ptr);\n" " return *(m_ptr + m_count - 1);\n" " }\n" "\n" " bool empty() const\n" " {\n" " return (m_count == 0);\n" " }\n" "\n" " uint32_t size() const\n" " {\n" " return m_count;\n" " }\n" "\n" " T * data() const\n" " {\n" " return m_ptr;\n" " }\n" "\n" " private:\n" " uint32_t m_count;\n" " T * m_ptr;\n" " };\n" "#endif\n" "\n" ); std::string const versionCheckHeader = { R"( #if !defined(VULKAN_HPP_HAS_UNRESTRICTED_UNIONS) # if defined(__clang__) # if __has_feature(cxx_unrestricted_unions) # define VULKAN_HPP_HAS_UNRESTRICTED_UNIONS # endif # elif defined(__GNUC__) # define GCC_VERSION (__GNUC__ * 10000 + __GNUC_MINOR__ * 100 + __GNUC_PATCHLEVEL__) # if 40600 <= GCC_VERSION # define VULKAN_HPP_HAS_UNRESTRICTED_UNIONS # endif # elif defined(_MSC_VER) # if 1900 <= _MSC_VER # define VULKAN_HPP_HAS_UNRESTRICTED_UNIONS # endif # endif #endif )" }; std::string const inlineHeader = {R"( #if !defined(VULKAN_HPP_INLINE) # if defined(__clang___) # if __has_attribute(always_inline) # define VULKAN_HPP_INLINE __attribute__((always_inline)) __inline__ # else # define VULKAN_HPP_INLINE inline # endif # elif defined(__GNUC__) # define VULKAN_HPP_INLINE __attribute__((always_inline)) __inline__ # elif defined(_MSC_VER) # define VULKAN_HPP_INLINE __forceinline # else # define VULKAN_HPP_INLINE inline # endif #endif )" }; std::string const explicitHeader = { R"( #if defined(VULKAN_HPP_TYPESAFE_CONVERSION) # define VULKAN_HPP_TYPESAFE_EXPLICIT #else # define VULKAN_HPP_TYPESAFE_EXPLICIT explicit #endif )" }; std::string const resultValueHeader = ( " template \n" " struct ResultValue\n" " {\n" " ResultValue( Result r, T & v )\n" " : result( r )\n" " , value( v )\n" " {}\n" "\n" " Result result;\n" " T value;\n" "\n" " operator std::tuple() { return std::tuple(result, value); }\n" " };\n" "\n" " template \n" " struct ResultValueType\n" " {\n" "#ifdef VULKAN_HPP_NO_EXCEPTIONS\n" " typedef ResultValue type;\n" "#else\n" " typedef T type;\n" "#endif\n" " };\n" "\n" " template <>" " struct ResultValueType\n" " {\n" "#ifdef VULKAN_HPP_NO_EXCEPTIONS\n" " typedef Result type;\n" "#else\n" " typedef void type;\n" "#endif\n" " };\n" "\n" ); std::string const createResultValueHeader = ( " VULKAN_HPP_INLINE ResultValueType::type createResultValue( Result result, char const * message )\n" " {\n" "#ifdef VULKAN_HPP_NO_EXCEPTIONS\n" " assert( result == Result::eSuccess );\n" " return result;\n" "#else\n" " if ( result != Result::eSuccess )\n" " {\n" " throw std::system_error( result, message );\n" " }\n" "#endif\n" " }\n" "\n" " template \n" " VULKAN_HPP_INLINE typename ResultValueType::type createResultValue( Result result, T & data, char const * message )\n" " {\n" "#ifdef VULKAN_HPP_NO_EXCEPTIONS\n" " assert( result == Result::eSuccess );\n" " return ResultValue( result, data );\n" "#else\n" " if ( result != Result::eSuccess )\n" " {\n" " throw std::system_error( result, message );\n" " }\n" " return data;\n" "#endif\n" " }\n" "\n" " VULKAN_HPP_INLINE Result createResultValue( Result result, char const * message, std::initializer_list successCodes )\n" " {\n" "#ifdef VULKAN_HPP_NO_EXCEPTIONS\n" " assert( std::find( successCodes.begin(), successCodes.end(), result ) != successCodes.end() );\n" "#else\n" " if ( std::find( successCodes.begin(), successCodes.end(), result ) == successCodes.end() )\n" " {\n" " throw std::system_error( result, message );\n" " }\n" "#endif\n" " return result;\n" " }\n" "\n" " template \n" " VULKAN_HPP_INLINE ResultValue createResultValue( Result result, T & data, char const * message, std::initializer_list successCodes )\n" " {\n" "#ifdef VULKAN_HPP_NO_EXCEPTIONS\n" " assert( std::find( successCodes.begin(), successCodes.end(), result ) != successCodes.end() );\n" "#else\n" " if ( std::find( successCodes.begin(), successCodes.end(), result ) == successCodes.end() )\n" " {\n" " throw std::system_error( result, message );\n" " }\n" "#endif\n" " return ResultValue( result, data );\n" " }\n" "\n" ); std::string const uniqueHandleHeader = { R"( #if defined(VULKAN_HPP_NO_EXCEPTIONS) && !defined(VULKAN_HPP_NO_SMART_HANDLE) # define VULKAN_HPP_NO_SMART_HANDLE #endif #ifndef VULKAN_HPP_NO_SMART_HANDLE template class UniqueHandle { public: explicit UniqueHandle( Type const& value = Type(), Deleter const& deleter = Deleter() ) : m_value( value ) , m_deleter( deleter ) {} UniqueHandle( UniqueHandle const& ) = delete; UniqueHandle( UniqueHandle && other ) : m_value( other.release() ) , m_deleter( std::move( other.m_deleter ) ) {} ~UniqueHandle() { destroy(); } UniqueHandle & operator=( UniqueHandle const& ) = delete; UniqueHandle & operator=( UniqueHandle && other ) { reset( other.release() ); m_deleter = std::move( other.m_deleter ); return *this; } explicit operator bool() const { return m_value.operator bool(); } Type const* operator->() const { return &m_value; } Type const& operator*() const { return m_value; } Type get() const { return m_value; } Deleter & getDeleter() { return m_deleter; } Deleter const& getDeleter() const { return m_deleter; } void reset( Type const& value = Type() ) { if ( m_value != value ) { destroy(); m_value = value; } } Type release() { Type value = m_value; m_value = nullptr; return value; } void swap( UniqueHandle & rhs ) { std::swap(m_value, rhs.m_value); std::swap(m_deleter, rhs.m_deleter); } private: void destroy() { if ( m_value ) { m_deleter( m_value ); } } private: Type m_value; Deleter m_deleter; }; template VULKAN_HPP_INLINE void swap( UniqueHandle & lhs, UniqueHandle & rhs ) { lhs.swap( rhs ); } #endif )" }; struct ParamData { std::string type; std::string name; std::string arraySize; std::string pureType; std::string len; bool optional; }; struct CommandData { CommandData(std::string const& t, std::string const& fn) : returnType(t) , fullName(fn) , returnParam(~0) , templateParam(~0) , twoStep(false) {} std::string className; std::string enhancedReturnType; std::string fullName; std::vector params; std::string protect; std::string reducedName; size_t returnParam; std::string returnType; std::set skippedParams; std::vector successCodes; size_t templateParam; bool twoStep; std::map vectorParams; }; struct DependencyData { enum class Category { COMMAND, ENUM, FLAGS, FUNC_POINTER, HANDLE, REQUIRED, SCALAR, STRUCT, UNION }; DependencyData(Category c, std::string const& n) : category(c) , name(n) {} Category category; std::string name; std::set dependencies; std::set forwardDependencies; }; struct NameValue { std::string name; std::string value; }; struct EnumData { EnumData(std::string const& n, bool b = false) : name(n) , bitmask(b) {} void addEnumMember(std::string const & name, std::string const& tag, bool appendTag); std::string name; std::string prefix; std::string postfix; std::vector members; std::string protect; bool bitmask; }; struct FlagData { std::string protect; }; struct HandleData { std::vector commands; std::string protect; }; struct ScalarData { std::string protect; }; struct MemberData { std::string type; std::string name; std::string arraySize; std::string pureType; }; struct StructData { StructData() : returnedOnly(false) {} bool returnedOnly; bool isUnion; std::vector members; std::string protect; }; struct DeleterData { std::string pool; std::string call; }; struct VkData { std::map commands; std::list dependencies; std::map> deleterTypes; // map from parent type to set of child types std::map deleterData; // map from child types to corresponding deleter data std::map enums; std::map flags; std::map handles; std::map scalars; std::map structs; std::set tags; std::string typesafeCheck; std::string version; std::set vkTypes; std::string vulkanLicenseHeader; }; void createDefaults( VkData const& vkData, std::map & defaultValues ); void determineEnhancedReturnType(CommandData & commandData); void determineReducedName(CommandData & commandData); void determineReturnParam(CommandData & commandData); void determineSkippedParams(CommandData & commandData); void determineTemplateParam(CommandData & commandData); void determineVectorParams(CommandData & commandData); void enterProtect(std::ostream &os, std::string const& protect); std::string extractTag(std::string const& name); std::string findTag(std::string const& name, std::set const& tags); std::string generateEnumNameForFlags(std::string const& name); bool hasPointerParam(std::vector const& params); bool isVectorSizeParam(std::map const& vectorParams, size_t idx); void leaveProtect(std::ostream &os, std::string const& protect); void linkCommandToHandle(VkData & vkData, CommandData & commandData); bool noDependencies(std::set const& dependencies, std::map & listedTypes); bool readCommandParam( tinyxml2::XMLElement * element, std::set & dependencies, std::vector & params ); void readCommandParamArraySize(tinyxml2::XMLNode * node, ParamData & param); void readCommandParams(tinyxml2::XMLElement* element, std::set & dependencies, CommandData & commandData); tinyxml2::XMLNode* readCommandParamType(tinyxml2::XMLNode* node, ParamData& param); CommandData& readCommandProto(tinyxml2::XMLElement * element, VkData & vkData); void readCommands( tinyxml2::XMLElement * element, VkData & vkData ); void readCommandsCommand(tinyxml2::XMLElement * element, VkData & vkData); std::vector readCommandSuccessCodes(tinyxml2::XMLElement* element, std::set const& tags); void readComment(tinyxml2::XMLElement * element, std::string & header); void readEnums( tinyxml2::XMLElement * element, VkData & vkData ); void readEnumsEnum( tinyxml2::XMLElement * element, EnumData & enumData, std::string const& tag ); void readDisabledExtensionRequire(tinyxml2::XMLElement * element, VkData & vkData); void readExtensionRequire(tinyxml2::XMLElement * element, VkData & vkData, std::string const& protect, std::string const& tag); void readExtensions( tinyxml2::XMLElement * element, VkData & vkData ); void readExtensionsExtension(tinyxml2::XMLElement * element, VkData & vkData); void readTypeBasetype( tinyxml2::XMLElement * element, std::list & dependencies ); void readTypeBitmask( tinyxml2::XMLElement * element, VkData & vkData); void readTypeDefine( tinyxml2::XMLElement * element, VkData & vkData ); void readTypeFuncpointer( tinyxml2::XMLElement * element, std::list & dependencies ); void readTypeHandle(tinyxml2::XMLElement * element, VkData & vkData); void readTypeStruct( tinyxml2::XMLElement * element, VkData & vkData ); void readTypeStructMember( tinyxml2::XMLElement * element, std::vector & members, std::set & dependencies ); void readTypeUnion( tinyxml2::XMLElement * element, VkData & vkData ); void readTypeUnionMember( tinyxml2::XMLElement * element, std::vector & members, std::set & dependencies ); void readTags(tinyxml2::XMLElement * element, std::set & tags); void readTypes(tinyxml2::XMLElement * element, VkData & vkData); std::string reduceName(std::string const& name, bool singular = false); void registerDeleter(VkData & vkData, CommandData const& commandData); void sortDependencies( std::list & dependencies ); std::string strip(std::string const& value, std::string const& prefix, std::string const& tag = std::string()); std::string stripCommand(std::string const& value); std::string toCamelCase(std::string const& value); std::string toUpperCase(std::string const& name); std::string trimStart(std::string const& input); std::string trimEnd(std::string const& input); void writeCall(std::ostream & os, CommandData const& commandData, std::set const& vkTypes, bool firstCall, bool singular); void writeDeleterClasses(std::ofstream & ofs, std::pair> const& deleterTypes, std::map const& deleterData); void writeDeleterForwardDeclarations(std::ofstream &ofs, std::pair> const& deleterTypes, std::map const& deleterData); void writeEnumsToString(std::ofstream & ofs, EnumData const& enumData); void writeEnumsToString(std::ofstream & ofs, VkData const& vkData); void writeFlagsToString(std::ofstream & ofs, std::string const& flagsName, EnumData const &enumData); void writeExceptionCheck(std::ofstream & ofs, std::string const& indentation, std::string const& className, std::string const& functionName, std::vector const& successCodes); void writeFunctionBody(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool enhanced, bool singular, bool unique); void writeFunctionBodyEnhanced(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool singular); void writeFunctionBodyStandard(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData); void writeFunctionBodyUnique(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool singular); void writeFunctionDeclaration(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, bool enhanced, bool singular, bool unique); void writeFunctionDefinition(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool enhanced, bool singular, bool unique); void writeFunctionHeaderArguments(std::ostream & os, VkData const& vkData, CommandData const& commandData, bool enhanced, bool singular, bool withDefaults); void writeFunctionHeaderName(std::ostream & os, std::string const& name, bool singular, bool unique); void writeFunctionHeaderReturnType(std::ostream & os, std::string const& indentation, CommandData const& commandData, bool enhanced, bool singular, bool unique); void writeFunctionHeaderTemplate(std::ostream & os, std::string const& indentation, CommandData const& commandData, bool singular, bool withDefault); void writeStructConstructor( std::ofstream & ofs, std::string const& name, StructData const& structData, std::set const& vkTypes, std::map const& defaultValues ); void writeStructSetter( std::ofstream & ofs, std::string const& name, MemberData const& memberData, std::set const& vkTypes, std::map const& structs ); void writeTypeCommand(std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData); void writeTypeCommandDeclaration(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData); void writeTypeCommandDeclarationEnhanced(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData); void writeTypeCommandDeclarationStandard(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData); void writeTypeCommandDefinition(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData); void writeTypeCommandDefinitionEnhanced(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData); void writeTypeCommandDefinitionStandard(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData); void writeTypeCommandParam(std::ostream & os, ParamData const& param, std::set const& vkTypes); void writeTypeEnum(std::ofstream & ofs, EnumData const& enumData); void writeTypeFlags(std::ofstream & ofs, std::string const& flagsName, FlagData const& flagData, EnumData const& enumData); void writeTypeHandle(std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData, HandleData const& handle, std::list const& dependencies); void writeTypeScalar( std::ofstream & ofs, DependencyData const& dependencyData ); void writeTypeStruct( std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData, std::map const& defaultValues ); void writeTypeUnion( std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData, StructData const& unionData, std::map const& defaultValues ); void writeTypes(std::ofstream & ofs, VkData const& vkData, std::map const& defaultValues); void writeVersionCheck(std::ofstream & ofs, std::string const& version); void writeTypesafeCheck(std::ofstream & ofs, std::string const& typesafeCheck); void EnumData::addEnumMember(std::string const & name, std::string const& tag, bool appendTag) { assert(tag.empty() || (name.find(tag) != std::string::npos)); members.push_back(NameValue()); members.back().name = "e" + toCamelCase(strip(name, prefix, tag)); members.back().value = name; if (!postfix.empty()) { size_t pos = members.back().name.find(postfix); if (pos != std::string::npos) { members.back().name.erase(pos); } } if (appendTag && !tag.empty()) { members.back().name += tag; } } void createDefaults( VkData const& vkData, std::map & defaultValues ) { for ( auto it = vkData.dependencies.begin(), end = vkData.dependencies.end() ; it != end ; ++it ) { assert( defaultValues.find( it->name ) == defaultValues.end() ); switch( it->category ) { case DependencyData::Category::COMMAND : // commands should never be asked for defaults break; case DependencyData::Category::ENUM : { assert(vkData.enums.find(it->name) != vkData.enums.end()); EnumData const & enumData = vkData.enums.find(it->name)->second; if (!enumData.members.empty()) { defaultValues[it->name] = it->name + "::" + vkData.enums.find(it->name)->second.members.front().name; } else { defaultValues[it->name] = it->name + "()"; } } break; case DependencyData::Category::FLAGS : case DependencyData::Category::HANDLE: case DependencyData::Category::STRUCT: case DependencyData::Category::UNION : // just call the default constructor for flags, structs, and structs (which are mapped to classes) defaultValues[it->name] = it->name + "()"; break; case DependencyData::Category::FUNC_POINTER : // func_pointers default to nullptr defaultValues[it->name] = "nullptr"; break; case DependencyData::Category::REQUIRED : // all required default to "0" case DependencyData::Category::SCALAR : // all scalars default to "0" defaultValues[it->name] = "0"; break; default : assert( false ); break; } } } void determineReducedName(CommandData & commandData) { commandData.reducedName = commandData.fullName; std::string searchName = commandData.params[0].pureType; size_t pos = commandData.fullName.find(searchName); if ((pos == std::string::npos) && isupper(searchName[0])) { searchName[0] = tolower(searchName[0]); pos = commandData.fullName.find(searchName); } if (pos != std::string::npos) { commandData.reducedName.erase(pos, searchName.length()); } else if ((searchName == "commandBuffer") && (commandData.fullName.find("cmd") == 0)) { commandData.reducedName.erase(0, 3); pos = 0; } if ((pos == 0) && isupper(commandData.reducedName[0])) { commandData.reducedName[0] = tolower(commandData.reducedName[0]); } } void determineEnhancedReturnType(CommandData & commandData) { std::string returnType; if ((commandData.returnParam != ~0) && ((commandData.returnType == "void") || ((commandData.returnType == "Result") && ((commandData.successCodes.size() == 1) || ((commandData.successCodes.size() == 2) && (commandData.successCodes[1] == "eIncomplete") && commandData.twoStep))))) { if (commandData.vectorParams.find(commandData.returnParam) != commandData.vectorParams.end()) { if (commandData.params[commandData.returnParam].pureType == "void") { commandData.enhancedReturnType = "std::vector"; } else { commandData.enhancedReturnType = "std::vector<" + commandData.params[commandData.returnParam].pureType + ",Allocator>"; } } else { assert(commandData.params[commandData.returnParam].type.back() == '*'); assert(commandData.params[commandData.returnParam].type.find("const") == std::string::npos); commandData.enhancedReturnType = commandData.params[commandData.returnParam].type; commandData.enhancedReturnType.pop_back(); } } else if ((commandData.returnType == "Result") && (commandData.successCodes.size() == 1)) { // an original return of type "Result" with just one successCode is changed to void, errors throw an exception commandData.enhancedReturnType = "void"; } else { // the return type just stays the original return type commandData.enhancedReturnType = commandData.returnType; } } void determineReturnParam(CommandData & commandData) { if ((commandData.returnType == "Result") || (commandData.returnType == "void")) { for (size_t i = 0; i < commandData.params.size(); i++) { if ((commandData.params[i].type.find('*') != std::string::npos) && (commandData.params[i].type.find("const") == std::string::npos) && !isVectorSizeParam(commandData.vectorParams, i) && ((commandData.vectorParams.find(i) == commandData.vectorParams.end()) || commandData.twoStep || (commandData.successCodes.size() == 1))) { auto paramIt = std::find_if(commandData.params.begin() + i + 1, commandData.params.end(), [](ParamData const& pd) { return (pd.type.find('*') != std::string::npos) && (pd.type.find("const") == std::string::npos); }); commandData.returnParam = paramIt != commandData.params.end() ? ~0 : i; } } } } void determineSkippedParams(CommandData & commandData) { std::for_each(commandData.vectorParams.begin(), commandData.vectorParams.end(), [&commandData](std::pair const& vp) { if (vp.second != ~0) commandData.skippedParams.insert(vp.second); }); if (commandData.returnParam != ~0) { commandData.skippedParams.insert(commandData.returnParam); } } void determineTemplateParam(CommandData & commandData) { for (size_t i = 0; i < commandData.params.size(); i++) { if (((commandData.params[i].name == "pData") || (commandData.params[i].name == "pValues")) && (commandData.vectorParams.find(i) != commandData.vectorParams.end())) { commandData.templateParam = i; } } assert((commandData.templateParam == ~0) || (commandData.vectorParams.find(commandData.templateParam) != commandData.vectorParams.end())); } void determineVectorParams(CommandData & commandData) { for (auto it = commandData.params.begin(), begin = it, end = commandData.params.end(); it != end; ++it) { if (!it->len.empty()) { auto findLambda = [it](ParamData const& pd) { return pd.name == it->len; }; auto findIt = std::find_if(begin, it, findLambda); assert((std::count_if(begin, end, findLambda) == 0) || (findIt < it)); commandData.vectorParams.insert(std::make_pair(std::distance(begin, it), findIt < it ? std::distance(begin, findIt) : ~0)); assert((commandData.vectorParams[std::distance(begin, it)] != ~0) || (it->len == "null-terminated") || (it->len == "pAllocateInfo::descriptorSetCount") || (it->len == "pAllocateInfo::commandBufferCount")); } } } void enterProtect(std::ostream &os, std::string const& protect) { if (!protect.empty()) { os << "#ifdef " << protect << std::endl; } } std::string extractTag(std::string const& name) { // the name is supposed to look like: VK__ size_t start = name.find('_'); assert(start != std::string::npos); size_t end = name.find('_', start + 1); assert(end != std::string::npos); return name.substr(start + 1, end - start - 1); } std::string findTag(std::string const& name, std::set const& tags) { auto tagIt = std::find_if(tags.begin(), tags.end(), [&name](std::string const& t) { size_t pos = name.find(t); return (pos != std::string::npos) && (pos == name.length() - t.length()); }); return tagIt != tags.end() ? *tagIt : ""; } std::string generateEnumNameForFlags(std::string const& name) { std::string generatedName = name; size_t pos = generatedName.rfind("Flags"); assert(pos != std::string::npos); generatedName.replace(pos, 5, "FlagBits"); return generatedName; } bool hasPointerParam(std::vector const& params) { auto it = std::find_if(params.begin(), params.end(), [](ParamData const& pd) { return (pd.type.find('*') != std::string::npos); }); return it != params.end(); } bool isVectorSizeParam(std::map const& vectorParams, size_t idx) { auto it = std::find_if(vectorParams.begin(), vectorParams.end(), [idx](std::pair const& vp) { return vp.second == idx; }); return it != vectorParams.end(); } void leaveProtect(std::ostream &os, std::string const& protect) { if (!protect.empty()) { os << "#endif /*" << protect << "*/" << std::endl; } } void linkCommandToHandle(VkData & vkData, CommandData & commandData) { assert(!commandData.params.empty()); std::map::iterator hit = vkData.handles.find(commandData.params[0].pureType); if (hit == vkData.handles.end()) { hit = vkData.handles.find(""); } assert(hit != vkData.handles.end()); hit->second.commands.push_back(commandData.fullName); commandData.className = hit->first; DependencyData const& commandDD = vkData.dependencies.back(); std::list::iterator handleDD = std::find_if(vkData.dependencies.begin(), vkData.dependencies.end(), [hit](DependencyData const& dd) { return dd.name == hit->first; }); assert((handleDD != vkData.dependencies.end()) || hit->first.empty()); if (handleDD != vkData.dependencies.end()) { std::copy_if(commandDD.dependencies.begin(), commandDD.dependencies.end(), std::inserter(handleDD->dependencies, handleDD->dependencies.end()), [hit](std::string const& d) { return d != hit->first; }); } } bool noDependencies(std::set const& dependencies, std::set & listedTypes) { // false, if any of the dependencies is not in listedTypes; true if all dependencies are in listedTypes auto it = std::find_if(dependencies.begin(), dependencies.end(), [&listedTypes](std::string const& d) { return listedTypes.find(d) == listedTypes.end(); }); return it == dependencies.end(); } bool readCommandParam( tinyxml2::XMLElement * element, std::set & dependencies, std::vector & params ) { ParamData param; if (element->Attribute("len")) { param.len = element->Attribute("len"); } tinyxml2::XMLNode * child = readCommandParamType(element->FirstChild(), param); dependencies.insert(param.pureType); assert( child->ToElement() && ( strcmp( child->Value(), "name" ) == 0 ) ); param.name = child->ToElement()->GetText(); readCommandParamArraySize(child, param); param.optional = element->Attribute("optional") && (strcmp(element->Attribute("optional"), "true") == 0); params.push_back(param); return element->Attribute("optional") && (strcmp(element->Attribute("optional"), "false,true") == 0); } void readCommandParamArraySize(tinyxml2::XMLNode * node, ParamData & param) { if (param.name.back() == ']') { assert(!node->NextSibling()); size_t pos = param.name.find('['); assert(pos != std::string::npos); param.arraySize = param.name.substr(pos + 1, param.name.length() - 2 - pos); param.name.erase(pos); } else { tinyxml2::XMLNode * child = node->NextSibling(); if (child) { if (child->ToText()) { std::string value = child->Value(); if (value == "[") { child = child->NextSibling(); assert(child && child->ToElement() && (strcmp(child->Value(), "enum") == 0)); param.arraySize = child->ToElement()->GetText(); child = child->NextSibling(); assert(child && child->ToText() && (strcmp(child->Value(), "]") == 0) && !child->NextSibling()); } else { assert((value.front() == '[') && (value.back() == ']')); param.arraySize = value.substr(1, value.length() - 2); assert(!child->NextSibling()); } } } } } void readCommandParams(tinyxml2::XMLElement* element, std::set & dependencies, CommandData & commandData) { assert(element); while (element = element->NextSiblingElement()) { std::string value = element->Value(); if (value == "param") { commandData.twoStep |= readCommandParam(element, dependencies, commandData.params); } else { assert((value == "implicitexternsyncparams") || (value == "validity")); } } } tinyxml2::XMLNode* readCommandParamType(tinyxml2::XMLNode* node, ParamData& param) { assert(node); if (node->ToText()) { std::string value = trimEnd(node->Value()); assert((value == "const") || (value == "struct")); param.type = value + " "; node = node->NextSibling(); assert(node); } assert(node->ToElement() && (strcmp(node->Value(), "type") == 0) && node->ToElement()->GetText()); std::string type = strip(node->ToElement()->GetText(), "Vk"); param.type += type; param.pureType = type; node = node->NextSibling(); assert(node); if (node->ToText()) { std::string value = trimEnd(node->Value()); assert((value == "*") || (value == "**") || (value == "* const*")); param.type += value; node = node->NextSibling(); } return node; } CommandData& readCommandProto(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * typeElement = element->FirstChildElement(); assert( typeElement && ( strcmp( typeElement->Value(), "type" ) == 0 ) ); tinyxml2::XMLElement * nameElement = typeElement->NextSiblingElement(); assert( nameElement && ( strcmp( nameElement->Value(), "name" ) == 0 ) ); assert( !nameElement->NextSiblingElement() ); std::string type = strip( typeElement->GetText(), "Vk" ); std::string name = stripCommand( nameElement->GetText() ); vkData.dependencies.push_back( DependencyData( DependencyData::Category::COMMAND, name ) ); assert( vkData.commands.find( name ) == vkData.commands.end() ); std::map::iterator it = vkData.commands.insert( std::make_pair( name, CommandData(type, name) ) ).first; return it->second; } void readCommands(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * child = element->FirstChildElement(); assert( child ); do { assert( strcmp( child->Value(), "command" ) == 0 ); readCommandsCommand( child, vkData ); } while ( child = child->NextSiblingElement() ); } void readCommandsCommand(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * child = element->FirstChildElement(); assert( child && ( strcmp( child->Value(), "proto" ) == 0 ) ); CommandData& commandData = readCommandProto(child, vkData); commandData.successCodes = readCommandSuccessCodes(element, vkData.tags); readCommandParams(child, vkData.dependencies.back().dependencies, commandData); determineReducedName(commandData); linkCommandToHandle(vkData, commandData); registerDeleter(vkData, commandData); determineVectorParams(commandData); determineReturnParam(commandData); determineTemplateParam(commandData); determineEnhancedReturnType(commandData); determineSkippedParams(commandData); } std::vector readCommandSuccessCodes(tinyxml2::XMLElement* element, std::set const& tags) { std::vector results; if (element->Attribute("successcodes")) { std::string successCodes = element->Attribute("successcodes"); size_t start = 0, end; do { end = successCodes.find(',', start); std::string code = successCodes.substr(start, end - start); std::string tag = findTag(code, tags); results.push_back(std::string("e") + toCamelCase(strip(code, "VK_", tag)) + tag); start = end + 1; } while (end != std::string::npos); } return results; } void readComment(tinyxml2::XMLElement * element, std::string & header) { assert(element->GetText()); assert(header.empty()); header = element->GetText(); assert(header.find("\nCopyright") == 0); size_t pos = header.find("\n\n-----"); assert(pos != std::string::npos); header.erase(pos); for (size_t pos = header.find('\n'); pos != std::string::npos; pos = header.find('\n', pos + 1)) { header.replace(pos, 1, "\n// "); } header += "\n\n// This header is generated from the Khronos Vulkan XML API Registry."; } void readEnums( tinyxml2::XMLElement * element, VkData & vkData ) { if (!element->Attribute("name")) { throw std::runtime_error(std::string("spec error: enums element is missing the name attribute")); } std::string name = strip(element->Attribute("name"), "Vk"); if ( name != "API Constants" ) { vkData.dependencies.push_back( DependencyData( DependencyData::Category::ENUM, name ) ); std::map::iterator it = vkData.enums.insert( std::make_pair( name, EnumData(name) ) ).first; std::string tag; if (name == "Result") { // special handling for VKResult, as its enums just have VK_ in common it->second.prefix = "VK_"; } else { if (!element->Attribute("type")) { throw std::runtime_error(std::string("spec error: enums name=\"" + name + "\" is missing the type attribute")); } std::string type = element->Attribute("type"); if (type != "bitmask" && type != "enum") { throw std::runtime_error(std::string("spec error: enums name=\"" + name + "\" has unknown type " + type)); } it->second.bitmask = (type == "bitmask"); std::string prefix, postfix; if (it->second.bitmask) { size_t pos = name.find("FlagBits"); assert(pos != std::string::npos); it->second.prefix = "VK" + toUpperCase(name.substr(0, pos)) + "_"; it->second.postfix = "Bit"; } else { it->second.prefix = "VK" + toUpperCase(name) + "_"; } // if the enum name contains a tag remove it from the prefix to generate correct enum value names. for (std::set::const_iterator tit = vkData.tags.begin(); tit != vkData.tags.end(); ++tit) { size_t pos = it->second.prefix.find(*tit); if ((pos != std::string::npos) && (pos == it->second.prefix.length() - tit->length() - 1)) { it->second.prefix.erase(pos); tag = *tit; break; } } } readEnumsEnum( element, it->second, tag ); assert( vkData.vkTypes.find( name ) == vkData.vkTypes.end() ); vkData.vkTypes.insert( name ); } } void readEnumsEnum( tinyxml2::XMLElement * element, EnumData & enumData, std::string const& tag ) { tinyxml2::XMLElement * child = element->FirstChildElement(); while (child) { if ( child->Attribute( "name" ) ) { enumData.addEnumMember(child->Attribute("name"), tag, false); } child = child->NextSiblingElement(); } } void readDisabledExtensionRequire(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * child = element->FirstChildElement(); do { std::string value = child->Value(); if ((value == "command") || (value == "type")) { assert(child->Attribute("name")); std::string name = (value == "command") ? stripCommand(child->Attribute("name")) : strip(child->Attribute("name"), "Vk"); std::list::const_iterator depIt = std::find_if(vkData.dependencies.begin(), vkData.dependencies.end(), [&name](DependencyData const& dd) { return(dd.name == name); }); assert(depIt != vkData.dependencies.end()); vkData.dependencies.erase(depIt); for (auto & dep : vkData.dependencies) { dep.dependencies.erase(name); } if (value == "command") { // first unlink the command from its class auto commandsIt = vkData.commands.find(name); assert(commandsIt != vkData.commands.end()); assert(!commandsIt->second.className.empty()); auto handlesIt = vkData.handles.find(commandsIt->second.className); assert(handlesIt != vkData.handles.end()); auto it = std::find(handlesIt->second.commands.begin(), handlesIt->second.commands.end(), name); assert(it != handlesIt->second.commands.end()); handlesIt->second.commands.erase(it); // then remove the command vkData.commands.erase(name); } else { assert((vkData.structs.find(name) != vkData.structs.end()) && (vkData.vkTypes.find(name) != vkData.vkTypes.end())); vkData.structs.erase(name); vkData.vkTypes.erase(name); } } else { // nothing to do for enums, no other values ever encountered assert(value == "enum"); } } while (child = child->NextSiblingElement()); } void readExtensionRequire(tinyxml2::XMLElement * element, VkData & vkData, std::string const& protect, std::string const& tag) { tinyxml2::XMLElement * child = element->FirstChildElement(); do { std::string value = child->Value(); if ( value == "command" ) { assert(child->Attribute("name")); std::string name = stripCommand(child->Attribute("name")); std::map::iterator cit = vkData.commands.find(name); assert(cit != vkData.commands.end()); cit->second.protect = protect; } else if (value == "type") { assert(child->Attribute("name")); std::string name = strip(child->Attribute("name"), "Vk"); std::map::iterator eit = vkData.enums.find(name); if (eit != vkData.enums.end()) { eit->second.protect = protect; } else { std::map::iterator fit = vkData.flags.find(name); if (fit != vkData.flags.end()) { fit->second.protect = protect; // if the enum of this flags is auto-generated, protect it as well std::string enumName = generateEnumNameForFlags(name); std::map::iterator eit = vkData.enums.find(enumName); assert(eit != vkData.enums.end()); if (eit->second.members.empty()) { eit->second.protect = protect; } } else { std::map::iterator hait = vkData.handles.find(name); if (hait != vkData.handles.end()) { hait->second.protect = protect; } else { std::map::iterator scit = vkData.scalars.find(name); if (scit != vkData.scalars.end()) { scit->second.protect = protect; } else { std::map::iterator stit = vkData.structs.find(name); assert(stit != vkData.structs.end() && stit->second.protect.empty()); stit->second.protect = protect; } } } } } else if ( value == "enum") { // TODO process enums which don't extend existing enums if (child->Attribute("extends")) { assert(child->Attribute("name")); assert(vkData.enums.find(strip(child->Attribute("extends"), "Vk")) != vkData.enums.end()); assert(!!child->Attribute("bitpos") + !!child->Attribute("offset") + !!child->Attribute("value") == 1); auto enumIt = vkData.enums.find(strip(child->Attribute("extends"), "Vk")); assert(enumIt != vkData.enums.end()); enumIt->second.addEnumMember(child->Attribute("name"), child->Attribute("value") ? "" : tag, true ); } } else { assert(value=="usage"); } } while ( child = child->NextSiblingElement() ); } void readExtensions(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * child = element->FirstChildElement(); assert( child ); do { assert( strcmp( child->Value(), "extension" ) == 0 ); readExtensionsExtension( child, vkData ); } while ( child = child->NextSiblingElement() ); } void readExtensionsExtension(tinyxml2::XMLElement * element, VkData & vkData) { assert( element->Attribute( "name" ) ); std::string tag = extractTag(element->Attribute("name")); //assert(vkData.tags.find(tag) != vkData.tags.end()); tinyxml2::XMLElement * child = element->FirstChildElement(); assert(child && (strcmp(child->Value(), "require") == 0) && !child->NextSiblingElement()); if (strcmp(element->Attribute("supported"), "disabled") == 0) { // kick out all the disabled stuff we've read before !! readDisabledExtensionRequire(child, vkData); } else { std::string protect; if (element->Attribute("protect")) { protect = element->Attribute("protect"); } readExtensionRequire(child, vkData, protect, tag); } } void readTypeBasetype( tinyxml2::XMLElement * element, std::list & dependencies ) { tinyxml2::XMLElement * typeElement = element->FirstChildElement(); assert( typeElement && ( strcmp( typeElement->Value(), "type" ) == 0 ) && typeElement->GetText() ); std::string type = typeElement->GetText(); assert( ( type == "uint32_t" ) || ( type == "uint64_t" ) ); tinyxml2::XMLElement * nameElement = typeElement->NextSiblingElement(); assert( nameElement && ( strcmp( nameElement->Value(), "name" ) == 0 ) && nameElement->GetText() ); std::string name = strip( nameElement->GetText(), "Vk" ); // skip "Flags", if ( name != "Flags" ) { dependencies.push_back( DependencyData( DependencyData::Category::SCALAR, name ) ); dependencies.back().dependencies.insert( type ); } else { assert( type == "uint32_t" ); } } void readTypeBitmask(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * typeElement = element->FirstChildElement(); assert( typeElement && ( strcmp( typeElement->Value(), "type" ) == 0 ) && typeElement->GetText() && ( strcmp( typeElement->GetText(), "VkFlags" ) == 0 ) ); std::string type = typeElement->GetText(); tinyxml2::XMLElement * nameElement = typeElement->NextSiblingElement(); assert( nameElement && ( strcmp( nameElement->Value(), "name" ) == 0 ) && nameElement->GetText() ); std::string name = strip( nameElement->GetText(), "Vk" ); assert( !nameElement->NextSiblingElement() ); std::string requires; if (element->Attribute("requires")) { requires = strip(element->Attribute("requires"), "Vk"); } else { // Generate FlagBits name requires = generateEnumNameForFlags(name); vkData.dependencies.push_back(DependencyData(DependencyData::Category::ENUM, requires)); vkData.enums.insert(std::make_pair(requires, EnumData(requires, true))); vkData.vkTypes.insert(requires); } vkData.dependencies.push_back( DependencyData( DependencyData::Category::FLAGS, name ) ); vkData.dependencies.back().dependencies.insert( requires ); vkData.flags.insert(std::make_pair(name, FlagData())); assert( vkData.vkTypes.find( name ) == vkData.vkTypes.end() ); vkData.vkTypes.insert( name ); } void readTypeDefine( tinyxml2::XMLElement * element, VkData & vkData ) { tinyxml2::XMLElement * child = element->FirstChildElement(); if (child && (strcmp(child->GetText(), "VK_HEADER_VERSION") == 0)) { vkData.version = element->LastChild()->ToText()->Value(); } else if (element->Attribute("name") && strcmp(element->Attribute("name"), "VK_DEFINE_NON_DISPATCHABLE_HANDLE") == 0) { std::string text = element->LastChild()->ToText()->Value(); size_t start = text.find("#if defined(__LP64__)"); size_t end = text.find_first_of("\r\n", start + 1); vkData.typesafeCheck = text.substr(start, end - start); } } void readTypeFuncpointer( tinyxml2::XMLElement * element, std::list & dependencies ) { tinyxml2::XMLElement * child = element->FirstChildElement(); assert( child && ( strcmp( child->Value(), "name" ) == 0 ) && child->GetText() ); dependencies.push_back( DependencyData( DependencyData::Category::FUNC_POINTER, child->GetText() ) ); } void readTypeHandle(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * typeElement = element->FirstChildElement(); assert( typeElement && ( strcmp( typeElement->Value(), "type" ) == 0 ) && typeElement->GetText() ); #if !defined(NDEBUG) std::string type = typeElement->GetText(); assert( type.find( "VK_DEFINE" ) == 0 ); #endif tinyxml2::XMLElement * nameElement = typeElement->NextSiblingElement(); assert( nameElement && ( strcmp( nameElement->Value(), "name" ) == 0 ) && nameElement->GetText() ); std::string name = strip( nameElement->GetText(), "Vk" ); vkData.dependencies.push_back( DependencyData( DependencyData::Category::HANDLE, name ) ); assert(vkData.vkTypes.find(name) == vkData.vkTypes.end()); vkData.vkTypes.insert(name); assert(vkData.handles.find(name) == vkData.handles.end()); vkData.handles[name]; // add this to the handles map } void readTypeStructMember( tinyxml2::XMLElement * element, std::vector & members, std::set & dependencies ) { members.push_back( MemberData() ); MemberData & member = members.back(); tinyxml2::XMLNode * child = element->FirstChild(); assert( child ); if ( child->ToText()) { std::string value = trimEnd(child->Value()); assert( (value == "const") || (value == "struct") ); member.type = value + " "; child = child->NextSibling(); assert( child ); } assert( child->ToElement() ); assert( ( strcmp( child->Value(), "type" ) == 0 ) && child->ToElement() && child->ToElement()->GetText() ); std::string type = strip( child->ToElement()->GetText(), "Vk" ); dependencies.insert( type ); member.type += type; member.pureType = type; child = child->NextSibling(); assert( child ); if ( child->ToText()) { std::string value = trimEnd(child->Value()); assert( ( value == "*" ) || ( value == "**" ) || ( value == "* const*" ) ); member.type += value; child = child->NextSibling(); } assert( ( child->ToElement() && strcmp( child->Value(), "name" ) == 0 )); member.name = child->ToElement()->GetText(); if ( member.name.back() == ']' ) { assert( !child->NextSibling() ); size_t pos = member.name.find( '[' ); assert( pos != std::string::npos ); member.arraySize = member.name.substr( pos + 1, member.name.length() - 2 - pos ); member.name.erase( pos ); } child = child->NextSibling(); if ( child ) { assert( member.arraySize.empty() ); if ( child->ToText() ) { std::string value = child->Value(); if ( value == "[" ) { child = child->NextSibling(); assert( child ); assert( child->ToElement() && ( strcmp( child->Value(), "enum" ) == 0 ) ); member.arraySize = child->ToElement()->GetText(); child = child->NextSibling(); assert( child ); assert( child->ToText() ); assert( strcmp( child->Value(), "]" ) == 0 ); assert( !child->NextSibling() ); } else { assert( ( value.front() == '[' ) && ( value.back() == ']' ) ); member.arraySize = value.substr( 1, value.length() - 2 ); assert( !child->NextSibling() ); } } } } void readTypeStruct( tinyxml2::XMLElement * element, VkData & vkData ) { assert( !element->Attribute( "returnedonly" ) || ( strcmp( element->Attribute( "returnedonly" ), "true" ) == 0 ) ); assert( element->Attribute( "name" ) ); std::string name = strip( element->Attribute( "name" ), "Vk" ); if ( name == "Rect3D" ) { return; } vkData.dependencies.push_back( DependencyData( DependencyData::Category::STRUCT, name ) ); assert( vkData.structs.find( name ) == vkData.structs.end() ); std::map::iterator it = vkData.structs.insert( std::make_pair( name, StructData() ) ).first; it->second.returnedOnly = !!element->Attribute( "returnedonly" ); it->second.isUnion = false; tinyxml2::XMLElement * child = element->FirstChildElement(); do { assert( child->Value() ); std::string value = child->Value(); if ( value == "member" ) { readTypeStructMember( child, it->second.members, vkData.dependencies.back().dependencies ); } else { assert( value == "validity" ); } } while ( child = child->NextSiblingElement() ); assert( vkData.vkTypes.find( name ) == vkData.vkTypes.end() ); vkData.vkTypes.insert( name ); } void readTypeUnionMember( tinyxml2::XMLElement * element, std::vector & members, std::set & dependencies ) { members.push_back( MemberData() ); MemberData & member = members.back(); tinyxml2::XMLNode * child = element->FirstChild(); assert( child ); if ( child->ToText() ) { assert( ( strcmp( child->Value(), "const" ) == 0 ) || ( strcmp( child->Value(), "struct" ) == 0 ) ); member.type = std::string( child->Value() ) + " "; child = child->NextSibling(); assert( child ); } assert( child->ToElement() ); assert( ( strcmp( child->Value(), "type" ) == 0 ) && child->ToElement() && child->ToElement()->GetText() ); std::string type = strip( child->ToElement()->GetText(), "Vk" ); dependencies.insert( type ); member.type += type; member.pureType = type; child = child->NextSibling(); assert( child ); if ( child->ToText() ) { std::string value = child->Value(); assert( ( value == "*" ) || ( value == "**" ) || ( value == "* const*" ) ); member.type += value; child = child->NextSibling(); } assert( child->ToElement() && ( strcmp( child->Value(), "name" ) == 0 ) ); member.name = child->ToElement()->GetText(); if ( member.name.back() == ']' ) { assert( !child->NextSibling() ); size_t pos = member.name.find( '[' ); assert( pos != std::string::npos ); member.arraySize = member.name.substr( pos + 1, member.name.length() - 2 - pos ); member.name.erase( pos ); } child = child->NextSibling(); if ( child ) { if ( child->ToText() ) { std::string value = child->Value(); if ( value == "[" ) { child = child->NextSibling(); assert( child ); assert( child->ToElement() && ( strcmp( child->Value(), "enum" ) == 0 ) ); member.arraySize = child->ToElement()->GetText(); child = child->NextSibling(); assert( child ); assert( child->ToText() ); assert( strcmp( child->Value(), "]" ) == 0 ); assert( !child->NextSibling() ); } else { assert( ( value.front() == '[' ) && ( value.back() == ']' ) ); member.arraySize = value.substr( 1, value.length() - 2 ); assert( !child->NextSibling() ); } } } } void readTypeUnion( tinyxml2::XMLElement * element, VkData & vkData ) { assert( element->Attribute( "name" ) ); std::string name = strip( element->Attribute( "name" ), "Vk" ); vkData.dependencies.push_back( DependencyData( DependencyData::Category::UNION, name ) ); assert( vkData.structs.find( name ) == vkData.structs.end() ); std::map::iterator it = vkData.structs.insert( std::make_pair( name, StructData() ) ).first; it->second.isUnion = true; tinyxml2::XMLElement * child = element->FirstChildElement(); do { assert( strcmp( child->Value(), "member" ) == 0 ); readTypeUnionMember( child, it->second.members, vkData.dependencies.back().dependencies ); } while ( child = child->NextSiblingElement() ); assert( vkData.vkTypes.find( name ) == vkData.vkTypes.end() ); vkData.vkTypes.insert( name ); } void readTags(tinyxml2::XMLElement * element, std::set & tags) { tags.insert("EXT"); tags.insert("KHR"); tinyxml2::XMLElement * child = element->FirstChildElement(); do { assert(child->Attribute("name")); tags.insert(child->Attribute("name")); } while (child = child->NextSiblingElement()); } void readTypes(tinyxml2::XMLElement * element, VkData & vkData) { tinyxml2::XMLElement * child = element->FirstChildElement(); do { assert( strcmp( child->Value(), "type" ) == 0 ); std::string type = child->Value(); assert( type == "type" ); if ( child->Attribute( "category" ) ) { std::string category = child->Attribute( "category" ); if ( category == "basetype" ) { readTypeBasetype( child, vkData.dependencies ); } else if ( category == "bitmask" ) { readTypeBitmask( child, vkData); } else if ( category == "define" ) { readTypeDefine( child, vkData ); } else if ( category == "funcpointer" ) { readTypeFuncpointer( child, vkData.dependencies ); } else if ( category == "handle" ) { readTypeHandle( child, vkData ); } else if ( category == "struct" ) { readTypeStruct( child, vkData ); } else if ( category == "union" ) { readTypeUnion( child, vkData ); } else { assert( ( category == "enum" ) || ( category == "include" ) ); } } else { assert( child->Attribute( "name" ) ); vkData.dependencies.push_back( DependencyData( DependencyData::Category::REQUIRED, child->Attribute( "name" ) ) ); } } while ( child = child->NextSiblingElement() ); } std::string reduceName(std::string const& name, bool singular) { std::string reducedName; if ((name[0] == 'p') && (1 < name.length()) && (isupper(name[1]) || name[1] == 'p')) { reducedName = strip(name, "p"); reducedName[0] = tolower(reducedName[0]); } else { reducedName = name; } if (singular) { size_t pos = reducedName.rfind('s'); assert(pos != std::string::npos); reducedName.erase(pos, 1); } return reducedName; } void registerDeleter(VkData & vkData, CommandData const& commandData) { if ((commandData.fullName.substr(0, 7) == "destroy") || (commandData.fullName.substr(0, 4) == "free")) { std::string key; size_t valueIndex; switch (commandData.params.size()) { case 2: case 3: assert(commandData.params.back().pureType == "AllocationCallbacks"); key = (commandData.params.size() == 2) ? "" : commandData.params[0].pureType; valueIndex = commandData.params.size() - 2; break; case 4: key = commandData.params[0].pureType; valueIndex = 3; assert(vkData.deleterData.find(commandData.params[valueIndex].pureType) == vkData.deleterData.end()); vkData.deleterData[commandData.params[valueIndex].pureType].pool = commandData.params[1].pureType; break; default: assert(false); } if (commandData.fullName == "destroyDevice") { key = "PhysicalDevice"; } assert(vkData.deleterTypes[key].find(commandData.params[valueIndex].pureType) == vkData.deleterTypes[key].end()); vkData.deleterTypes[key].insert(commandData.params[valueIndex].pureType); vkData.deleterData[commandData.params[valueIndex].pureType].call = commandData.reducedName; } } void sortDependencies( std::list & dependencies ) { std::set listedTypes = { "VkFlags" }; std::list sortedDependencies; while ( !dependencies.empty() ) { bool found = false; for ( std::list::iterator it = dependencies.begin() ; it != dependencies.end() ; ++it ) { if ( noDependencies( it->dependencies, listedTypes ) ) { sortedDependencies.push_back( *it ); listedTypes.insert( it->name ); dependencies.erase( it ); found = true; break; } } if (!found) { // resolve direct circular dependencies for (std::list::iterator it = dependencies.begin(); !found && it != dependencies.end(); ++it) { for (std::set::const_iterator dit = it->dependencies.begin(); dit != it->dependencies.end(); ++dit) { std::list::const_iterator depIt = std::find_if(dependencies.begin(), dependencies.end(), [&dit](DependencyData const& dd) { return(dd.name == *dit); }); if (depIt != dependencies.end()) { if (depIt->dependencies.find(it->name) != depIt->dependencies.end()) { // we only have just one case, for now! assert((it->category == DependencyData::Category::HANDLE) && (depIt->category == DependencyData::Category::STRUCT)); it->forwardDependencies.insert(*dit); it->dependencies.erase(*dit); found = true; break; } } #if !defined(NDEBUG) else { assert(std::find_if(sortedDependencies.begin(), sortedDependencies.end(), [&dit](DependencyData const& dd) { return(dd.name == *dit); }) != sortedDependencies.end()); } #endif } } } assert( found ); } dependencies.swap(sortedDependencies); } std::string strip(std::string const& value, std::string const& prefix, std::string const& postfix) { std::string strippedValue = value; if (strippedValue.find(prefix) == 0) { strippedValue.erase(0, prefix.length()); } if (!postfix.empty()) { size_t pos = strippedValue.rfind(postfix); assert(pos != std::string::npos); strippedValue.erase(pos); } return strippedValue; } std::string stripCommand(std::string const& value) { std::string stripped = strip(value, "vk"); assert(isupper(stripped[0])); stripped[0] = tolower(stripped[0]); return stripped; } std::string toCamelCase(std::string const& value) { assert(!value.empty() && (isupper(value[0]) || isdigit(value[0]))); std::string result; result.reserve(value.size()); result.push_back(value[0]); for (size_t i = 1; i < value.size(); i++) { if (value[i] != '_') { if ((value[i - 1] == '_') || isdigit(value[i-1])) { result.push_back(value[i]); } else { result.push_back(tolower(value[i])); } } } return result; } std::string toUpperCase(std::string const& name) { assert(isupper(name.front())); std::string convertedName; for (size_t i = 0; i(std::isspace)))); return result; } // trim from end std::string trimEnd(std::string const& input) { std::string result = input; result.erase(std::find_if(result.rbegin(), result.rend(), std::not1(std::ptr_fun(std::isspace))).base(), result.end()); return result; } void writeCall(std::ostream & os, CommandData const& commandData, std::set const& vkTypes, bool firstCall, bool singular) { std::map countIndices; for (std::map::const_iterator it = commandData.vectorParams.begin(); it != commandData.vectorParams.end(); ++it) { countIndices.insert(std::make_pair(it->second, it->first)); } if ((commandData.vectorParams.size() == 1) && ((commandData.params[commandData.vectorParams.begin()->first].len == "dataSize/4") || (commandData.params[commandData.vectorParams.begin()->first].len == "latexmath:[$dataSize \\over 4$]"))) { assert(commandData.params[3].name == "dataSize"); countIndices.insert(std::make_pair(3, commandData.vectorParams.begin()->first)); } assert(islower(commandData.fullName[0])); os << "vk" << static_cast(toupper(commandData.fullName[0])) << commandData.fullName.substr(1) << "( "; size_t i = 0; if (!commandData.className.empty()) { os << "m_" << commandData.params[0].name; i++; } for (; i < commandData.params.size(); i++) { if (0 < i) { os << ", "; } std::map::const_iterator it = countIndices.find(i); if (it != countIndices.end()) { if ((commandData.returnParam == it->second) && commandData.twoStep) { os << "&" << reduceName(commandData.params[it->first].name); } else { if (singular) { os << "1 "; } else { os << reduceName(commandData.params[it->second].name) << ".size() "; } if (commandData.templateParam == it->second) { os << "* sizeof( T ) "; } } } else { it = commandData.vectorParams.find(i); if (it != commandData.vectorParams.end()) { assert(commandData.params[it->first].type.back() == '*'); if ((commandData.returnParam == it->first) && commandData.twoStep && firstCall) { os << "nullptr"; } else { std::set::const_iterator vkit = vkTypes.find(commandData.params[it->first].pureType); if ((vkit != vkTypes.end()) || (it->first == commandData.templateParam)) { os << "reinterpret_cast<"; if (commandData.params[it->first].type.find("const") == 0) { os << "const "; } if (vkit != vkTypes.end()) { os << "Vk"; } os << commandData.params[it->first].pureType; if (commandData.params[it->first].type.rfind("* const") != std::string::npos) { os << "* const"; } os << "*>( " << (singular ? "&" : "") << reduceName(commandData.params[it->first].name, singular) << (singular ? "" : ".data()") << " )"; } else if (commandData.params[it->first].pureType == "char") { os << reduceName(commandData.params[it->first].name); if (commandData.params[it->first].optional) { os << " ? " << reduceName(commandData.params[it->first].name) << "->c_str() : nullptr"; } else { os << ".c_str()"; } } else { os << reduceName(commandData.params[it->first].name) << ".data()"; } } } else if (vkTypes.find(commandData.params[i].pureType) != vkTypes.end()) { if (commandData.params[i].type.back() == '*') { if (commandData.params[i].type.find("const") != std::string::npos) { os << "reinterpret_cast( "; if (commandData.params[i].optional) { os << "static_cast( "; } else { os << "&"; } os << reduceName(commandData.params[i].name) << (commandData.params[i].optional ? "))" : " )"); } else { assert(!commandData.params[i].optional); os << "reinterpret_cast( &" << reduceName(commandData.params[i].name) << " )"; } } else { os << "static_cast( " << commandData.params[i].name << " )"; } } else { if (commandData.params[i].type.back() == '*') { if (commandData.params[i].type.find("const") != std::string::npos) { if (commandData.params[i].pureType == "char") { os << reduceName(commandData.params[i].name); if (commandData.params[i].optional) { os << " ? " << reduceName(commandData.params[i].name) << "->c_str() : nullptr"; } else { os << ".c_str()"; } } else { assert((commandData.params[i].pureType == "void") && !commandData.params[i].optional); os << commandData.params[i].name; } } else { assert(commandData.params[i].type.find("char") == std::string::npos); os << "&" << reduceName(commandData.params[i].name); } } else { os << commandData.params[i].name; } } } } os << " )"; } void writeExceptionCheck(std::ofstream & ofs, std::string const& indentation, std::string const& className, std::string const& functionName, std::vector const& successCodes) { assert(!successCodes.empty()); ofs << indentation << " if ("; if (successCodes.size() == 1) { assert(successCodes.front() == "eSuccess"); ofs << " result != Result::eSuccess"; } else { for (size_t i = 0; i < successCodes.size() - 1; i++) { ofs << " ( result != Result::" << successCodes[i] << " ) &&"; } ofs << " ( result != Result::" << successCodes.back() << " )"; } ofs << " )" << std::endl; ofs << indentation << " {" << std::endl << indentation << " throw std::system_error( result, \"vk::"; if (!className.empty()) { ofs << className << "::"; } ofs << functionName << "\" );" << std::endl << indentation << " }" << std::endl; } void writeFunctionBody(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool enhanced, bool singular, bool unique) { os << indentation << "{" << std::endl; if (enhanced) { if (unique) { writeFunctionBodyUnique(os, indentation, vkData, commandData, dependencyData, singular); } else { writeFunctionBodyEnhanced(os, indentation, vkData, commandData, dependencyData, singular); } } else { writeFunctionBodyStandard(os, indentation, vkData, commandData, dependencyData); } os << indentation << "}" << std::endl; } void writeFunctionBodyEnhanced(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool singular) { // add a static_assert if a type is templated and its size needs to be some multiple of the original size if ((commandData.templateParam != ~0) && (commandData.params[commandData.templateParam].pureType != "void")) { os << indentation << " static_assert( sizeof( T ) % sizeof( " << commandData.params[commandData.templateParam].pureType << " ) == 0, \"wrong size of template type T\" );" << std::endl; } // add some error checks if multiple vectors need to have the same size if (1 < commandData.vectorParams.size()) { for (std::map::const_iterator it0 = commandData.vectorParams.begin(); it0 != commandData.vectorParams.end(); ++it0) { if (it0->first != commandData.returnParam) { for (std::map::const_iterator it1 = std::next(it0); it1 != commandData.vectorParams.end(); ++it1) { if ((it1->first != commandData.returnParam) && (it0->second == it1->second)) { os << "#ifdef VULKAN_HPP_NO_EXCEPTIONS" << std::endl << indentation << " assert( " << reduceName(commandData.params[it0->first].name) << ".size() == " << reduceName(commandData.params[it1->first].name) << ".size() );" << std::endl << "#else" << std::endl << indentation << " if ( " << reduceName(commandData.params[it0->first].name) << ".size() != " << reduceName(commandData.params[it1->first].name) << ".size() )" << std::endl << indentation << " {" << std::endl << indentation << " throw std::logic_error( \"vk::" << commandData.className << "::" << commandData.reducedName << ": " << reduceName(commandData.params[it0->first].name) << ".size() != " << reduceName(commandData.params[it1->first].name) << ".size()\" );" << std::endl << indentation << " }" << std::endl << "#endif // VULKAN_HPP_NO_EXCEPTIONS" << std::endl; } } } } } // write the local variable to hold a returned value if (commandData.returnParam != ~0) { if (commandData.returnType != commandData.enhancedReturnType) { os << indentation << " " << (singular ? commandData.params[commandData.returnParam].pureType : commandData.enhancedReturnType) << " " << reduceName(commandData.params[commandData.returnParam].name, singular); if (!singular) { std::map::const_iterator it = commandData.vectorParams.find(commandData.returnParam); if (it != commandData.vectorParams.end() && !commandData.twoStep) { std::string size; if ((it->second == ~0) && !commandData.params[commandData.returnParam].len.empty()) { size = reduceName(commandData.params[commandData.returnParam].len); size_t pos = size.find("->"); if (pos == std::string::npos) { pos = size.find("::"); } assert(pos != std::string::npos); size.replace(pos, 2, "."); } else { for (std::map::const_iterator sit = commandData.vectorParams.begin(); sit != commandData.vectorParams.end(); ++sit) { if ((sit->first != commandData.returnParam) && (sit->second == it->second)) { size = reduceName(commandData.params[sit->first].name) + ".size()"; break; } } } assert(!size.empty()); os << "( " << size << " )"; } } os << ";" << std::endl; } else if (1 < commandData.successCodes.size()) { os << indentation << " " << commandData.params[commandData.returnParam].pureType << " " << reduceName(commandData.params[commandData.returnParam].name) << ";" << std::endl; } } // local count variable to hold the size of the vector to fill if (commandData.twoStep) { assert(commandData.returnParam != ~0); std::map::const_iterator returnit = commandData.vectorParams.find(commandData.returnParam); assert(returnit != commandData.vectorParams.end() && (returnit->second != ~0)); assert((commandData.returnType == "Result") || (commandData.returnType == "void")); os << indentation << " " << commandData.params[returnit->second].pureType << " " << reduceName(commandData.params[returnit->second].name) << ";" << std::endl; } // write the function call os << indentation << " "; std::string localIndentation = " "; if (commandData.returnType == "Result") { os << "Result result"; if (commandData.twoStep && (1 < commandData.successCodes.size())) { os << ";" << std::endl << indentation << " do" << std::endl << indentation << " {" << std::endl << indentation << " result"; localIndentation += " "; } os << " = static_cast( "; } else if (commandData.returnType != "void") { assert(!commandData.twoStep); os << "return "; } assert(dependencyData.name == commandData.fullName); writeCall(os, commandData, vkData.vkTypes, true, singular); if (commandData.returnType == "Result") { os << " )"; } os << ";" << std::endl; if (commandData.twoStep) { std::map::const_iterator returnit = commandData.vectorParams.find(commandData.returnParam); if (commandData.returnType == "Result") { os << indentation << localIndentation << "if ( ( result == Result::eSuccess ) && " << reduceName(commandData.params[returnit->second].name) << " )" << std::endl << indentation << localIndentation << "{" << std::endl << indentation << localIndentation << " "; } else { os << indentation << " "; } // resize the vector to hold the data according to the result from the first call os << reduceName(commandData.params[returnit->first].name) << ".resize( " << reduceName(commandData.params[returnit->second].name) << " );" << std::endl; // write the function call a second time if (commandData.returnType == "Result") { os << indentation << localIndentation << " result = static_cast( "; } else { os << indentation << " "; } assert(!singular); assert(dependencyData.name == commandData.fullName); writeCall(os, commandData, vkData.vkTypes, false, singular); if (commandData.returnType == "Result") { os << " )"; } os << ";" << std::endl; if (commandData.returnType == "Result") { os << indentation << localIndentation << "}" << std::endl; if (1 < commandData.successCodes.size()) { os << indentation << " } while ( result == Result::eIncomplete );" << std::endl << indentation << " assert( " << reduceName(commandData.params[returnit->second].name) << " <= " << reduceName(commandData.params[returnit->first].name) << ".size() ); " << std::endl << indentation << " " << reduceName(commandData.params[returnit->first].name) << ".resize( " << reduceName(commandData.params[returnit->second].name) << " ); " << std::endl; } } } if ((commandData.returnType == "Result") || !commandData.successCodes.empty()) { os << indentation << " return createResultValue( result, "; if (commandData.returnParam != ~0) { os << reduceName(commandData.params[commandData.returnParam].name, singular) << ", "; } os << "\"vk::" << (commandData.className.empty() ? "" : commandData.className + "::") << reduceName(commandData.reducedName, singular) << "\""; if (1 < commandData.successCodes.size() && !commandData.twoStep) { os << ", { Result::" << commandData.successCodes[0]; for (size_t i = 1; i < commandData.successCodes.size(); i++) { os << ", Result::" << commandData.successCodes[i]; } os << " }"; } os << " );" << std::endl; } else if ((commandData.returnParam != ~0) && (commandData.returnType != commandData.enhancedReturnType)) { os << indentation << " return " << reduceName(commandData.params[commandData.returnParam].name) << ";" << std::endl; } } void writeFunctionBodyStandard(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData) { os << indentation << " "; bool castReturn = false; if (commandData.returnType != "void") { os << "return "; castReturn = (vkData.vkTypes.find(commandData.returnType) != vkData.vkTypes.end()); if (castReturn) { os << "static_cast<" << commandData.returnType << ">( "; } } std::string callName(dependencyData.name); assert(islower(callName[0])); callName[0] = toupper(callName[0]); os << "vk" << callName << "( "; if (!commandData.className.empty()) { os << "m_" << commandData.params[0].name; } bool argEncountered = false; for (size_t i = commandData.className.empty() ? 0 : 1; i < commandData.params.size(); i++) { if (0 < i) { os << ", "; } writeTypeCommandParam(os, commandData.params[i], vkData.vkTypes); } os << " )"; if (castReturn) { os << " )"; } os << ";" << std::endl; } void writeFunctionBodyUnique(std::ostream & os, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool singular) { std::string type = commandData.params[commandData.returnParam].pureType; std::string typeValue = char(tolower(type[0])) + type.substr(1); os << indentation << " " << type << "Deleter deleter( "; if (vkData.deleterData.find(commandData.className) != vkData.deleterData.end()) { os << "*this, "; } std::map::const_iterator ddit = vkData.deleterData.find(type); assert(ddit != vkData.deleterData.end()); if (ddit->second.pool.empty()) { os << "allocator"; } else { os << reduceName(commandData.params[1].name) << "." << char(tolower(ddit->second.pool[0])) << ddit->second.pool.substr(1); } os << " );" << std::endl; bool returnsVector = !singular && (commandData.vectorParams.find(commandData.returnParam) != commandData.vectorParams.end()); if (returnsVector) { os << indentation << " std::vector<" << type << ",Allocator> " << typeValue << "s = "; } else { os << indentation << " return Unique" << type << "( "; } os << reduceName(commandData.fullName, singular) << "( "; bool argEncountered = false; for (size_t i = commandData.className.empty() ? 0 : 1; i < commandData.params.size(); i++) { if (commandData.skippedParams.find(i) == commandData.skippedParams.end()) { if (argEncountered) { os << ", "; } argEncountered = true; os << reduceName(commandData.params[i].name, singular && (commandData.vectorParams.find(i) != commandData.vectorParams.end())); } } os << " )"; if (returnsVector) { os << ";" << std::endl << indentation << " std::vector unique" << type << "s;" << std::endl << indentation << " unique" << type << "s.reserve( " << typeValue << "s.size() );" << std::endl << indentation << " for ( auto " << typeValue << " : " << typeValue << "s )" << std::endl << indentation << " {" << std::endl << indentation << " unique" << type << "s.push_back( Unique" << type << "( " << typeValue << ", deleter ) );" << std::endl << indentation << " }" << std::endl << indentation << " return unique" << type << "s;" << std::endl; } else { os << ", deleter );" << std::endl; } } void writeFunctionDeclaration(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, bool enhanced, bool singular, bool unique) { if (enhanced) { writeFunctionHeaderTemplate(ofs, indentation, commandData, singular, true); } ofs << indentation; writeFunctionHeaderReturnType(ofs, indentation, commandData, enhanced, singular, unique); writeFunctionHeaderName(ofs, commandData.reducedName, singular, unique); writeFunctionHeaderArguments(ofs, vkData, commandData, enhanced, singular, true); ofs << ";" << std::endl; } void writeFunctionDefinition(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData, bool enhanced, bool singular, bool unique) { if (enhanced) { writeFunctionHeaderTemplate(ofs, indentation, commandData, singular, false); } ofs << indentation << "VULKAN_HPP_INLINE "; writeFunctionHeaderReturnType(ofs, indentation, commandData, enhanced, singular, unique); if (!commandData.className.empty()) { ofs << commandData.className << "::"; } writeFunctionHeaderName(ofs, commandData.reducedName, singular, unique); writeFunctionHeaderArguments(ofs, vkData, commandData, enhanced, singular, false); ofs << std::endl; writeFunctionBody(ofs, indentation, vkData, commandData, dependencyData, enhanced, singular, unique); } void writeFunctionHeaderArguments(std::ostream & os, VkData const& vkData, CommandData const& commandData, bool enhanced, bool singular, bool withDefaults) { os << "("; if (enhanced) { if (commandData.skippedParams.size() + (commandData.className.empty() ? 0 : 1) < commandData.params.size()) { size_t lastArgument = ~0; for (size_t i = commandData.params.size() - 1; i < commandData.params.size(); i--) { if (commandData.skippedParams.find(i) == commandData.skippedParams.end()) { lastArgument = i; break; } } os << " "; bool argEncountered = false; for (size_t i = commandData.className.empty() ? 0 : 1; i < commandData.params.size(); i++) { if (commandData.skippedParams.find(i) == commandData.skippedParams.end()) { if (argEncountered) { os << ", "; } std::map::const_iterator it = commandData.vectorParams.find(i); size_t pos = commandData.params[i].type.rfind('*'); if (it == commandData.vectorParams.end()) { if (pos == std::string::npos) { os << commandData.params[i].type << " " << reduceName(commandData.params[i].name); if (!commandData.params[i].arraySize.empty()) { os << "[" << commandData.params[i].arraySize << "]"; } if (lastArgument == i) { std::map::const_iterator flagIt = vkData.flags.find(commandData.params[i].pureType); if (flagIt != vkData.flags.end()) { std::list::const_iterator depIt = std::find_if(vkData.dependencies.begin(), vkData.dependencies.end(), [&flagIt](DependencyData const& dd) { return(dd.name == flagIt->first); }); assert(depIt != vkData.dependencies.end()); assert(depIt->dependencies.size() == 1); if (withDefaults) { std::map::const_iterator enumIt = vkData.enums.find(*depIt->dependencies.begin()); assert(enumIt != vkData.enums.end()); if (enumIt->second.members.empty()) { os << " = " << commandData.params[i].pureType << "()"; } } } } } else { assert(commandData.params[i].type[pos] == '*'); if (commandData.params[i].optional) { os << "Optional<" << trimEnd(commandData.params[i].type.substr(0, pos)) << "> " << reduceName(commandData.params[i].name); if (withDefaults) { os << " = nullptr"; } } else if (commandData.params[i].pureType == "void") { os << commandData.params[i].type << " " << commandData.params[i].name; } else if (commandData.params[i].pureType != "char") { os << trimEnd(commandData.params[i].type.substr(0, pos)) << " & " << reduceName(commandData.params[i].name); } else { os << "const std::string & " << reduceName(commandData.params[i].name); } } } else { bool optional = commandData.params[i].optional && ((it == commandData.vectorParams.end()) || (it->second == ~0)); assert(pos != std::string::npos); assert(commandData.params[i].type[pos] == '*'); if (commandData.params[i].type.find("char") != std::string::npos) { if (optional) { os << "Optional " << reduceName(commandData.params[i].name); if (withDefaults) { os << " = nullptr"; } } else { os << "const std::string & " << reduceName(commandData.params[i].name); } } else { assert(!optional); if (singular) { os << trimEnd(commandData.params[i].type.substr(0, pos)) << " & " << reduceName(commandData.params[i].name, true); } else { bool isConst = (commandData.params[i].type.find("const") != std::string::npos); os << "ArrayProxy<" << ((commandData.templateParam == i) ? (isConst ? "const T" : "T") : trimEnd(commandData.params[i].type.substr(0, pos))) << "> " << reduceName(commandData.params[i].name); } } } argEncountered = true; } } os << " "; } } else { bool argEncountered = false; for (size_t i = commandData.className.empty() ? 0 : 1; i < commandData.params.size(); i++) { if (argEncountered) { os << ","; } os << " " << commandData.params[i].type << " " << commandData.params[i].name; if (!commandData.params[i].arraySize.empty()) { os << "[" << commandData.params[i].arraySize << "]"; } argEncountered = true; } if (argEncountered) { os << " "; } } os << ")"; if (!commandData.className.empty()) { os << " const"; } } void writeFunctionHeaderName(std::ostream & os, std::string const& name, bool singular, bool unique) { os << reduceName(name, singular); if (unique) { os << "Unique"; } } void writeFunctionHeaderReturnType(std::ostream & os, std::string const& indentation, CommandData const& commandData, bool enhanced, bool singular, bool unique) { if (enhanced && !singular && (commandData.enhancedReturnType.find("Allocator") != std::string::npos)) { if (!unique && (commandData.enhancedReturnType != commandData.returnType) && (commandData.returnType != "void")) { os << "typename "; } } if (enhanced) { if (unique) { bool returnsVector = !singular && (commandData.vectorParams.find(commandData.returnParam) != commandData.vectorParams.end()); if (returnsVector) { os << "std::vector<"; } os << "Unique" << commandData.params[commandData.returnParam].pureType; if (returnsVector) { os << ">"; } os << " "; } else if ((commandData.enhancedReturnType != commandData.returnType) && (commandData.returnType != "void")) { assert(commandData.returnType == "Result"); os << "ResultValueType<" << (singular ? commandData.params[commandData.returnParam].pureType : commandData.enhancedReturnType) << ">::type "; } else if ((commandData.returnParam != ~0) && (1 < commandData.successCodes.size())) { assert(commandData.returnType == "Result"); os << "ResultValue<" << commandData.params[commandData.returnParam].pureType << "> "; } else { os << commandData.enhancedReturnType << " "; } } else { os << commandData.returnType << " "; } } void writeFunctionHeaderTemplate(std::ostream & os, std::string const& indentation, CommandData const& commandData, bool singular, bool withDefault) { if (!singular && (commandData.templateParam != ~0) && ((commandData.templateParam != commandData.returnParam) || (commandData.enhancedReturnType == "Result"))) { assert(commandData.enhancedReturnType.find("Allocator") == std::string::npos); os << indentation << "template " << std::endl; } else if (!singular && (commandData.enhancedReturnType.find("Allocator") != std::string::npos)) { assert((commandData.enhancedReturnType.substr(0, 12) == "std::vector<") && (commandData.enhancedReturnType.find(',') != std::string::npos) && (12 < commandData.enhancedReturnType.find(','))); os << indentation << "template "; } os << "> " << std::endl; } } void writeStructConstructor( std::ofstream & ofs, std::string const& name, StructData const& structData, std::set const& vkTypes, std::map const& defaultValues ) { // the constructor with all the elements as arguments, with defaults ofs << " " << name << "( "; bool listedArgument = false; for (size_t i = 0; i::const_iterator defaultIt = defaultValues.find(structData.members[i].pureType); assert(defaultIt != defaultValues.end()); if (structData.members[i].arraySize.empty()) { ofs << structData.members[i].type + " " + structData.members[i].name << "_ = " << (structData.members[i].type.back() == '*' ? "nullptr" : defaultIt->second); } else { ofs << "std::array<" + structData.members[i].type + "," + structData.members[i].arraySize + "> const& " + structData.members[i].name << "_ = { { " << defaultIt->second; size_t n = atoi(structData.members[i].arraySize.c_str()); assert(0 < n); for (size_t j = 1; j < n; j++) { ofs << ", " << defaultIt->second; } ofs << " } }"; } listedArgument = true; } } ofs << " )" << std::endl; // copy over the simple arguments bool firstArgument = true; for (size_t i = 0; i < structData.members.size(); i++) { if (structData.members[i].arraySize.empty()) { ofs << " " << (firstArgument ? ":" : ",") << " " << structData.members[i].name << "( "; if (structData.members[i].name == "pNext") { ofs << "nullptr"; } else if (structData.members[i].name == "sType") { ofs << "StructureType::e" << name; } else { ofs << structData.members[i].name << "_"; } ofs << " )" << std::endl; firstArgument = false; } } // the body of the constructor, copying over data from argument list into wrapped struct ofs << " {" << std::endl; for ( size_t i=0 ; i const& vkTypes ) { if (memberData.type != "StructureType") // filter out StructureType, which is supposed to be immutable ! { ofs << " " << name << "& set" << static_cast(toupper(memberData.name[0])) << memberData.name.substr(1) << "( "; if (memberData.arraySize.empty()) { ofs << memberData.type << " "; } else { ofs << "std::array<" << memberData.type << "," << memberData.arraySize << "> "; } ofs << memberData.name << "_ )" << std::endl << " {" << std::endl; if (!memberData.arraySize.empty()) { ofs << " memcpy( &" << memberData.name << ", " << memberData.name << "_.data(), " << memberData.arraySize << " * sizeof( " << memberData.type << " ) )"; } else { ofs << " " << memberData.name << " = " << memberData.name << "_"; } ofs << ";" << std::endl << " return *this;" << std::endl << " }" << std::endl << std::endl; } } void writeTypeCommand(std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData) { assert(vkData.commands.find(dependencyData.name) != vkData.commands.end()); CommandData const& commandData = vkData.commands.find(dependencyData.name)->second; if (commandData.className.empty()) { if (commandData.fullName == "createInstance") { auto deleterTypesIt = vkData.deleterTypes.find(""); assert((deleterTypesIt != vkData.deleterTypes.end()) && (deleterTypesIt->second.size() == 1)); writeDeleterForwardDeclarations(ofs, *deleterTypesIt, vkData.deleterData); } writeTypeCommandDeclaration(ofs, " ", vkData, commandData); if (commandData.fullName == "createInstance") { auto deleterTypesIt = vkData.deleterTypes.find(""); assert((deleterTypesIt != vkData.deleterTypes.end()) && (deleterTypesIt->second.size() == 1)); writeDeleterClasses(ofs, *deleterTypesIt, vkData.deleterData); } writeTypeCommandDefinition(ofs, " ", vkData, commandData, dependencyData); ofs << std::endl; } } void writeTypeCommandDeclaration(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData) { enterProtect(ofs, commandData.protect); bool unchangedInterface = !hasPointerParam(commandData.params); if ((commandData.enhancedReturnType == commandData.returnType) && commandData.vectorParams.empty() && unchangedInterface && (commandData.successCodes.size() <= 1)) { writeTypeCommandDeclarationStandard(ofs, " ", vkData, commandData); } else { if (unchangedInterface) { ofs << "#ifdef VULKAN_HPP_DISABLE_ENHANCED_MODE" << std::endl; } writeTypeCommandDeclarationStandard(ofs, " ", vkData, commandData); ofs << (unchangedInterface ? "#else" : "#ifndef VULKAN_HPP_DISABLE_ENHANCED_MODE") << std::endl; writeTypeCommandDeclarationEnhanced(ofs, " ", vkData, commandData); ofs << "#endif /*VULKAN_HPP_DISABLE_ENHANCED_MODE*/" << std::endl; } leaveProtect(ofs, commandData.protect); ofs << std::endl; } void writeTypeCommandDeclarationEnhanced(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData) { writeFunctionDeclaration(ofs, indentation, vkData, commandData, true, false, false); // determine candidates for singular version of function std::map::const_iterator returnVector = commandData.vectorParams.find(commandData.returnParam); bool singular = (returnVector != commandData.vectorParams.end()) && (returnVector->second != ~0) && (commandData.params[returnVector->second].type.back() != '*'); if (singular) { writeFunctionDeclaration(ofs, indentation, vkData, commandData, true, true, false); } // special handling for createDevice and createInstance ! bool specialWriteUnique = (commandData.reducedName == "createDevice") || (commandData.reducedName == "createInstance"); if (((vkData.deleterData.find(commandData.className) != vkData.deleterData.end()) || specialWriteUnique) && ((commandData.reducedName.substr(0, 8) == "allocate") || (commandData.reducedName.substr(0, 6) == "create"))) { ofs << "#ifndef VULKAN_HPP_NO_SMART_HANDLE" << std::endl; writeFunctionDeclaration(ofs, indentation, vkData, commandData, true, false, true); if (singular) { writeFunctionDeclaration(ofs, indentation, vkData, commandData, true, true, true); } ofs << "#endif /*VULKAN_HPP_NO_SMART_HANDLE*/" << std::endl; } } void writeTypeCommandDeclarationStandard(std::ofstream & ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData) { writeFunctionDeclaration(ofs, indentation, vkData, commandData, false, false, false); } void writeTypeCommandDefinition(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData) { enterProtect(ofs, commandData.protect); bool unchangedInterface = !hasPointerParam(commandData.params); if ((commandData.enhancedReturnType == commandData.returnType) && commandData.vectorParams.empty() && unchangedInterface && (commandData.successCodes.size() <= 1)) { writeTypeCommandDefinitionStandard(ofs, indentation, vkData, commandData, dependencyData); } else { if (unchangedInterface) { ofs << "#ifdef VULKAN_HPP_DISABLE_ENHANCED_MODE" << std::endl; } writeTypeCommandDefinitionStandard(ofs, indentation, vkData, commandData, dependencyData); ofs << (unchangedInterface ? "#else" : "#ifndef VULKAN_HPP_DISABLE_ENHANCED_MODE") << std::endl; writeTypeCommandDefinitionEnhanced(ofs, indentation, vkData, commandData, dependencyData); ofs << "#endif /*VULKAN_HPP_DISABLE_ENHANCED_MODE*/" << std::endl; } leaveProtect(ofs, commandData.protect); } void writeTypeCommandDefinitionEnhanced(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData) { writeFunctionDefinition(ofs, indentation, vkData, commandData, dependencyData, true, false, false); // determine candidates for singular version of function std::map::const_iterator returnVector = commandData.vectorParams.find(commandData.returnParam); bool singular = (returnVector != commandData.vectorParams.end()) && (returnVector->second != ~0) && (commandData.params[returnVector->second].type.back() != '*'); if (singular) { writeFunctionDefinition(ofs, indentation, vkData, commandData, dependencyData, true, true, false); } // special handling for createDevice and createInstance ! bool specialWriteUnique = (commandData.reducedName == "createDevice") || (commandData.reducedName == "createInstance"); if (((vkData.deleterData.find(commandData.className) != vkData.deleterData.end()) || specialWriteUnique) && ((commandData.reducedName.substr(0, 8) == "allocate") || (commandData.reducedName.substr(0, 6) == "create"))) { ofs << "#ifndef VULKAN_HPP_NO_SMART_HANDLE" << std::endl; writeFunctionDefinition(ofs, indentation, vkData, commandData, dependencyData, true, false, true); if (singular) { writeFunctionDefinition(ofs, indentation, vkData, commandData, dependencyData, true, true, true); } ofs << "#endif /*VULKAN_HPP_NO_SMART_HANDLE*/" << std::endl; } } void writeTypeCommandDefinitionStandard(std::ofstream &ofs, std::string const& indentation, VkData const& vkData, CommandData const& commandData, DependencyData const& dependencyData) { writeFunctionDefinition(ofs, indentation, vkData, commandData, dependencyData, false, false, false); } void writeTypeCommandParam(std::ostream & os, ParamData const& param, std::set const& vkTypes) { if (vkTypes.find(param.pureType) != vkTypes.end()) { if (param.type.back() == '*') { os << "reinterpret_cast<"; if (param.type.find("const") == 0) { os << "const "; } os << "Vk" << param.pureType; if (param.type.find("* const") != std::string::npos) { os << "* const"; } os << '*'; } else { os << "static_cast( " << param.name << " )"; } else { os << param.name; } } void writeTypeEnum( std::ofstream & ofs, EnumData const& enumData ) { enterProtect(ofs, enumData.protect); ofs << " enum class " << enumData.name << std::endl << " {" << std::endl; for ( size_t i=0 ; i> const& deleterTypes, std::map const& deleterData) { ofs << "#ifndef VULKAN_HPP_NO_SMART_HANDLE" << std::endl; bool first = true; std::string firstName = deleterTypes.first.empty() ? "" : (char(tolower(deleterTypes.first[0])) + deleterTypes.first.substr(1)); for (auto const& dt : deleterTypes.second) { std::string value = char(tolower(dt[0])) + dt.substr(1); if (!first) { ofs << std::endl; } first = false; ofs << " class " << dt << "Deleter" << std::endl << " {" << std::endl << " public:" << std::endl << " " << dt << "Deleter( "; if (!deleterTypes.first.empty() && (dt != "Device")) { ofs << deleterTypes.first << " " << firstName << " = " << deleterTypes.first << "(), "; } auto const& dd = deleterData.find(dt); assert(dd != deleterData.end()); std::string poolName = (dd->second.pool.empty() ? "" : (char(tolower(dd->second.pool[0])) + dd->second.pool.substr(1))); if (poolName.empty()) { ofs << "Optional allocator = nullptr )" << std::endl; } else { assert(!dd->second.pool.empty()); poolName = char(tolower(dd->second.pool[0])) + dd->second.pool.substr(1); ofs << dd->second.pool << " " << poolName << " = " << dd->second.pool << "() )" << std::endl; } ofs << " : "; if (!deleterTypes.first.empty() && (dt != "Device")) { ofs << "m_" << firstName << "( " << firstName << " )" << std::endl << " , "; } if (poolName.empty()) { ofs << "m_allocator( allocator )" << std::endl; } else { ofs << "m_" << poolName << "( " << poolName << " )" << std::endl; } ofs << " {}" << std::endl << std::endl << " void operator()( " << dt << " " << char(tolower(dt[0])) + dt.substr(1) << " )" << std::endl << " {" << std::endl; if (dt == "Device") { ofs << " device."; } else if (deleterTypes.first.empty()) { ofs << " " << value << "."; } else { ofs << " m_" << char(tolower(deleterTypes.first[0])) << deleterTypes.first.substr(1) << "."; } ofs << dd->second.call << "( "; if (!dd->second.pool.empty()) { ofs << "m_" << char(tolower(dd->second.pool[0])) << dd->second.pool.substr(1) << ", "; } if (!deleterTypes.first.empty() && (dt != "Device")) { ofs << value; } if (dd->second.pool.empty()) { if (!deleterTypes.first.empty() && (dt != "Device")) { ofs << ", "; } ofs << "m_allocator"; } ofs << " );" << std::endl << " }" << std::endl << std::endl << " private:" << std::endl; if (!deleterTypes.first.empty() && (dt != "Device")) { ofs << " " << deleterTypes.first << " m_" << firstName << ";" << std::endl; } if (poolName.empty()) { ofs << " Optional m_allocator;" << std::endl; } else { ofs << " " << dd->second.pool << " m_" << poolName << ";" << std::endl; } ofs << " };" << std::endl; } ofs << "#endif /*VULKAN_HPP_NO_SMART_HANDLE*/" << std::endl << std::endl; } void writeDeleterForwardDeclarations(std::ofstream &ofs, std::pair> const& deleterTypes, std::map const& deleterData) { ofs << "#ifndef VULKAN_HPP_NO_SMART_HANDLE" << std::endl; bool first = true; std::string firstName = deleterTypes.first.empty() ? "" : (char(tolower(deleterTypes.first[0])) + deleterTypes.first.substr(1)); for (auto const& dt : deleterTypes.second) { ofs << " class " << dt << "Deleter;" << std::endl; ofs << " using Unique" << dt << " = UniqueHandle<" << dt << ", " << dt << "Deleter>;" << std::endl; } ofs << "#endif /*VULKAN_HPP_NO_SMART_HANDLE*/" << std::endl << std::endl; } void writeEnumsToString(std::ofstream & ofs, EnumData const& enumData) { enterProtect(ofs, enumData.protect); ofs << " VULKAN_HPP_INLINE std::string to_string(" << enumData.name << (enumData.members.empty() ? ")" : " value)") << std::endl << " {" << std::endl; if (enumData.members.empty()) { ofs << " return \"(void)\";" << std::endl; } else { ofs << " switch (value)" << std::endl << " {" << std::endl; for (auto itMember = enumData.members.begin(); itMember != enumData.members.end(); ++itMember) { ofs << " case " << enumData.name << "::" << itMember->name << ": return \"" << itMember->name.substr(1) << "\";" << std::endl; } ofs << " default: return \"invalid\";" << std::endl << " }" << std::endl; } ofs << " }" << std::endl; leaveProtect(ofs, enumData.protect); ofs << std::endl; } void writeFlagsToString(std::ofstream & ofs, std::string const& flagsName, EnumData const &enumData) { enterProtect(ofs, enumData.protect); ofs << " VULKAN_HPP_INLINE std::string to_string(" << flagsName << (enumData.members.empty() ? ")" : " value)") << std::endl << " {" << std::endl; if (enumData.members.empty()) { ofs << " return \"{}\";" << std::endl; } else { ofs << " if (!value) return \"{}\";" << std::endl << " std::string result;" << std::endl; for (auto itMember = enumData.members.begin(); itMember != enumData.members.end(); ++itMember) { ofs << " if (value & " << enumData.name << "::" << itMember->name << ") result += \"" << itMember->name.substr(1) << " | \";" << std::endl; } ofs << " return \"{\" + result.substr(0, result.size() - 3) + \"}\";" << std::endl; } ofs << " }" << std::endl; leaveProtect(ofs, enumData.protect); ofs << std::endl; } void writeEnumsToString(std::ofstream & ofs, VkData const& vkData) { for (auto it = vkData.dependencies.begin(); it != vkData.dependencies.end(); ++it) { switch (it->category) { case DependencyData::Category::ENUM: assert(vkData.enums.find(it->name) != vkData.enums.end()); writeEnumsToString(ofs, vkData.enums.find(it->name)->second); break; case DependencyData::Category::FLAGS: writeFlagsToString(ofs, it->name, vkData.enums.find(*it->dependencies.begin())->second); break; } } } void writeTypeFlags(std::ofstream & ofs, std::string const& flagsName, FlagData const& flagData, EnumData const& enumData) { enterProtect(ofs, flagData.protect); ofs << " using " << flagsName << " = Flags<" << enumData.name << ", Vk" << flagsName << ">;" << std::endl << std::endl << " VULKAN_HPP_INLINE " << flagsName << " operator|( " << enumData.name << " bit0, " << enumData.name << " bit1 )" << std::endl << " {" << std::endl << " return " << flagsName << "( bit0 ) | bit1;" << std::endl << " }" << std::endl; if (!enumData.members.empty()) { ofs << std::endl << " VULKAN_HPP_INLINE " << flagsName << " operator~( " << enumData.name << " bits )" << std::endl << " {" << std::endl << " return ~( " << flagsName << "( bits ) );" << std::endl << " }" << std::endl << std::endl << " template <> struct FlagTraits<" << enumData.name << ">" << std::endl << " {" << std::endl << " enum" << std::endl << " {" << std::endl << " allFlags = "; for (size_t i = 0; i < enumData.members.size(); i++) { if (i != 0) { ofs << " | "; } ofs << "VkFlags(" << enumData.name << "::" << enumData.members[i].name << ")"; } ofs << std::endl << " };" << std::endl << " };" << std::endl; } leaveProtect(ofs, flagData.protect); ofs << std::endl; } void writeTypeHandle(std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData, HandleData const& handleData, std::list const& dependencies) { std::string memberName = dependencyData.name; assert(isupper(memberName[0])); memberName[0] = tolower(memberName[0]); enterProtect(ofs, handleData.protect); if (!dependencyData.forwardDependencies.empty()) { ofs << " // forward declarations" << std::endl; for (std::set::const_iterator it = dependencyData.forwardDependencies.begin(); it != dependencyData.forwardDependencies.end(); ++it) { assert(vkData.structs.find(*it) != vkData.structs.end()); ofs << " struct " << *it << ";" << std::endl; } ofs << std::endl; } std::map>::const_iterator deleterTypesIt = vkData.deleterTypes.find(dependencyData.name); if (deleterTypesIt != vkData.deleterTypes.end()) { writeDeleterForwardDeclarations(ofs, *deleterTypesIt, vkData.deleterData); } ofs << " class " << dependencyData.name << std::endl << " {" << std::endl << " public:" << std::endl << " " << dependencyData.name << "()" << std::endl << " : m_" << memberName << "(VK_NULL_HANDLE)" << std::endl << " {}" << std::endl << std::endl << " " << dependencyData.name << "( nullptr_t )" << std::endl << " : m_" << memberName << "(VK_NULL_HANDLE)" << std::endl << " {}" << std::endl << std::endl // construct from native handle << " VULKAN_HPP_TYPESAFE_EXPLICIT " << dependencyData.name << "(Vk" << dependencyData.name << " " << memberName << ")" << std::endl << " : m_" << memberName << "(" << memberName << ")" << std::endl << " {}" << std::endl << std::endl // assignment from native handle << "#if defined(VULKAN_HPP_TYPESAFE_CONVERSION)" << std::endl << " " << dependencyData.name << "& operator=(Vk" << dependencyData.name << " " << memberName << ")" << std::endl << " {" << std::endl << " m_" << memberName << " = " << memberName << ";" << std::endl << " return *this;" << std::endl << " }" << std::endl << "#endif\n" << std::endl // assignment from nullptr_t << " " << dependencyData.name << "& operator=( nullptr_t )" << std::endl << " {" << std::endl << " m_" << memberName << " = VK_NULL_HANDLE;" << std::endl << " return *this;" << std::endl << " }" << std::endl << std::endl // operator== << " bool operator==" << "(" << dependencyData.name << " const &rhs) const" << std::endl << " {" << std::endl << " return m_" << memberName << " == rhs.m_" << memberName << ";" << std::endl << " }" << std::endl << std::endl // operator!= << " bool operator!=" << "(" << dependencyData.name << " const &rhs) const" << std::endl << " {" << std::endl << " return m_" << memberName << " != rhs.m_" << memberName << ";" << std::endl << " }" << std::endl << std::endl // operator< << " bool operator<" << "(" << dependencyData.name << " const &rhs) const" << std::endl << " {" << std::endl << " return m_" << memberName << " < rhs.m_" << memberName << ";" << std::endl << " }" << std::endl << std::endl; for (size_t i = 0; i < handleData.commands.size(); i++) { std::string commandName = handleData.commands[i]; std::map::const_iterator cit = vkData.commands.find(commandName); assert((cit != vkData.commands.end()) && !cit->second.className.empty()); writeTypeCommandDeclaration(ofs, " ", vkData, cit->second); } ofs << " VULKAN_HPP_TYPESAFE_EXPLICIT operator Vk" << dependencyData.name << "() const" << std::endl << " {" << std::endl << " return m_" << memberName << ";" << std::endl << " }" << std::endl << std::endl << " explicit operator bool() const" << std::endl << " {" << std::endl << " return m_" << memberName << " != VK_NULL_HANDLE;" << std::endl << " }" << std::endl << std::endl << " bool operator!() const" << std::endl << " {" << std::endl << " return m_" << memberName << " == VK_NULL_HANDLE;" << std::endl << " }" << std::endl << std::endl << " private:" << std::endl << " Vk" << dependencyData.name << " m_" << memberName << ";" << std::endl << " };" << std::endl #if 1 << " static_assert( sizeof( " << dependencyData.name << " ) == sizeof( Vk" << dependencyData.name << " ), \"handle and wrapper have different size!\" );" << std::endl #endif << std::endl; deleterTypesIt = vkData.deleterTypes.find(dependencyData.name); if (deleterTypesIt != vkData.deleterTypes.end()) { writeDeleterClasses(ofs, *deleterTypesIt, vkData.deleterData); } for (size_t i = 0; i < handleData.commands.size(); i++) { std::string commandName = handleData.commands[i]; std::map::const_iterator cit = vkData.commands.find(commandName); assert((cit != vkData.commands.end()) && !cit->second.className.empty()); std::list::const_iterator dep = std::find_if(dependencies.begin(), dependencies.end(), [commandName](DependencyData const& dd) { return dd.name == commandName; }); assert(dep != dependencies.end() && (dep->name == cit->second.fullName)); writeTypeCommandDefinition(ofs, " ", vkData, cit->second, *dep); if (i < handleData.commands.size() - 1) { ofs << std::endl; } } leaveProtect(ofs, handleData.protect); } void writeTypeScalar( std::ofstream & ofs, DependencyData const& dependencyData ) { assert( dependencyData.dependencies.size() == 1 ); ofs << " using " << dependencyData.name << " = " << *dependencyData.dependencies.begin() << ";" << std::endl << std::endl; } bool containsUnion(std::string const& type, std::map const& structs) { std::map::const_iterator sit = structs.find(type); bool found = (sit != structs.end()); if (found) { found = sit->second.isUnion; for (std::vector::const_iterator mit = sit->second.members.begin(); mit != sit->second.members.end() && !found; ++mit) { found = (mit->type == mit->pureType) && containsUnion(mit->type, structs); } } return found; } void writeTypeStruct( std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData, std::map const& defaultValues ) { std::map::const_iterator it = vkData.structs.find( dependencyData.name ); assert( it != vkData.structs.end() ); enterProtect(ofs, it->second.protect); ofs << " struct " << dependencyData.name << std::endl << " {" << std::endl; // only structs that are not returnedOnly get a constructor! if ( !it->second.returnedOnly ) { writeStructConstructor( ofs, dependencyData.name, it->second, vkData.vkTypes, defaultValues ); } // create the setters if (!it->second.returnedOnly) { for (size_t i = 0; isecond.members.size(); i++) { writeStructSetter( ofs, dependencyData.name, it->second.members[i], vkData.vkTypes ); } } // the cast-operator to the wrapped struct ofs << " operator const Vk" << dependencyData.name << "&() const" << std::endl << " {" << std::endl << " return *reinterpret_cast(this);" << std::endl << " }" << std::endl << std::endl; // operator==() and operator!=() // only structs without a union as a member can have a meaningfull == and != operation; we filter them out if (!containsUnion(dependencyData.name, vkData.structs)) { ofs << " bool operator==( " << dependencyData.name << " const& rhs ) const" << std::endl << " {" << std::endl << " return "; for (size_t i = 0; i < it->second.members.size(); i++) { if (i != 0) { ofs << std::endl << " && "; } if (!it->second.members[i].arraySize.empty()) { ofs << "( memcmp( " << it->second.members[i].name << ", rhs." << it->second.members[i].name << ", " << it->second.members[i].arraySize << " * sizeof( " << it->second.members[i].type << " ) ) == 0 )"; } else { ofs << "( " << it->second.members[i].name << " == rhs." << it->second.members[i].name << " )"; } } ofs << ";" << std::endl << " }" << std::endl << std::endl << " bool operator!=( " << dependencyData.name << " const& rhs ) const" << std::endl << " {" << std::endl << " return !operator==( rhs );" << std::endl << " }" << std::endl << std::endl; } // the member variables for (size_t i = 0; i < it->second.members.size(); i++) { if (it->second.members[i].type == "StructureType") { assert((i == 0) && (it->second.members[i].name == "sType")); ofs << " private:" << std::endl << " StructureType sType;" << std::endl << std::endl << " public:" << std::endl; } else { ofs << " " << it->second.members[i].type << " " << it->second.members[i].name; if (!it->second.members[i].arraySize.empty()) { ofs << "[" << it->second.members[i].arraySize << "]"; } ofs << ";" << std::endl; } } ofs << " };" << std::endl << " static_assert( sizeof( " << dependencyData.name << " ) == sizeof( Vk" << dependencyData.name << " ), \"struct and wrapper have different size!\" );" << std::endl; leaveProtect(ofs, it->second.protect); ofs << std::endl; } void writeTypeUnion( std::ofstream & ofs, VkData const& vkData, DependencyData const& dependencyData, StructData const& unionData, std::map const& defaultValues ) { std::ostringstream oss; ofs << " union " << dependencyData.name << std::endl << " {" << std::endl; for ( size_t i=0 ; i& "; } ofs << unionData.members[i].name << "_"; // just the very first constructor gets default arguments if ( i == 0 ) { std::map::const_iterator it = defaultValues.find( unionData.members[i].pureType ); assert( it != defaultValues.end() ); if ( unionData.members[i].arraySize.empty() ) { ofs << " = " << it->second; } else { ofs << " = { {" << it->second << "} }"; } } ofs << " )" << std::endl << " {" << std::endl << " "; if ( unionData.members[i].arraySize.empty() ) { ofs << unionData.members[i].name << " = " << unionData.members[i].name << "_"; } else { ofs << "memcpy( &" << unionData.members[i].name << ", " << unionData.members[i].name << "_.data(), " << unionData.members[i].arraySize << " * sizeof( " << unionData.members[i].type << " ) )"; } ofs << ";" << std::endl << " }" << std::endl << std::endl; } for (size_t i = 0; i(this);" << std::endl << " }" << std::endl << std::endl; // the union member variables // if there's at least one Vk... type in this union, check for unrestricted unions support bool needsUnrestrictedUnions = false; for (size_t i = 0; i < unionData.members.size() && !needsUnrestrictedUnions; i++) { needsUnrestrictedUnions = (vkData.vkTypes.find(unionData.members[i].type) != vkData.vkTypes.end()); } if (needsUnrestrictedUnions) { ofs << "#ifdef VULKAN_HPP_HAS_UNRESTRICTED_UNIONS" << std::endl; for (size_t i = 0; i < unionData.members.size(); i++) { ofs << " " << unionData.members[i].type << " " << unionData.members[i].name; if (!unionData.members[i].arraySize.empty()) { ofs << "[" << unionData.members[i].arraySize << "]"; } ofs << ";" << std::endl; } ofs << "#else" << std::endl; } for (size_t i = 0; i < unionData.members.size(); i++) { ofs << " "; if (vkData.vkTypes.find(unionData.members[i].type) != vkData.vkTypes.end()) { ofs << "Vk"; } ofs << unionData.members[i].type << " " << unionData.members[i].name; if (!unionData.members[i].arraySize.empty()) { ofs << "[" << unionData.members[i].arraySize << "]"; } ofs << ";" << std::endl; } if (needsUnrestrictedUnions) { ofs << "#endif // VULKAN_HPP_HAS_UNRESTRICTED_UNIONS" << std::endl; } ofs << " };" << std::endl << std::endl; } void writeTypes(std::ofstream & ofs, VkData const& vkData, std::map const& defaultValues) { for ( std::list::const_iterator it = vkData.dependencies.begin() ; it != vkData.dependencies.end() ; ++it ) { switch( it->category ) { case DependencyData::Category::COMMAND : writeTypeCommand( ofs, vkData, *it ); break; case DependencyData::Category::ENUM : assert( vkData.enums.find( it->name ) != vkData.enums.end() ); writeTypeEnum( ofs, vkData.enums.find( it->name )->second ); break; case DependencyData::Category::FLAGS : assert(vkData.flags.find(it->name) != vkData.flags.end()); writeTypeFlags( ofs, it->name, vkData.flags.find( it->name)->second, vkData.enums.find(generateEnumNameForFlags(it->name))->second ); break; case DependencyData::Category::FUNC_POINTER : case DependencyData::Category::REQUIRED : // skip FUNC_POINTER and REQUIRED, they just needed to be in the dependencies list to resolve dependencies break; case DependencyData::Category::HANDLE : assert(vkData.handles.find(it->name) != vkData.handles.end()); writeTypeHandle(ofs, vkData, *it, vkData.handles.find(it->name)->second, vkData.dependencies); break; case DependencyData::Category::SCALAR : writeTypeScalar( ofs, *it ); break; case DependencyData::Category::STRUCT : writeTypeStruct( ofs, vkData, *it, defaultValues ); break; case DependencyData::Category::UNION : assert( vkData.structs.find( it->name ) != vkData.structs.end() ); writeTypeUnion( ofs, vkData, *it, vkData.structs.find( it->name )->second, defaultValues ); break; default : assert( false ); break; } } } void writeVersionCheck(std::ofstream & ofs, std::string const& version) { ofs << "static_assert( VK_HEADER_VERSION == " << version << " , \"Wrong VK_HEADER_VERSION!\" );" << std::endl << std::endl; } void writeTypesafeCheck(std::ofstream & ofs, std::string const& typesafeCheck) { ofs << "// 32-bit vulkan is not typesafe for handles, so don't allow copy constructors on this platform by default." << std::endl << "// To enable this feature on 32-bit platforms please define VULKAN_HPP_TYPESAFE_CONVERSION" << std::endl << typesafeCheck << std::endl << "# if !defined( VULKAN_HPP_TYPESAFE_CONVERSION )" << std::endl << "# define VULKAN_HPP_TYPESAFE_CONVERSION" << std::endl << "# endif" << std::endl << "#endif" << std::endl; } int main( int argc, char **argv ) { try { tinyxml2::XMLDocument doc; std::string filename = (argc == 1) ? VK_SPEC : argv[1]; std::cout << "Loading vk.xml from " << filename << std::endl; std::cout << "Writing vulkan.hpp to " << VULKAN_HPP << std::endl; tinyxml2::XMLError error = doc.LoadFile(filename.c_str()); if (error != tinyxml2::XML_SUCCESS) { std::cout << "VkGenerate: failed to load file " << argv[1] << " . Error code: " << error << std::endl; return -1; } tinyxml2::XMLElement * registryElement = doc.FirstChildElement(); assert(strcmp(registryElement->Value(), "registry") == 0); assert(!registryElement->NextSiblingElement()); VkData vkData; vkData.handles[""]; // insert the default "handle" without class (for createInstance, and such) tinyxml2::XMLElement * child = registryElement->FirstChildElement(); do { assert(child->Value()); const std::string value = child->Value(); if (value == "commands") { readCommands(child, vkData); } else if (value == "comment") { readComment(child, vkData.vulkanLicenseHeader); vkData.vulkanLicenseHeader = trimStart(vkData.vulkanLicenseHeader); } else if (value == "enums") { readEnums(child, vkData); } else if (value == "extensions") { readExtensions(child, vkData); } else if (value == "tags") { readTags(child, vkData.tags); } else if (value == "types") { readTypes(child, vkData); } else { assert((value == "feature") || (value == "vendorids")); } } while (child = child->NextSiblingElement()); sortDependencies(vkData.dependencies); std::map defaultValues; createDefaults(vkData, defaultValues); std::ofstream ofs(VULKAN_HPP); ofs << vkData.vulkanLicenseHeader << std::endl << std::endl << std::endl << "#ifndef VULKAN_HPP" << std::endl << "#define VULKAN_HPP" << std::endl << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#include " << std::endl << "#ifndef VULKAN_HPP_DISABLE_ENHANCED_MODE" << std::endl << "# include " << std::endl << "# include " << std::endl << "#endif /*VULKAN_HPP_DISABLE_ENHANCED_MODE*/" << std::endl << std::endl; writeVersionCheck(ofs, vkData.version); writeTypesafeCheck(ofs, vkData.typesafeCheck); ofs << versionCheckHeader << inlineHeader << explicitHeader << std::endl << "namespace vk" << std::endl << "{" << std::endl << flagsHeader << optionalClassHeader << arrayProxyHeader << uniqueHandleHeader; // first of all, write out vk::Result and the exception handling stuff std::list::const_iterator it = std::find_if(vkData.dependencies.begin(), vkData.dependencies.end(), [](DependencyData const& dp) { return dp.name == "Result"; }); assert(it != vkData.dependencies.end()); writeTypeEnum(ofs, vkData.enums.find(it->name)->second); writeEnumsToString(ofs, vkData.enums.find(it->name)->second); vkData.dependencies.erase(it); ofs << exceptionHeader; ofs << "} // namespace vk" << std::endl << std::endl << "namespace std" << std::endl << "{" << std::endl << " template <>" << std::endl << " struct is_error_code_enum : public true_type" << std::endl << " {};" << std::endl << "}" << std::endl << std::endl << "namespace vk" << std::endl << "{" << std::endl << resultValueHeader << createResultValueHeader; assert(vkData.deleterTypes.find("") != vkData.deleterTypes.end()); writeTypes(ofs, vkData, defaultValues); writeEnumsToString(ofs, vkData); ofs << "} // namespace vk" << std::endl << std::endl << "#endif" << std::endl; } catch (std::exception const& e) { std::cout << "caught exception: " << e.what() << std::endl; return -1; } catch (...) { std::cout << "caught unknown exception" << std::endl; return -1; } }