blob: 2f65e5a20a852427d5b520a4ca23aceb806595c1 [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.
package {{ kotlin_package }}
import kotlin.coroutines.resume
import kotlin.coroutines.suspendCoroutine
{% from 'art/api_kotlin_types.kt' import kotlin_declaration, kotlin_definition with context %}
//* Provide an async wrapper for the 'callback info' type of async methods.
{% for obj in by_category['object'] %}
{% for method in obj.methods if has_callbackInfoStruct(method) %}
{% set callback_info = method.arguments[-1].type %}
{% set callback_function = callback_info.members[-1].type %}
{% set return_name = callback_function.name.chunks[:-1] | map('title') | join + 'Return' %}
//* We make a return class for every callback method so that it can be used inline
//* (without callbacks) in a suspend (async) function.
public data class {{ return_name }}(
{% for arg in kotlin_record_members(callback_function.arguments) %}
val {{ as_varName(arg.name) }}: {{ kotlin_declaration(arg) }},
{% endfor %})
//* Every method that is identified as using callbacks is given a helper method that wraps
//* call with a suspend function.
public suspend fun {{ obj.name.CamelCase() }}.{{ method.name.camelCase() }}(
{%- for arg in method.arguments[:-1] %}
{{- as_varName(arg.name) }}: {{ kotlin_definition(arg) }},
{%- endfor %}): {{ return_name }} = suspendCoroutine {
{{ method.name.camelCase() }}(
{%- for arg in method.arguments %}
{{- as_varName(arg.name) }}
{%- if loop.last %}
//* The final parameter of a callback method is always callback info.
//* We make this and include our generated callback.
{{- ' = ' }}
{{- callback_info.name.CamelCase() }}(CallbackMode.AllowSpontaneous)
{%- else %}
//* Non-final parameters are whatever the client supplied.
{{- ', ' }}
{%- endif %}
{%- endfor %}{
{%- for arg in kotlin_record_members(callback_function.arguments) %}
{{- as_varName(arg.name) }},
{%- endfor %} -> it.resume({{ return_name }}(
//* We make an instance of the callback parameters -> return type wrapper.
{%- for arg in kotlin_record_members(callback_function.arguments) %}
{{- as_varName(arg.name) }} {{ ', ' }}
{%- endfor %})
)
})
}
{% endfor %}
{% endfor %}