//* 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/Assert.h" #include "dawn_wire/server/Server.h" namespace dawn_wire { namespace server { {% for command in cmd_records["command"] %} {% set type = command.derived_object %} {% set method = command.derived_method %} {% set is_method = method != None %} {% set returns = is_method and method.return_type.name.canonical_case() != "void" %} {% set Suffix = command.name.CamelCase() %} {% if Suffix not in client_side_commands %} //* The generic command handlers bool Server::Handle{{Suffix}}(const char** commands, size_t* size) { {{Suffix}}Cmd cmd; DeserializeResult deserializeResult = cmd.Deserialize(commands, size, &mAllocator {%- if command.has_dawn_object -%} , *this {%- endif -%} ); if (deserializeResult == DeserializeResult::FatalError) { return false; } {% if Suffix in server_custom_pre_handler_commands %} if (!PreHandle{{Suffix}}(cmd)) { return false; } {% endif %} {% if is_method %} //* Unpack 'self' auto* selfData = {{type.name.CamelCase()}}Objects().Get(cmd.selfId); ASSERT(selfData != nullptr); {% endif %} //* Allocate any result objects {%- for member in command.members if member.is_return_value -%} {{ assert(member.handle_type) }} {% set Type = member.handle_type.name.CamelCase() %} {% set name = as_varName(member.name) %} auto* {{name}}Data = {{Type}}Objects().Allocate(cmd.{{name}}.id); if ({{name}}Data == nullptr) { return false; } {{name}}Data->serial = cmd.{{name}}.serial; {% if type.is_builder %} selfData->builtObject = cmd.{{name}}; {% endif %} {% endfor %} //* After the data is allocated, apply the argument error propagation mechanism if (deserializeResult == DeserializeResult::ErrorObject) { {% if type.is_builder %} selfData->valid = false; //* If we are in GetResult, fake an error callback {% if returns %} On{{type.name.CamelCase()}}Error(DAWN_BUILDER_ERROR_STATUS_ERROR, "Maybe monad", cmd.selfId, selfData->serial); {% endif %} {% endif %} return true; } //* Do command bool success = Do{{Suffix}}( {%- for member in command.members -%} {%- if member.is_return_value -%} {%- if member.handle_type -%} &{{as_varName(member.name)}}Data->handle //* Pass the handle of the output object to be written by the doer {%- else -%} &cmd.{{as_varName(member.name)}} {%- endif -%} {%- else -%} cmd.{{as_varName(member.name)}} {%- endif -%} {%- if not loop.last -%}, {% endif %} {%- endfor -%} ); //* Mark output object handles as valid/invalid {%- for member in command.members if member.is_return_value and member.handle_type -%} {% set name = as_varName(member.name) %} {{name}}Data->valid = {{name}}Data->handle != nullptr; {% endfor %} if (!success) { return false; } {%- for member in command.members if member.is_return_value and member.handle_type -%} {% set Type = member.handle_type.name.CamelCase() %} {% set name = as_varName(member.name) %} {% if Type in server_reverse_lookup_objects %} //* For created objects, store a mapping from them back to their client IDs if ({{name}}Data->valid) { {{Type}}ObjectIdTable().Store({{name}}Data->handle, cmd.{{name}}.id); } {% endif %} //* builders remember the ID of the object they built so that they can send it //* in the callback to the client. {% if member.handle_type.is_builder %} if ({{name}}Data->valid) { uint64_t userdata1 = static_cast(reinterpret_cast(this)); uint64_t userdata2 = (uint64_t({{name}}Data->serial) << uint64_t(32)) + cmd.{{name}}.id; mProcs.{{as_varName(member.handle_type.name, Name("set error callback"))}}({{name}}Data->handle, Forward{{Type}}, userdata1, userdata2); } {% endif %} {% endfor %} return true; } {% endif %} {% endfor %} const char* Server::HandleCommands(const char* commands, size_t size) { mProcs.deviceTick(DeviceObjects().Get(1)->handle); while (size >= sizeof(WireCmd)) { WireCmd cmdId = *reinterpret_cast(commands); bool success = false; switch (cmdId) { {% for command in cmd_records["command"] %} case WireCmd::{{command.name.CamelCase()}}: success = Handle{{command.name.CamelCase()}}(&commands, &size); break; {% endfor %} default: success = false; } if (!success) { return nullptr; } mAllocator.Reset(); } if (size != 0) { return nullptr; } return commands; } }} // namespace dawn_wire::server