mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-05-14 11:21:40 +00:00
This directory used to contain both the state tracking code for the backends, and the common utilities that could be used both by the backends and the rest of the code. Things are now: - src/common is utility code for the whole repo - src/backend contains libNXT's code - src/utils is utility code that we don't want in libNXT This commit also changes all includes to use global paths from src/ bacause it had to touch a bunch of #include statements anyway.
204 lines
9.1 KiB
C++
204 lines
9.1 KiB
C++
//* Copyright 2017 The NXT 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 "nxt/nxt.h"
|
|
#include "nxt/nxtcpp.h"
|
|
|
|
#include "backend/{{namespace}}/GeneratedCodeIncludes.h"
|
|
|
|
namespace backend {
|
|
namespace {{namespace}} {
|
|
|
|
namespace {
|
|
|
|
//* Helper functions to check the value of enums
|
|
{% for type in by_category["enum"] %}
|
|
{% set cType = as_cType(type.name) %}
|
|
bool CheckEnum{{cType}}({{cType}} value) {
|
|
switch (value) {
|
|
{% for value in type.values %}
|
|
case {{as_cEnum(type.name, value.name)}}:
|
|
return true;
|
|
{% endfor %}
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
{% endfor %}
|
|
|
|
{% for type in by_category["bitmask"] %}
|
|
{% set cType = as_cType(type.name) %}
|
|
bool CheckBitmask{{cType}}({{cType}} value) {
|
|
return (value & ~{{type.full_mask}}) == 0;
|
|
}
|
|
{% endfor %}
|
|
|
|
{% set methodsWithExtraValidation = (
|
|
"CommandBufferBuilderGetResult",
|
|
"QueueSubmit",
|
|
) %}
|
|
|
|
{% for type in by_category["object"] %}
|
|
{% for method in native_methods(type) %}
|
|
{% set suffix = as_MethodSuffix(type.name, method.name) %}
|
|
|
|
//* Entry point without validation, forwards the arguments to the method directly
|
|
{{as_backendType(method.return_type)}} NonValidating{{suffix}}(
|
|
{{-as_backendType(type)}} self
|
|
{%- for arg in method.arguments -%}
|
|
, {{as_annotated_backendType(arg)}}
|
|
{%- endfor -%}
|
|
) {
|
|
{% if method.return_type.name.canonical_case() != "void" %}
|
|
auto result =
|
|
{%- endif %}
|
|
self->{{method.name.CamelCase()}}(
|
|
{%- for arg in method.arguments -%}
|
|
{%- if not loop.first %}, {% endif -%}
|
|
{%- if arg.type.category in ["enum", "bitmask"] -%}
|
|
static_cast<nxt::{{as_cppType(arg.type.name)}}>({{as_varName(arg.name)}})
|
|
{%- else -%}
|
|
{{as_varName(arg.name)}}
|
|
{%- endif -%}
|
|
{%- endfor -%}
|
|
);
|
|
{% if method.return_type.name.canonical_case() != "void" %}
|
|
return reinterpret_cast<{{as_backendType(method.return_type)}}>(result);
|
|
{% endif %}
|
|
}
|
|
|
|
//* Autogenerated part of the entry point validation
|
|
//* - Check that enum and bitmaks are in the correct range
|
|
//* - Check that builders have not been consumed already
|
|
//* - Others TODO
|
|
bool ValidateBase{{suffix}}(
|
|
{{-as_backendType(type)}} self
|
|
{%- for arg in method.arguments -%}
|
|
, {{as_annotated_backendType(arg)}}
|
|
{%- endfor -%}
|
|
) {
|
|
{% if type.is_builder and method.name.canonical_case() not in ("release", "reference") %}
|
|
if (!self->CanBeUsed()) {
|
|
self->GetDevice()->HandleError("Builder cannot be used after GetResult");
|
|
return false;
|
|
}
|
|
{% else %}
|
|
(void) self;
|
|
{% endif %}
|
|
bool error = false;
|
|
{% for arg in method.arguments %}
|
|
{% if arg.type.category == "enum" %}
|
|
if (!CheckEnum{{as_cType(arg.type.name)}}({{as_varName(arg.name)}})) error = true;;
|
|
{% elif arg.type.category == "bitmask" %}
|
|
if (!CheckBitmask{{as_cType(arg.type.name)}}({{as_varName(arg.name)}})) error = true;
|
|
{% else %}
|
|
(void) {{as_varName(arg.name)}};
|
|
{% endif %}
|
|
if (error) {
|
|
{% if type.is_builder %}
|
|
self->HandleError("Bad value in {{suffix}}");
|
|
{% else %}
|
|
self->GetDevice()->HandleError("Bad value in {{suffix}}");
|
|
{% endif %}
|
|
return false;
|
|
}
|
|
{% endfor %}
|
|
return true;
|
|
}
|
|
|
|
//* Entry point with validation
|
|
{{as_backendType(method.return_type)}} Validating{{suffix}}(
|
|
{{-as_backendType(type)}} self
|
|
{%- for arg in method.arguments -%}
|
|
, {{as_annotated_backendType(arg)}}
|
|
{%- endfor -%}
|
|
) {
|
|
//* Do the autogenerated checks
|
|
bool valid = ValidateBase{{suffix}}(self
|
|
{%- for arg in method.arguments -%}
|
|
, {{as_varName(arg.name)}}
|
|
{%- endfor -%}
|
|
);
|
|
|
|
//* Some function have very heavy checks in a seperate method, so that they
|
|
//* can be skipped in the NonValidatingEntryPoints.
|
|
{% if suffix in methodsWithExtraValidation %}
|
|
if (valid) {
|
|
valid = self->Validate{{method.name.CamelCase()}}(
|
|
{%- for arg in method.arguments -%}
|
|
{% if not loop.first %}, {% endif %}{{as_varName(arg.name)}}
|
|
{%- endfor -%}
|
|
);
|
|
}
|
|
{% endif %}
|
|
|
|
//* HACK(cwallez@chromium.org): special casing GetResult so that the error callback
|
|
//* is called if needed. Without this, no call to HandleResult would happen, and the
|
|
//* error callback would always get called with an Unknown status
|
|
{% if type.is_builder and method.name.canonical_case() == "get result" %}
|
|
if (!valid) {
|
|
{{as_backendType(method.return_type)}} fakeResult = nullptr;
|
|
bool shouldBeFalse = self->HandleResult(fakeResult);
|
|
assert(shouldBeFalse == false);
|
|
}
|
|
{% endif %}
|
|
|
|
{% if method.return_type.name.canonical_case() == "void" %}
|
|
if (!valid) return;
|
|
{% else %}
|
|
if (!valid) {
|
|
return {};
|
|
}
|
|
auto result =
|
|
{%- endif %}
|
|
self->{{method.name.CamelCase()}}(
|
|
{%- for arg in method.arguments -%}
|
|
{%- if not loop.first %}, {% endif -%}
|
|
{%- if arg.type.category in ["enum", "bitmask"] -%}
|
|
static_cast<nxt::{{as_cppType(arg.type.name)}}>({{as_varName(arg.name)}})
|
|
{%- else -%}
|
|
{{as_varName(arg.name)}}
|
|
{%- endif -%}
|
|
{%- endfor -%}
|
|
);
|
|
{% if method.return_type.name.canonical_case() != "void" %}
|
|
return reinterpret_cast<{{as_backendType(method.return_type)}}>(result);
|
|
{% endif %}
|
|
}
|
|
{% endfor %}
|
|
{% endfor %}
|
|
}
|
|
|
|
nxtProcTable GetNonValidatingProcs() {
|
|
nxtProcTable table;
|
|
{% for type in by_category["object"] %}
|
|
{% for method in native_methods(type) %}
|
|
table.{{as_varName(type.name, method.name)}} = reinterpret_cast<{{as_cProc(type.name, method.name)}}>(NonValidating{{as_MethodSuffix(type.name, method.name)}});
|
|
{% endfor %}
|
|
{% endfor %}
|
|
return table;
|
|
}
|
|
|
|
nxtProcTable GetValidatingProcs() {
|
|
nxtProcTable table;
|
|
{% for type in by_category["object"] %}
|
|
{% for method in native_methods(type) %}
|
|
table.{{as_varName(type.name, method.name)}} = reinterpret_cast<{{as_cProc(type.name, method.name)}}>(Validating{{as_MethodSuffix(type.name, method.name)}});
|
|
{% endfor %}
|
|
{% endfor %}
|
|
return table;
|
|
}
|
|
}
|
|
}
|