| //* Copyright 2017 The Dawn & Tint Authors | 
 | //* | 
 | //* Redistribution and use in source and binary forms, with or without | 
 | //* modification, are permitted provided that the following conditions are met: | 
 | //* | 
 | //* 1. Redistributions of source code must retain the above copyright notice, this | 
 | //*    list of conditions and the following disclaimer. | 
 | //* | 
 | //* 2. Redistributions in binary form must reproduce the above copyright notice, | 
 | //*    this list of conditions and the following disclaimer in the documentation | 
 | //*    and/or other materials provided with the distribution. | 
 | //* | 
 | //* 3. Neither the name of the copyright holder nor the names of its | 
 | //*    contributors may be used to endorse or promote products derived from | 
 | //*    this software without specific prior written permission. | 
 | //* | 
 | //* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
 | //* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
 | //* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
 | //* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
 | //* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
 | //* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
 | //* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
 | //* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
 | //* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
 | //* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
 |  | 
 | {% set api = metadata.api.lower() %} | 
 | #include "dawn/common/Log.h" | 
 | #include "mock_{{api}}.h" | 
 |  | 
 | using namespace testing; | 
 |  | 
 | namespace { | 
 |     {% for type in by_category["object"] %} | 
 |         {% for method in c_methods(type) %} | 
 |             {{as_cType(method.return_type.name)}} Forward{{as_MethodSuffix(type.name, method.name)}}( | 
 |                 {{-as_cType(type.name)}} self | 
 |                 {%- for arg in method.arguments -%} | 
 |                     , {{as_annotated_cType(arg)}} | 
 |                 {%- endfor -%} | 
 |             ) { | 
 |                 auto object = reinterpret_cast<ProcTableAsClass::Object*>(self); | 
 |                 return object->procs->{{as_CppMethodSuffix(type.name, method.name)}}(self | 
 |                     {%- for arg in method.arguments -%} | 
 |                         , {{as_varName(arg.name)}} | 
 |                     {%- endfor -%} | 
 |                 ); | 
 |             } | 
 |         {% endfor %} | 
 |  | 
 |     {% endfor %} | 
 | } | 
 |  | 
 | ProcTableAsClass::~ProcTableAsClass() { | 
 | } | 
 |  | 
 | {% set Prefix = metadata.proc_table_prefix %} | 
 | void ProcTableAsClass::GetProcTable({{Prefix}}ProcTable* table) { | 
 |     {% for type in by_category["object"] %} | 
 |         {% for method in c_methods(type) %} | 
 |             table->{{as_varName(type.name, method.name)}} = reinterpret_cast<{{as_cProc(type.name, method.name)}}>(Forward{{as_MethodSuffix(type.name, method.name)}}); | 
 |         {% endfor %} | 
 |     {% endfor %} | 
 |  | 
 |     {% for type in by_category["structure"] if type.has_free_members_function %} | 
 |         table->{{as_varName(type.name, Name("free members"))}} = []({{as_cType(type.name)}} {{as_varName(type.name)}}) { | 
 |             static bool calledOnce = false; | 
 |             if (!calledOnce) { | 
 |                 calledOnce = true; | 
 |                 dawn::WarningLog() << "No mock available for {{as_varName(type.name, Name('free members'))}}"; | 
 |             } | 
 |         }; | 
 |     {% endfor %} | 
 | } | 
 |  | 
 | //* Generate the older Call*Callback if there is no Future call equivalent. | 
 | //* Includes: | 
 | //*   - setLoggingCallback | 
 | {% set LegacyCallbackFunctions = ['set logging callback'] %} | 
 |  | 
 | //* Manually implemented mock functions due to incompatibility. | 
 | {% set ManuallyMockedFunctions = ['set device lost callback', 'set uncaptured error callback'] %} | 
 |  | 
 | {% for type in by_category["object"] %} | 
 |     {% for method in type.methods if method.name.get() not in ManuallyMockedFunctions %} | 
 |         {% set Suffix = as_CppMethodSuffix(type.name, method.name) %} | 
 |         {% if has_callback_arguments(method) %} | 
 |             {{as_cType(method.return_type.name)}} ProcTableAsClass::{{Suffix}}( | 
 |                 {{-as_cType(type.name)}} {{as_varName(type.name)}} | 
 |                 {%- for arg in method.arguments -%} | 
 |                     , {{as_annotated_cType(arg)}} | 
 |                 {%- endfor -%} | 
 |             ) { | 
 |                 ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>({{as_varName(type.name)}}); | 
 |                 {% for arg in method.arguments if arg.type.category == 'function pointer' %} | 
 |                     object->m{{Suffix + arg.name.CamelCase()}} = {{as_varName(arg.name)}}; | 
 |                     object->m{{Suffix}}Userdata = userdata; | 
 |                 {% endfor %} | 
 |  | 
 |                 {% if method.name.get() == 'pop error scope' %} | 
 |                     //* Currently special casing popErrorScope since it has an old callback type. | 
 |                     On{{Suffix}}({{-as_varName(type.name)}}, {.nextInChain = nullptr, .mode = WGPUCallbackMode_AllowProcessEvents, .callback = nullptr, .oldCallback = oldCallback, .userdata = userdata}); | 
 |                 {% elif method.name.get() not in LegacyCallbackFunctions %} | 
 |                     On{{Suffix}}( | 
 |                         {{-as_varName(type.name)}} | 
 |                         {%- for arg in method.arguments if arg.type.category != 'function pointer' and arg.type.name.get() != 'void *' -%} | 
 |                             , {{as_varName(arg.name)}} | 
 |                         {%- endfor -%} | 
 |                         , {.nextInChain = nullptr, .mode = WGPUCallbackMode_AllowProcessEvents | 
 |                         {%- for arg in method.arguments if arg.type.category == 'function pointer' -%} | 
 |                             , .{{as_varName(arg.name)}} = {{as_varName(arg.name)}} | 
 |                         {%- endfor -%} | 
 |                         , .userdata = userdata}); | 
 |                 {% else %} | 
 |                     On{{Suffix}}( | 
 |                         {{-as_varName(type.name)}} | 
 |                         {%- for arg in method.arguments -%} | 
 |                             , {{as_varName(arg.name)}} | 
 |                         {%- endfor -%} | 
 |                     ); | 
 |                 {% endif %} | 
 |             } | 
 |         {% elif has_callback_info(method) %} | 
 |             {{as_cType(method.return_type.name)}} ProcTableAsClass::{{Suffix}}( | 
 |                 {{-as_cType(type.name)}} {{as_varName(type.name)}} | 
 |                 {%- for arg in method.arguments -%} | 
 |                     , {{as_annotated_cType(arg)}} | 
 |                 {%- endfor -%} | 
 |             ) { | 
 |                 ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>({{as_varName(type.name)}}); | 
 |                 {% for arg in method.arguments %} | 
 |                     {% if arg.name.get() == 'callback info' %} | 
 |                         {% for callback in types[arg.type.name.get()].members if callback.type.category == 'function pointer' %} | 
 |                             object->m{{Suffix + callback.name.CamelCase()}} = {{as_varName(arg.name)}}.{{as_varName(callback.name)}}; | 
 |                             object->m{{Suffix}}Userdata = {{as_varName(arg.name)}}.userdata; | 
 |                         {% endfor %} | 
 |                     {% endif %} | 
 |                 {% endfor %} | 
 |  | 
 |                 On{{Suffix}}( | 
 |                     {{-as_varName(type.name)}} | 
 |                     {%- for arg in method.arguments -%} | 
 |                         , {{as_varName(arg.name)}} | 
 |                     {%- endfor -%} | 
 |                 ); | 
 |                 return {mNextFutureID++}; | 
 |             } | 
 |         {% elif has_callbackInfoStruct(method) %} | 
 |             {{as_cType(method.return_type.name)}} ProcTableAsClass::{{Suffix}}( | 
 |                 {{-as_cType(type.name)}} {{as_varName(type.name)}} | 
 |                 {%- for arg in method.arguments -%} | 
 |                     , {{as_annotated_cType(arg)}} | 
 |                 {%- endfor -%} | 
 |             ) { | 
 |                 ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>({{as_varName(type.name)}}); | 
 |                 object->m{{Suffix}}Callback = callbackInfo.callback; | 
 |                 object->m{{Suffix}}Userdata1 = callbackInfo.userdata1; | 
 |                 object->m{{Suffix}}Userdata2 = callbackInfo.userdata2; | 
 |  | 
 |                 On{{Suffix}}( | 
 |                     {{-as_varName(type.name)}} | 
 |                     {%- for arg in method.arguments -%} | 
 |                         , {{as_varName(arg.name)}} | 
 |                     {%- endfor -%} | 
 |                 ); | 
 |                 return {mNextFutureID++}; | 
 |             } | 
 |             {% set CallbackInfoType = (method.arguments|last).type %} | 
 |             {% set CallbackType = find_by_name(CallbackInfoType.members, "callback").type %} | 
 |             void ProcTableAsClass::Call{{Suffix}}Callback( | 
 |                 {{-as_cType(type.name)}} {{as_varName(type.name)}} | 
 |                 {%- for arg in CallbackType.arguments -%} | 
 |                     , {{as_annotated_cType(arg)}} | 
 |                 {%- endfor -%} | 
 |             ) { | 
 |                 ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>({{as_varName(type.name)}}); | 
 |                 object->m{{Suffix}}Callback( | 
 |                     {%- for arg in CallbackType.arguments -%} | 
 |                         {{as_varName(arg.name)}}{{", "}} | 
 |                     {%- endfor -%} | 
 |                     object->m{{Suffix}}Userdata1, object->m{{Suffix}}Userdata2); | 
 |             } | 
 |         {% endif %} | 
 |     {% endfor %} | 
 |  | 
 |     {% for method in type.methods if has_callback_info(method) or method.name.get() in LegacyCallbackFunctions %} | 
 |         {% set Suffix = as_CppMethodSuffix(type.name, method.name) %} | 
 |         {% for arg in method.arguments %} | 
 |             {% if arg.name.get() == 'callback info' %} | 
 |                 {% for callback in types[arg.type.name.get()].members if callback.type.category == 'function pointer' %} | 
 |                     void ProcTableAsClass::Call{{Suffix + callback.name.CamelCase()}}( | 
 |                         {{-as_cType(type.name)}} {{as_varName(type.name)}} | 
 |                         {%- for arg in callback.type.arguments -%} | 
 |                             {%- if not loop.last -%}, {{as_annotated_cType(arg)}}{%- endif -%} | 
 |                         {%- endfor -%} | 
 |                     ) { | 
 |                         ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>({{as_varName(type.name)}}); | 
 |                         object->m{{Suffix + callback.name.CamelCase()}}( | 
 |                             {%- for arg in callback.type.arguments -%} | 
 |                                 {%- if not loop.last -%}{{as_varName(arg.name)}}, {% endif -%} | 
 |                             {%- endfor -%} | 
 |                             object->m{{Suffix}}Userdata); | 
 |                     } | 
 |                 {% endfor %} | 
 |             {% elif arg.type.category == 'function pointer' %} | 
 |                 void ProcTableAsClass::Call{{Suffix + arg.name.CamelCase()}}( | 
 |                     {{-as_cType(type.name)}} {{as_varName(type.name)}} | 
 |                     {%- for arg in arg.type.arguments -%} | 
 |                         {%- if not loop.last -%}, {{as_annotated_cType(arg)}}{%- endif -%} | 
 |                     {%- endfor -%} | 
 |                 ) { | 
 |                     ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>({{as_varName(type.name)}}); | 
 |                     object->m{{Suffix + arg.name.CamelCase()}}( | 
 |                         {%- for arg in arg.type.arguments -%} | 
 |                             {%- if not loop.last -%}{{as_varName(arg.name)}}, {% endif -%} | 
 |                         {%- endfor -%} | 
 |                         object->m{{Suffix}}Userdata); | 
 |                 } | 
 |             {% endif %} | 
 |         {% endfor %} | 
 |     {% endfor %} | 
 | {% endfor %} | 
 |  | 
 | // Manually implement some callback helpers for testing. | 
 | void ProcTableAsClass::DeviceSetDeviceLostCallback(WGPUDevice device, | 
 |                                                    WGPUDeviceLostCallback callback, | 
 |                                                    void* userdata) { | 
 |     ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>(device); | 
 |     object->mDeviceLostCallback = [](WGPUDevice const*, WGPUDeviceLostReason reason, | 
 |                                      WGPUStringView message, void* callback, void* userdata) { | 
 |         if (callback == nullptr) { | 
 |             return; | 
 |         } | 
 |         auto cb = reinterpret_cast<WGPUDeviceLostCallback>(callback); | 
 |         cb(reason, message, userdata); | 
 |     }; | 
 |     object->mDeviceLostUserdata1 = reinterpret_cast<void*>(callback); | 
 |     object->mDeviceLostUserdata2 = userdata; | 
 |  | 
 |     OnDeviceSetDeviceLostCallback(device, callback, userdata); | 
 | } | 
 | void ProcTableAsClass::CallDeviceSetDeviceLostCallbackCallback(WGPUDevice device, | 
 |                                                                WGPUDeviceLostReason reason, | 
 |                                                                WGPUStringView message) { | 
 |     ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>(device); | 
 |     object->mDeviceLostCallback(&device, reason, message, object->mDeviceLostUserdata1, | 
 |                                 object->mDeviceLostUserdata2); | 
 | } | 
 | void ProcTableAsClass::DeviceSetUncapturedErrorCallback(WGPUDevice device, | 
 |                                                         WGPUErrorCallback callback, | 
 |                                                         void* userdata) { | 
 |     ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>(device); | 
 |     object->mUncapturedErrorCallback = [](WGPUDevice const*, WGPUErrorType type, | 
 |                                           WGPUStringView message, void* callback, void* userdata) { | 
 |         if (callback == nullptr) { | 
 |             return; | 
 |         } | 
 |         auto cb = reinterpret_cast<WGPUErrorCallback>(callback); | 
 |         cb(type, message, userdata); | 
 |     }; | 
 |     object->mUncapturedErrorUserdata1 = reinterpret_cast<void*>(callback); | 
 |     object->mUncapturedErrorUserdata2 = userdata; | 
 |  | 
 |     OnDeviceSetUncapturedErrorCallback(device, callback, userdata); | 
 | } | 
 | void ProcTableAsClass::CallDeviceSetUncapturedErrorCallbackCallback(WGPUDevice device, | 
 |                                                                     WGPUErrorType type, | 
 |                                                                     WGPUStringView message) { | 
 |     ProcTableAsClass::Object* object = reinterpret_cast<ProcTableAsClass::Object*>(device); | 
 |     object->mUncapturedErrorCallback(&device, type, message, object->mUncapturedErrorUserdata1, | 
 |                                      object->mUncapturedErrorUserdata2); | 
 | } | 
 |  | 
 | {% for type in by_category["object"] %} | 
 |     {{as_cType(type.name)}} ProcTableAsClass::GetNew{{type.name.CamelCase()}}() { | 
 |         mObjects.emplace_back(new Object); | 
 |         mObjects.back()->procs = this; | 
 |         return reinterpret_cast<{{as_cType(type.name)}}>(mObjects.back().get()); | 
 |     } | 
 | {% endfor %} | 
 |  | 
 | MockProcTable::MockProcTable() = default; | 
 |  | 
 | MockProcTable::~MockProcTable() = default; | 
 |  | 
 | void MockProcTable::IgnoreAllReleaseCalls() { | 
 |     {% for type in by_category["object"] %} | 
 |         EXPECT_CALL(*this, {{as_CppMethodSuffix(type.name, Name("release"))}}(_)).Times(AnyNumber()); | 
 |     {% endfor %} | 
 | } |