//===-- APIDefs.td - Base definitions for Offload tablegen -*- tablegen -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // This file contains the class definitions used to implement the Offload API, // as well as helper functions used to help populate relevant records. // See offload/API/README.md for more detailed documentation. // //===----------------------------------------------------------------------===// // Prefix for API naming. This could be hard-coded in the future when a value // is agreed upon. defvar PREFIX = "OL"; defvar prefix = !tolower(PREFIX); // Parameter flags defvar PARAM_IN = 0x1; defvar PARAM_OUT = 0x2; defvar PARAM_OPTIONAL = 0x4; defvar PARAM_IN_OPTIONAL = !or(PARAM_IN, PARAM_OPTIONAL); defvar PARAM_OUT_OPTIONAL = !or(PARAM_OUT, PARAM_OPTIONAL); // Does the type end with '_handle_t'? class IsHandleType { // size("_handle_t") == 9 bit ret = !if(!lt(!size(Type), 9), 0, !ne(!find(Type, "_handle_t", !sub(!size(Type), 9)), -1)); } // Does the type end with '_cb_t'? class IsCallbackType { // size("_cb_t") == 5 bit ret = !if(!lt(!size(Type), 5), 0, !ne(!find(Type, "_cb_t", !sub(!size(Type), 5)), -1)); } // Does the type end with '*'? class IsPointerType { bit ret = !ne(!find(Type, "*", !sub(!size(Type), 1)), -1); } // Describes the valid range of a pointer parameter that represents an array class Range { string begin = Begin; string end = End; } // Names the parameters that indicate the type and size of the data pointed to // by an opaque pointer parameter class TypeInfo { string enum = TypeEnum; string size = TypeSize; } class Param Flags = 0> { string type = Type; string name = Name; string desc = Desc; bits<3> flags = Flags; Range range = Range<"", "">; TypeInfo type_info = TypeInfo<"", "">; bit IsHandle = IsHandleType.ret; bit IsPointer = IsPointerType.ret; bit IsCallback = IsCallbackType.ret; } // A parameter whose range is described by other parameters in the function. class RangedParam Flags, Range ParamRange> : Param { let range = ParamRange; } // A parameter (normally of type void*) which has its pointee type and size // described by other parameters in the function. class TypeTaggedParam Flags, TypeInfo ParamTypeInfo> : Param { let type_info = ParamTypeInfo; } class Return Conditions = []> { string value = Value; list conditions = Conditions; } class ShouldCheckHandle { bit ret = !and(P.IsHandle, !eq(!and(PARAM_OPTIONAL, P.flags), 0)); } class ShouldCheckPointer { bit ret = !and(!or(P.IsPointer, P.IsCallback), !eq(!and(PARAM_OPTIONAL, P.flags), 0)); } // For a list of returns that contains a specific return code, find and append // new conditions to that return class AppendConditionsToReturn Returns, string ReturnValue, list Conditions> { list ret = !foreach(Ret, Returns, !if(!eq(Ret.value, ReturnValue), Return, Ret)); } // Add null handle checks to a function's return value descriptions class AddHandleChecksToReturns Params, list Returns> { list handle_params = !foreach(P, Params, !if(ShouldCheckHandle

.ret, P.name, "")); list handle_params_filt = !filter(param, handle_params, !ne(param, "")); list handle_param_conds = !foreach(handle, handle_params_filt, "`NULL == "#handle#"`"); // Does the list of returns already contain ERROR_INVALID_NULL_HANDLE? bit returns_has_inv_handle = !foldl( 0, Returns, HasErr, Ret, !or(HasErr, !eq(Ret.value, PREFIX#"_ERRC_INVALID_NULL_HANDLE"))); list returns_out = !if(returns_has_inv_handle, AppendConditionsToReturn.ret, !listconcat(Returns, [Return]) ); } // Add null pointer checks to a function's return value descriptions class AddPointerChecksToReturns Params, list Returns> { list ptr_params = !foreach(P, Params, !if(ShouldCheckPointer

.ret, P.name, "")); list ptr_params_filt = !filter(param, ptr_params, !ne(param, "")); list ptr_param_conds = !foreach(ptr, ptr_params_filt, "`NULL == "#ptr#"`"); // Does the list of returns already contain ERROR_INVALID_NULL_POINTER? bit returns_has_inv_ptr = !foldl( 0, Returns, HasErr, Ret, !or(HasErr, !eq(Ret.value, PREFIX#"_ERRC_INVALID_NULL_POINTER"))); list returns_out = !if(returns_has_inv_ptr, AppendConditionsToReturn.ret, !listconcat(Returns, [Return]) ); } defvar DefaultReturns = [Return, Return, Return]; class APIObject { string desc; } class Function : APIObject { list params; list returns; list details = []; list analogues = []; list returns_with_def = !listconcat(DefaultReturns, returns); list all_returns = AddPointerChecksToReturns.returns_out>.returns_out; } class Etor { string name = Name; string desc = Desc; string tagged_type; } class TaggedEtor : Etor { let tagged_type = Type; } class Enum : APIObject { // This refers to whether the enumerator descriptions specify a return // type for functions where this enum may be used as an output type. If set, // all Etor values must be TaggedEtor records bit is_typed = 0; // This refers to whether the enumerator is used to name bits of a bit field, // where consecutive values are bit-shifted rather than incremented. bit is_bit_field = 0; list etors = []; } class StructMember { string type = Type; string name = Name; string desc = Desc; } defvar DefaultPropStructMembers = [StructMember, StructMember<"void*", "pNext", "pointer to extension-specific structure">]; class StructHasInheritedMembers { bit ret = !or(!eq(BaseClass, prefix#"_base_properties_t"), !eq(BaseClass, prefix#"_base_desc_t")); } class Struct : APIObject { string base_class = ""; list members; list all_members = !if(StructHasInheritedMembers.ret, DefaultPropStructMembers, [])#members; } class Typedef : APIObject { string value; } class FptrTypedef : APIObject { list params; string return; } class Macro : APIObject { string value; string condition; string alt_value; } class Handle : APIObject;