mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-10-24 02:35:52 +00:00
This is a temporary fix until dawn_wire can support multiple devices. When using objects from different devices, the wire will inject an error into the receiver object's device. Methods that return objects will return a dummy object. Using the dummy object will cause a fatal error on the server. Without this fix, the server would blindly lookup an ObjectId which could point to some other object. This would bypass same-device validation and have incorrect results. Bug: dawn:383 Change-Id: I898f07d4b26f2a97ef952b82af488e7f807c36f0 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/19261 Commit-Queue: Austin Eng <enga@chromium.org> Reviewed-by: Kai Ninomiya <kainino@chromium.org>
315 lines
14 KiB
C++
315 lines
14 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 "common/Log.h"
|
|
#include "dawn_wire/client/ApiObjects.h"
|
|
#include "dawn_wire/client/ApiProcs_autogen.h"
|
|
#include "dawn_wire/client/Client.h"
|
|
|
|
#include <algorithm>
|
|
#include <cstring>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
namespace dawn_wire { namespace client {
|
|
namespace {
|
|
|
|
//* Outputs an rvalue that's the number of elements a pointer member points to.
|
|
{% macro member_length(member, accessor) -%}
|
|
{%- if member.length == "constant" -%}
|
|
{{member.constant_length}}
|
|
{%- else -%}
|
|
{{accessor}}{{as_varName(member.length.name)}}
|
|
{%- endif -%}
|
|
{%- endmacro %}
|
|
|
|
{% for type in by_category["object"] %}
|
|
DAWN_DECLARE_UNUSED bool DeviceMatches(const Device* device, const {{as_cType(type.name)}} obj) {
|
|
return device == reinterpret_cast<const {{as_wireType(type)}}>(obj)->device;
|
|
}
|
|
|
|
DAWN_DECLARE_UNUSED bool DeviceMatches(const Device* device, const {{as_cType(type.name)}} *const obj, uint32_t count = 1) {
|
|
ASSERT(count == 0 || obj != nullptr);
|
|
for (uint32_t i = 0; i < count; ++i) {
|
|
if (!DeviceMatches(device, obj[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
{% endfor %}
|
|
|
|
bool DeviceMatches(const Device* device, WGPUChainedStruct const* chainedStruct);
|
|
|
|
{% for type in by_category["structure"] if type.may_have_dawn_object %}
|
|
DAWN_DECLARE_UNUSED bool DeviceMatches(const Device* device, const {{as_cType(type.name)}}& obj) {
|
|
{% if type.extensible %}
|
|
if (!DeviceMatches(device, obj.nextInChain)) {
|
|
return false;
|
|
}
|
|
{% endif %}
|
|
{% for member in type.members if member.type.may_have_dawn_object or member.type.category == "object" %}
|
|
{% if member.optional %}
|
|
if (obj.{{as_varName(member.name)}} != nullptr)
|
|
{% endif %}
|
|
{
|
|
if (!DeviceMatches(device, obj.{{as_varName(member.name)}}
|
|
{%- if member.annotation != "value" and member.length != "strlen" -%}
|
|
, {{member_length(member, "obj.")}}
|
|
{%- endif -%})) {
|
|
return false;
|
|
}
|
|
}
|
|
{% endfor %}
|
|
return true;
|
|
}
|
|
|
|
DAWN_DECLARE_UNUSED bool DeviceMatches(const Device* device, const {{as_cType(type.name)}} *const obj, uint32_t count = 1) {
|
|
for (uint32_t i = 0; i < count; ++i) {
|
|
if (!DeviceMatches(device, obj[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
{% endfor %}
|
|
|
|
bool DeviceMatches(const Device* device, WGPUChainedStruct const* chainedStruct) {
|
|
while (chainedStruct != nullptr) {
|
|
switch (chainedStruct->sType) {
|
|
{% for sType in types["s type"].values if sType.valid %}
|
|
{% set CType = as_cType(sType.name) %}
|
|
case {{as_cEnum(types["s type"].name, sType.name)}}: {
|
|
{% if types[sType.name.get()].may_have_dawn_object %}
|
|
if (!DeviceMatches(device, reinterpret_cast<const {{CType}}*>(chainedStruct))) {
|
|
return false;
|
|
}
|
|
{% endif %}
|
|
break;
|
|
}
|
|
{% endfor %}
|
|
case WGPUSType_Invalid:
|
|
break;
|
|
default:
|
|
UNREACHABLE();
|
|
dawn::WarningLog()
|
|
<< "All objects may not be from the same device. "
|
|
<< "Unknown sType " << chainedStruct->sType << " discarded.";
|
|
return false;
|
|
}
|
|
chainedStruct = chainedStruct->next;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
//* Implementation of the client API functions.
|
|
{% for type in by_category["object"] %}
|
|
{% set Type = type.name.CamelCase() %}
|
|
{% set cType = as_cType(type.name) %}
|
|
|
|
{% for method in type.methods %}
|
|
{% set Suffix = as_MethodSuffix(type.name, method.name) %}
|
|
|
|
{% if Suffix in client_handwritten_commands %}
|
|
static
|
|
{% endif %}
|
|
{{as_cType(method.return_type.name)}} Client{{Suffix}}(
|
|
{{-cType}} cSelf
|
|
{%- for arg in method.arguments -%}
|
|
, {{as_annotated_cType(arg)}}
|
|
{%- endfor -%}
|
|
) {
|
|
{% if len(method.arguments) > 0 %}
|
|
{
|
|
bool sameDevice = true;
|
|
auto self = reinterpret_cast<{{as_wireType(type)}}>(cSelf);
|
|
Device* device = self->device;
|
|
DAWN_UNUSED(device);
|
|
|
|
do {
|
|
{% for arg in method.arguments if arg.type.may_have_dawn_object or arg.type.category == "object" %}
|
|
{% if arg.optional %}
|
|
if ({{as_varName(arg.name)}} != nullptr)
|
|
{% endif %}
|
|
{
|
|
if (!DeviceMatches(device, {{as_varName(arg.name)}}
|
|
{%- if arg.annotation != "value" and arg.length != "strlen" -%}
|
|
, {{member_length(arg, "")}}
|
|
{%- endif -%})) {
|
|
sameDevice = false;
|
|
break;
|
|
}
|
|
}
|
|
{% endfor %}
|
|
} while (false);
|
|
|
|
if (DAWN_UNLIKELY(!sameDevice)) {
|
|
ClientDeviceInjectError(reinterpret_cast<WGPUDevice>(device),
|
|
WGPUErrorType_Validation,
|
|
"All objects must be from the same device.");
|
|
{% if method.return_type.category == "object" %}
|
|
// Allocate an object without registering it on the server. This is backed by a real allocation on
|
|
// the client so commands can be sent with it. But because it's not allocated on the server, it will
|
|
// be a fatal error to use it.
|
|
auto self = reinterpret_cast<{{as_wireType(type)}}>(cSelf);
|
|
auto* allocation = self->device->GetClient()->{{method.return_type.name.CamelCase()}}Allocator().New(self->device);
|
|
return reinterpret_cast<{{as_cType(method.return_type.name)}}>(allocation->object.get());
|
|
{% elif method.return_type.name.canonical_case() == "void" %}
|
|
return;
|
|
{% else %}
|
|
return {};
|
|
{% endif %}
|
|
}
|
|
}
|
|
{% endif %}
|
|
|
|
{% if Suffix not in client_handwritten_commands %}
|
|
auto self = reinterpret_cast<{{as_wireType(type)}}>(cSelf);
|
|
Device* device = self->device;
|
|
{{Suffix}}Cmd cmd;
|
|
|
|
//* Create the structure going on the wire on the stack and fill it with the value
|
|
//* arguments so it can compute its size.
|
|
cmd.self = cSelf;
|
|
|
|
//* For object creation, store the object ID the client will use for the result.
|
|
{% if method.return_type.category == "object" %}
|
|
auto* allocation = self->device->GetClient()->{{method.return_type.name.CamelCase()}}Allocator().New(self->device);
|
|
cmd.result = ObjectHandle{allocation->object->id, allocation->generation};
|
|
{% endif %}
|
|
|
|
{% for arg in method.arguments %}
|
|
cmd.{{as_varName(arg.name)}} = {{as_varName(arg.name)}};
|
|
{% endfor %}
|
|
|
|
//* Allocate space to send the command and copy the value args over.
|
|
size_t requiredSize = cmd.GetRequiredSize();
|
|
char* allocatedBuffer = static_cast<char*>(device->GetClient()->GetCmdSpace(requiredSize));
|
|
cmd.Serialize(allocatedBuffer, *device->GetClient());
|
|
|
|
{% if method.return_type.category == "object" %}
|
|
return reinterpret_cast<{{as_cType(method.return_type.name)}}>(allocation->object.get());
|
|
{% endif %}
|
|
{% else %}
|
|
return ClientHandwritten{{Suffix}}(cSelf
|
|
{%- for arg in method.arguments -%}
|
|
, {{as_varName(arg.name)}}
|
|
{%- endfor -%});
|
|
{% endif %}
|
|
}
|
|
{% endfor %}
|
|
|
|
{% if not type.name.canonical_case() == "device" %}
|
|
//* When an object's refcount reaches 0, notify the server side of it and delete it.
|
|
void Client{{as_MethodSuffix(type.name, Name("release"))}}({{cType}} cObj) {
|
|
{{Type}}* obj = reinterpret_cast<{{Type}}*>(cObj);
|
|
obj->refcount --;
|
|
|
|
if (obj->refcount > 0) {
|
|
return;
|
|
}
|
|
|
|
DestroyObjectCmd cmd;
|
|
cmd.objectType = ObjectType::{{type.name.CamelCase()}};
|
|
cmd.objectId = obj->id;
|
|
|
|
size_t requiredSize = cmd.GetRequiredSize();
|
|
char* allocatedBuffer = static_cast<char*>(obj->device->GetClient()->GetCmdSpace(requiredSize));
|
|
cmd.Serialize(allocatedBuffer);
|
|
|
|
obj->device->GetClient()->{{type.name.CamelCase()}}Allocator().Free(obj);
|
|
}
|
|
|
|
void Client{{as_MethodSuffix(type.name, Name("reference"))}}({{cType}} cObj) {
|
|
{{Type}}* obj = reinterpret_cast<{{Type}}*>(cObj);
|
|
obj->refcount ++;
|
|
}
|
|
{% endif %}
|
|
{% endfor %}
|
|
|
|
namespace {
|
|
WGPUInstance ClientCreateInstance(WGPUInstanceDescriptor const* descriptor) {
|
|
UNREACHABLE();
|
|
return nullptr;
|
|
}
|
|
|
|
struct ProcEntry {
|
|
WGPUProc proc;
|
|
const char* name;
|
|
};
|
|
static const ProcEntry sProcMap[] = {
|
|
{% for (type, method) in c_methods_sorted_by_name %}
|
|
{ reinterpret_cast<WGPUProc>(Client{{as_MethodSuffix(type.name, method.name)}}), "{{as_cMethod(type.name, method.name)}}" },
|
|
{% endfor %}
|
|
};
|
|
static constexpr size_t sProcMapSize = sizeof(sProcMap) / sizeof(sProcMap[0]);
|
|
} // anonymous namespace
|
|
|
|
WGPUProc ClientGetProcAddress(WGPUDevice, const char* procName) {
|
|
if (procName == nullptr) {
|
|
return nullptr;
|
|
}
|
|
|
|
const ProcEntry* entry = std::lower_bound(&sProcMap[0], &sProcMap[sProcMapSize], procName,
|
|
[](const ProcEntry &a, const char *b) -> bool {
|
|
return strcmp(a.name, b) < 0;
|
|
}
|
|
);
|
|
|
|
if (entry != &sProcMap[sProcMapSize] && strcmp(entry->name, procName) == 0) {
|
|
return entry->proc;
|
|
}
|
|
|
|
// Special case the two free-standing functions of the API.
|
|
if (strcmp(procName, "wgpuGetProcAddress") == 0) {
|
|
return reinterpret_cast<WGPUProc>(ClientGetProcAddress);
|
|
}
|
|
|
|
if (strcmp(procName, "wgpuCreateInstance") == 0) {
|
|
return reinterpret_cast<WGPUProc>(ClientCreateInstance);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
std::vector<const char*> GetProcMapNamesForTesting() {
|
|
std::vector<const char*> result;
|
|
result.reserve(sProcMapSize);
|
|
for (const ProcEntry& entry : sProcMap) {
|
|
result.push_back(entry.name);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
//* Some commands don't have a custom wire format, but need to be handled manually to update
|
|
//* some client-side state tracking. For these we have two functions:
|
|
//* - An autogenerated Client{{suffix}} method that sends the command on the wire
|
|
//* - A manual ProxyClient{{suffix}} method that will be inserted in the proctable instead of
|
|
//* the autogenerated one, and that will have to call Client{{suffix}}
|
|
DawnProcTable GetProcs() {
|
|
DawnProcTable table;
|
|
table.getProcAddress = ClientGetProcAddress;
|
|
table.createInstance = ClientCreateInstance;
|
|
{% for type in by_category["object"] %}
|
|
{% for method in c_methods(type) %}
|
|
{% set suffix = as_MethodSuffix(type.name, method.name) %}
|
|
table.{{as_varName(type.name, method.name)}} = Client{{suffix}};
|
|
{% endfor %}
|
|
{% endfor %}
|
|
return table;
|
|
}
|
|
}} // namespace dawn_wire::client
|