mirror of
				https://github.com/encounter/dawn-cmake.git
				synced 2025-10-20 16:55:51 +00:00 
			
		
		
		
	Notes: - Separates ChainedStruct to be reusable without cpp header. (Also updates native structs to directly use it.) - Manually implements the descriptor in DawnNative. - Reworks ChainUtils with mapping from struct to STypes. - Updates the tests to use either SetPlatformForTesting which is still required because DawnTest uses a "global" instance for all tests and some tests require setting (and cleaning up) a test specific platform. Bug: dawn:1374 Change-Id: I078c78f22c5137030cf3cf0e8358fe4373ee9c6c Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/132268 Reviewed-by: Austin Eng <enga@chromium.org> Kokoro: Kokoro <noreply+kokoro@google.com> Commit-Queue: Loko Kung <lokokung@google.com>
		
			
				
	
	
		
			153 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			153 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2021 The Dawn Authors
 | |
| //
 | |
| // 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.
 | |
| 
 | |
| {% set namespace_name = Name(metadata.native_namespace) %}
 | |
| {% set DIR = namespace_name.concatcase().upper() %}
 | |
| #ifndef {{DIR}}_CHAIN_UTILS_H_
 | |
| #define {{DIR}}_CHAIN_UTILS_H_
 | |
| 
 | |
| {% set impl_dir = metadata.impl_dir + "/" if metadata.impl_dir else "" %}
 | |
| {% set namespace = metadata.namespace %}
 | |
| {% set namespace_name = Name(metadata.native_namespace) %}
 | |
| {% set native_namespace = namespace_name.namespace_case() %}
 | |
| {% set native_dir = impl_dir + namespace_name.Dirs() %}
 | |
| {% set prefix = metadata.proc_table_prefix.lower() %}
 | |
| #include "{{native_dir}}/{{prefix}}_platform.h"
 | |
| #include "{{native_dir}}/Error.h"
 | |
| 
 | |
| namespace {{native_namespace}} {
 | |
| namespace detail {
 | |
|     // Mapping from native types to the expected STypes is implemented as template specializations.
 | |
|     template <typename T>
 | |
|     struct STypeForImpl;
 | |
|     {% for value in types["s type"].values %}
 | |
|         {% if value.valid and value.name.get() in types %}
 | |
|             template <>
 | |
|             struct STypeForImpl<{{as_cppEnum(value.name)}}> {
 | |
|                 static constexpr {{namespace}}::SType value = {{namespace}}::SType::{{as_cppEnum(value.name)}};
 | |
|             };
 | |
|         {% endif %}
 | |
|     {% endfor %}
 | |
|     template <>
 | |
|     struct STypeForImpl<DawnInstanceDescriptor> {
 | |
|         static constexpr {{namespace}}::SType value = {{namespace}}::SType::DawnInstanceDescriptor;
 | |
|     };
 | |
| }  // namespace detail
 | |
| 
 | |
|     template <typename T>
 | |
|     constexpr {{namespace}}::SType STypeFor = detail::STypeForImpl<T>::value;
 | |
|     template <typename T>
 | |
|     void FindInChain(const ChainedStruct* chain, const T** out) {
 | |
|         for (; chain; chain = chain->nextInChain) {
 | |
|             if (chain->sType == STypeFor<T>) {
 | |
|                 *out = static_cast<const T*>(chain);
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     template <typename T>
 | |
|     void FindInChain(ChainedStructOut* chain, T** out) {
 | |
|         for (; chain; chain = chain->nextInChain) {
 | |
|             if (chain->sType == STypeFor<T>) {
 | |
|                 *out = static_cast<T*>(chain);
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Verifies that |chain| only contains ChainedStructs of types enumerated in
 | |
|     // |oneOfConstraints| and contains no duplicate sTypes. Each vector in
 | |
|     // |oneOfConstraints| defines a set of sTypes that cannot coexist in the same chain.
 | |
|     // For example:
 | |
|     //   ValidateSTypes(chain, { { ShaderModuleSPIRVDescriptor, ShaderModuleWGSLDescriptor } }))
 | |
|     //   ValidateSTypes(chain, { { Extension1 }, { Extension2 } })
 | |
|     MaybeError ValidateSTypes(const ChainedStruct* chain,
 | |
|                               std::vector<std::vector<{{namespace}}::SType>> oneOfConstraints);
 | |
|     MaybeError ValidateSTypes(const ChainedStructOut* chain,
 | |
|                               std::vector<std::vector<{{namespace}}::SType>> oneOfConstraints);
 | |
| 
 | |
|     template <typename T>
 | |
|     MaybeError ValidateSingleSTypeInner(const ChainedStruct* chain, T sType) {
 | |
|         DAWN_INVALID_IF(chain->sType != sType,
 | |
|             "Unsupported sType (%s). Expected (%s)", chain->sType, sType);
 | |
|         return {};
 | |
|     }
 | |
|     template <typename T>
 | |
|     MaybeError ValidateSingleSTypeInner(const ChainedStructOut* chain, T sType) {
 | |
|         DAWN_INVALID_IF(chain->sType != sType,
 | |
|             "Unsupported sType (%s). Expected (%s)", chain->sType, sType);
 | |
|         return {};
 | |
|     }
 | |
| 
 | |
|     template <typename T, typename... Args>
 | |
|     MaybeError ValidateSingleSTypeInner(const ChainedStruct* chain, T sType, Args... sTypes) {
 | |
|         if (chain->sType == sType) {
 | |
|             return {};
 | |
|         }
 | |
|         return ValidateSingleSTypeInner(chain, sTypes...);
 | |
|     }
 | |
|     template <typename T, typename... Args>
 | |
|     MaybeError ValidateSingleSTypeInner(const ChainedStructOut* chain, T sType, Args... sTypes) {
 | |
|         if (chain->sType == sType) {
 | |
|             return {};
 | |
|         }
 | |
|         return ValidateSingleSTypeInner(chain, sTypes...);
 | |
|     }
 | |
| 
 | |
|     // Verifies that |chain| contains a single ChainedStruct of type |sType| or no ChainedStructs
 | |
|     // at all.
 | |
|     template <typename T>
 | |
|     MaybeError ValidateSingleSType(const ChainedStruct* chain, T sType) {
 | |
|         if (chain == nullptr) {
 | |
|             return {};
 | |
|         }
 | |
|         DAWN_INVALID_IF(chain->nextInChain != nullptr,
 | |
|             "Chain can only contain a single chained struct.");
 | |
|         return ValidateSingleSTypeInner(chain, sType);
 | |
|     }
 | |
|     template <typename T>
 | |
|     MaybeError ValidateSingleSType(const ChainedStructOut* chain, T sType) {
 | |
|         if (chain == nullptr) {
 | |
|             return {};
 | |
|         }
 | |
|         DAWN_INVALID_IF(chain->nextInChain != nullptr,
 | |
|             "Chain can only contain a single chained struct.");
 | |
|         return ValidateSingleSTypeInner(chain, sType);
 | |
|     }
 | |
| 
 | |
|     // Verifies that |chain| contains a single ChainedStruct with a type enumerated in the
 | |
|     // parameter pack or no ChainedStructs at all.
 | |
|     template <typename T, typename... Args>
 | |
|     MaybeError ValidateSingleSType(const ChainedStruct* chain, T sType, Args... sTypes) {
 | |
|         if (chain == nullptr) {
 | |
|             return {};
 | |
|         }
 | |
|         DAWN_INVALID_IF(chain->nextInChain != nullptr,
 | |
|             "Chain can only contain a single chained struct.");
 | |
|         return ValidateSingleSTypeInner(chain, sType, sTypes...);
 | |
|     }
 | |
|     template <typename T, typename... Args>
 | |
|     MaybeError ValidateSingleSType(const ChainedStructOut* chain, T sType, Args... sTypes) {
 | |
|         if (chain == nullptr) {
 | |
|             return {};
 | |
|         }
 | |
|         DAWN_INVALID_IF(chain->nextInChain != nullptr,
 | |
|             "Chain can only contain a single chained struct.");
 | |
|         return ValidateSingleSTypeInner(chain, sType, sTypes...);
 | |
|     }
 | |
| 
 | |
| }  // namespace {{native_namespace}}
 | |
| 
 | |
| #endif  // {{DIR}}_CHAIN_UTILS_H_
 |