blob: fc910e69ce3a4baea8997dfb2fe743c6132d9ae9 [file] [log] [blame]
//* Copyright 2024 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.
{% from 'art/api_jni_types.cpp' import convert_to_kotlin, jni_signature with context %}
{% from 'art/kotlin_record_conversion.cpp' import define_kotlin_record_structure, define_kotlin_to_struct_conversion with context %}
#include "structures.h"
#include <cassert>
#include <string>
#include <jni.h>
#include <webgpu/webgpu.h>
#include "dawn/common/Assert.h"
#include "dawn/common/Log.h"
#include "JNIContext.h"
// Converts Kotlin objects representing Dawn structures into native structures that can be passed
// into the native Dawn API.
namespace dawn::kotlin_api {
// Helper functions to call the correct JNIEnv::Call*Method depending on what return type we expect.
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jboolean* result) {
*result = env->CallBooleanMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jbyte* result) {
*result = env->CallByteMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jchar* result) {
*result = env->CallCharMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jshort* result) {
*result = env->CallShortMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jint* result) {
*result = env->CallIntMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jlong* result) {
*result = env->CallLongMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jfloat* result) {
*result = env->CallFloatMethod(obj, getter);
}
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, jdouble* result) {
*result = env->CallDoubleMethod(obj, getter);
}
template <typename T>
void CallGetter(JNIEnv* env, jmethodID getter, jobject obj, T** result) {
*result = reinterpret_cast<T*>(env->CallObjectMethod(obj, getter));
}
// Special-case [Nullable]StringView
void ToNative(JNIContext* c, JNIEnv* env, jstring obj, WGPUStringView* s) {
if (obj == nullptr) {
*s = {nullptr, WGPU_STRLEN};
return;
}
*s = {c->GetStringUTFChars(obj), static_cast<size_t>(env->GetStringUTFLength(obj))};
}
jobject ToKotlin(JNIEnv* env, const WGPUStringView* s) {
if (s->length == WGPU_STRLEN) {
if (s->data == nullptr) {
return nullptr;
}
return env->NewStringUTF(s->data);
}
std::string nullTerminated(s->data, s->length);
return env->NewStringUTF(nullTerminated.c_str());
}
{%- for structure in by_category['structure'] + by_category['callback info'] if include_structure(structure) %}
//* Native -> Kotlin converter.
//* TODO(b/354411474): Filter the structures for which to add a ToKotlin conversion.
jobject ToKotlin(JNIEnv *env, const {{ as_cType(structure.name) }}* input) {
if (!input) {
return nullptr;
}
//* Make a new Kotlin object to receive a copy of the structure.
jclass clz = env->FindClass("{{ jni_name(structure) }}");
//* JNI signature needs to be built using the same logic used in the Kotlin structure spec.
jmethodID ctor = env->GetMethodID(clz, "<init>", "(
{%- for member in kotlin_record_members(structure.members) %}
{{- jni_signature(member) -}}
{%- endfor -%}
{%- for structure in chain_children[structure.name.get()] -%}
{{- jni_signature({'type': structure}) -}}
{%- endfor %})V");
//* Each field converted using the individual value converter.
{% for member in kotlin_record_members(structure.members) %}
{{ convert_to_kotlin('input->' + member.name.camelCase(), member.name.camelCase(),
'input->' + member.length.name.camelCase() if member.length.name,
member) | indent(4) -}}
{% endfor %}
//* Allow conversion of every child structure.
{%- for structure in chain_children[structure.name.get()] %}
jobject {{ structure.name.camelCase() }} = nullptr;
{% endfor %}
{% set ChainedStruct = "WGPUChainedStructOut" if structure.extensible == "out" else "WGPUChainedStruct" %}
//* Walk the chain to find and convert (recursively) all child structures.
{%- if chain_children[structure.name.get()] %}
for (const {{ChainedStruct}}* child = input->nextInChain;
child != nullptr; child = child->next) {
switch (child->sType) {
{% for structure in chain_children[structure.name.get()] %}
case WGPUSType_{{ structure.name.CamelCase() }}:
{{ structure.name.camelCase() }} = ToKotlin(env,
reinterpret_cast<const {{ as_cType(structure.name) }}*>(child));
break;
{% endfor %}
default:
DAWN_UNREACHABLE();
}
}
{% endif %}
//* Now all the fields are converted, invoke the constructor.
jobject converted = env->NewObject(
clz,
ctor
{%- for member in kotlin_record_members(structure.members) %},
{{ member.name.camelCase() -}}
{%- endfor -%}
{%- for structure in chain_children[structure.name.get()] -%},
{{ structure.name.camelCase() -}}
{%- endfor -%}
);
return converted;
}
{% set Struct = as_cType(structure.name) %}
{% set KotlinRecord = "KotlinRecord" + structure.name.CamelCase() %}
{{ define_kotlin_record_structure(KotlinRecord, structure.members)}}
{{ define_kotlin_to_struct_conversion("ConvertInternal", KotlinRecord, Struct, structure.members)}}
void ToNative(JNIContext* c, JNIEnv* env, jobject obj, {{ as_cType(structure.name) }}* converted) {
jclass clz = env->FindClass("{{ jni_name(structure) }}");
//* Use getters to fill in the Kotlin record that will get converted to our struct.
{{KotlinRecord}} kotlinRecord;
{% for member in kotlin_record_members(structure.members) %}
{
jmethodID getter = env->GetMethodID(clz, "get{{member.name.CamelCase()}}", "(){{jni_signature(member)}}");
CallGetter(env, getter, obj, &kotlinRecord.{{as_varName(member.name)}});
}
{% endfor %}
//* Fill all struct members from the Kotlin record.
ConvertInternal(c, kotlinRecord, converted);
//* Set up the chain type and links for child objects.
{% if structure.chained %}
converted->chain = {.sType = WGPUSType_{{ structure.name.CamelCase() }}};
{% endif %}
{% for child in chain_children[structure.name.get()] %}
{
jobject child = env->CallObjectMethod(obj,
env->GetMethodID(clz, "get{{ child.name.CamelCase() }}", "()L{{ jni_name(child) }};"));
if (child) {
auto out = c->Alloc<{{ as_cType(child.name) }}>();
ToNative(c, env, child, out);
out->chain.next = converted->nextInChain;
converted->nextInChain = &out->chain;
}
}
{% endfor %}
}
{% endfor %}
} // namespace dawn::kotlin_api