mirror of
				https://github.com/encounter/dawn-cmake.git
				synced 2025-10-25 11:10:29 +00:00 
			
		
		
		
	Bug: dawn:1275 Change-Id: I56535bf4d387c4bd423140705ea70812be073eac Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/79081 Reviewed-by: Corentin Wallez <cwallez@chromium.org>
		
			
				
	
	
		
			122 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //* Copyright 2019 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.
 | |
| 
 | |
| #include "dawn/common/Assert.h"
 | |
| #include "dawn/wire/server/Server.h"
 | |
| 
 | |
| namespace dawn::wire::server {
 | |
|     //* Implementation of the command doers
 | |
|     {% for command in cmd_records["command"] %}
 | |
|         {% set type = command.derived_object %}
 | |
|         {% set method = command.derived_method %}
 | |
|         {% set is_method = method is not none %}
 | |
| 
 | |
|         {% set Suffix = command.name.CamelCase() %}
 | |
|         {% if Suffix not in client_side_commands %}
 | |
|             {% if is_method and Suffix not in server_handwritten_commands %}
 | |
|                 bool Server::Do{{Suffix}}(
 | |
|                     {%- for member in command.members -%}
 | |
|                         {%- if member.is_return_value -%}
 | |
|                             {%- if member.handle_type -%}
 | |
|                                 {{as_cType(member.handle_type.name)}}* {{as_varName(member.name)}}
 | |
|                             {%- else -%}
 | |
|                                 {{as_cType(member.type.name)}}* {{as_varName(member.name)}}
 | |
|                             {%- endif -%}
 | |
|                         {%- else -%}
 | |
|                             {{as_annotated_cType(member)}}
 | |
|                         {%- endif -%}
 | |
|                         {%- if not loop.last -%}, {% endif %}
 | |
|                     {%- endfor -%}
 | |
|                 ) {
 | |
|                     {% set ret = command.members|selectattr("is_return_value")|list %}
 | |
|                     //* If there is a return value, assign it.
 | |
|                     {% if ret|length == 1 %}
 | |
|                         *{{as_varName(ret[0].name)}} =
 | |
|                     {% else %}
 | |
|                         //* Only one member should be a return value.
 | |
|                         {{ assert(ret|length == 0) }}
 | |
|                     {% endif %}
 | |
|                     mProcs.{{as_varName(type.name, method.name)}}(
 | |
|                         {%- for member in command.members if not member.is_return_value -%}
 | |
|                             {{as_varName(member.name)}}
 | |
|                             {%- if not loop.last -%}, {% endif %}
 | |
|                         {%- endfor -%}
 | |
|                     );
 | |
|                     {% if ret|length == 1 %}
 | |
|                         //* WebGPU error handling guarantees that no null object can be returned by
 | |
|                         //* object creation functions.
 | |
|                         ASSERT(*{{as_varName(ret[0].name)}} != nullptr);
 | |
|                     {% endif %}
 | |
|                     return true;
 | |
|                 }
 | |
|             {% endif %}
 | |
|         {% endif %}
 | |
|     {% endfor %}
 | |
| 
 | |
|     bool Server::DoDestroyObject(ObjectType objectType, ObjectId objectId) {
 | |
|         //* ID 0 are reserved for nullptr and cannot be destroyed.
 | |
|         if (objectId == 0) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         switch(objectType) {
 | |
|             {% for type in by_category["object"] %}
 | |
|                 case ObjectType::{{type.name.CamelCase()}}: {
 | |
|                     auto* data = {{type.name.CamelCase()}}Objects().Get(objectId);
 | |
|                     if (data == nullptr) {
 | |
|                         return false;
 | |
|                     }
 | |
|                     if (data->deviceInfo != nullptr) {
 | |
|                         if (!UntrackDeviceChild(data->deviceInfo, objectType, objectId)) {
 | |
|                             return false;
 | |
|                         }
 | |
|                     }
 | |
|                     if (data->state == AllocationState::Allocated) {
 | |
|                         ASSERT(data->handle != nullptr);
 | |
|                         {% if type.name.CamelCase() in server_reverse_lookup_objects %}
 | |
|                             {{type.name.CamelCase()}}ObjectIdTable().Remove(data->handle);
 | |
|                         {% endif %}
 | |
| 
 | |
|                         {% if type.name.get() == "device" %}
 | |
|                             //* TODO(crbug.com/dawn/384): This is a hack to make sure that all child objects
 | |
|                             //* are destroyed before their device. We should have a solution in
 | |
|                             //* Dawn native that makes all child objects internally null if their
 | |
|                             //* Device is destroyed.
 | |
|                             while (data->info->childObjectTypesAndIds.size() > 0) {
 | |
|                                 auto [childObjectType, childObjectId] = UnpackObjectTypeAndId(
 | |
|                                     *data->info->childObjectTypesAndIds.begin());
 | |
|                                 if (!DoDestroyObject(childObjectType, childObjectId)) {
 | |
|                                     return false;
 | |
|                                 }
 | |
|                             }
 | |
|                             if (data->handle != nullptr) {
 | |
|                                 //* Deregisters uncaptured error and device lost callbacks since
 | |
|                                 //* they should not be forwarded if the device no longer exists on the wire.
 | |
|                                 ClearDeviceCallbacks(data->handle);
 | |
|                             }
 | |
|                         {% endif %}
 | |
| 
 | |
|                         mProcs.{{as_varName(type.name, Name("release"))}}(data->handle);
 | |
|                     }
 | |
|                     {{type.name.CamelCase()}}Objects().Free(objectId);
 | |
|                     return true;
 | |
|                 }
 | |
|             {% endfor %}
 | |
|             default:
 | |
|                 return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
| }  // namespace dawn::wire::server
 |