2021-05-29 08:08:23 +08:00
|
|
|
/*===- c_api.h - C API for the ORC runtime ------------------------*- C -*-===*\
|
|
|
|
|* *|
|
|
|
|
|* 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 defines the C API for the ORC runtime *|
|
|
|
|
|* *|
|
|
|
|
|* NOTE: The OrtRTWrapperFunctionResult type must be kept in sync with the *|
|
|
|
|
|* definition in llvm/include/llvm-c/OrcShared.h. *|
|
|
|
|
|* *|
|
|
|
|
\*===----------------------------------------------------------------------===*/
|
|
|
|
|
|
|
|
#ifndef ORC_RT_C_API_H
|
|
|
|
#define ORC_RT_C_API_H
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
2021-06-09 08:53:14 +08:00
|
|
|
#include <stdlib.h>
|
2021-05-29 08:08:23 +08:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
/* Helper to suppress strict prototype warnings. */
|
|
|
|
#ifdef __clang__
|
|
|
|
#define ORC_RT_C_STRICT_PROTOTYPES_BEGIN \
|
|
|
|
_Pragma("clang diagnostic push") \
|
|
|
|
_Pragma("clang diagnostic error \"-Wstrict-prototypes\"")
|
|
|
|
#define ORC_RT_C_STRICT_PROTOTYPES_END _Pragma("clang diagnostic pop")
|
|
|
|
#else
|
|
|
|
#define ORC_RT_C_STRICT_PROTOTYPES_BEGIN
|
|
|
|
#define ORC_RT_C_STRICT_PROTOTYPES_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Helper to wrap C code for C++ */
|
|
|
|
#ifdef __cplusplus
|
|
|
|
#define ORC_RT_C_EXTERN_C_BEGIN \
|
|
|
|
extern "C" { \
|
|
|
|
ORC_RT_C_STRICT_PROTOTYPES_BEGIN
|
|
|
|
#define ORC_RT_C_EXTERN_C_END \
|
|
|
|
ORC_RT_C_STRICT_PROTOTYPES_END \
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define ORC_RT_C_EXTERN_C_BEGIN ORC_RT_C_STRICT_PROTOTYPES_BEGIN
|
|
|
|
#define ORC_RT_C_EXTERN_C_END ORC_RT_C_STRICT_PROTOTYPES_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ORC_RT_C_EXTERN_C_BEGIN
|
|
|
|
|
|
|
|
typedef union {
|
2021-07-19 19:01:14 +08:00
|
|
|
char *ValuePtr;
|
2021-08-13 04:55:16 +08:00
|
|
|
char Value[sizeof(char *)];
|
2021-06-01 06:37:00 +08:00
|
|
|
} __orc_rt_CWrapperFunctionResultDataUnion;
|
2021-05-29 08:08:23 +08:00
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* __orc_rt_CWrapperFunctionResult is a kind of C-SmallVector with an
|
|
|
|
* out-of-band error state.
|
2021-05-29 08:08:23 +08:00
|
|
|
*
|
|
|
|
* If Size == 0 and Data.ValuePtr is non-zero then the value is in the
|
|
|
|
* 'out-of-band error' state, and Data.ValuePtr points at a malloc-allocated,
|
|
|
|
* null-terminated string error message.
|
|
|
|
*
|
2021-06-01 06:37:00 +08:00
|
|
|
* If Size <= sizeof(__orc_rt_CWrapperFunctionResultData) then the value is in
|
|
|
|
* the 'small' state and the content is held in the first Size bytes of
|
|
|
|
* Data.Value.
|
2021-05-29 08:08:23 +08:00
|
|
|
*
|
|
|
|
* If Size > sizeof(OrtRTCWrapperFunctionResultData) then the value is in the
|
|
|
|
* 'large' state and the content is held in the first Size bytes of the
|
|
|
|
* memory pointed to by Data.ValuePtr. This memory must have been allocated by
|
|
|
|
* malloc, and will be freed with free when this value is destroyed.
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2021-06-01 06:37:00 +08:00
|
|
|
__orc_rt_CWrapperFunctionResultDataUnion Data;
|
2021-05-29 08:08:23 +08:00
|
|
|
size_t Size;
|
2021-06-01 06:37:00 +08:00
|
|
|
} __orc_rt_CWrapperFunctionResult;
|
2021-05-29 08:08:23 +08:00
|
|
|
|
2021-06-01 06:37:00 +08:00
|
|
|
typedef struct __orc_rt_CSharedOpaqueJITProcessControl
|
|
|
|
*__orc_rt_SharedJITProcessControlRef;
|
2021-05-29 08:08:23 +08:00
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* Zero-initialize an __orc_rt_CWrapperFunctionResult.
|
2021-05-29 08:08:23 +08:00
|
|
|
*/
|
|
|
|
static inline void
|
2021-06-01 06:37:00 +08:00
|
|
|
__orc_rt_CWrapperFunctionResultInit(__orc_rt_CWrapperFunctionResult *R) {
|
2021-05-29 08:08:23 +08:00
|
|
|
R->Size = 0;
|
|
|
|
R->Data.ValuePtr = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* Create an __orc_rt_CWrapperFunctionResult with an uninitialized buffer of
|
|
|
|
* size Size. The buffer is returned via the DataPtr argument.
|
2021-05-29 08:08:23 +08:00
|
|
|
*/
|
2021-08-24 15:16:20 +08:00
|
|
|
static inline __orc_rt_CWrapperFunctionResult
|
|
|
|
__orc_rt_CWrapperFunctionResultAllocate(size_t Size) {
|
|
|
|
__orc_rt_CWrapperFunctionResult R;
|
|
|
|
R.Size = Size;
|
2021-08-25 02:10:11 +08:00
|
|
|
// If Size is 0 ValuePtr must be 0 or it is considered an out-of-band error.
|
|
|
|
R.Data.ValuePtr = 0;
|
2021-08-24 15:16:20 +08:00
|
|
|
if (Size > sizeof(R.Data.Value))
|
|
|
|
R.Data.ValuePtr = (char *)malloc(Size);
|
|
|
|
return R;
|
2021-05-29 08:08:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* Create an __orc_rt_WrapperFunctionResult from the given data range.
|
2021-05-29 08:08:23 +08:00
|
|
|
*/
|
2021-06-01 06:37:00 +08:00
|
|
|
static inline __orc_rt_CWrapperFunctionResult
|
|
|
|
__orc_rt_CreateCWrapperFunctionResultFromRange(const char *Data, size_t Size) {
|
|
|
|
__orc_rt_CWrapperFunctionResult R;
|
2021-05-29 08:08:23 +08:00
|
|
|
R.Size = Size;
|
|
|
|
if (R.Size > sizeof(R.Data.Value)) {
|
|
|
|
char *Tmp = (char *)malloc(Size);
|
|
|
|
memcpy(Tmp, Data, Size);
|
|
|
|
R.Data.ValuePtr = Tmp;
|
|
|
|
} else
|
|
|
|
memcpy(R.Data.Value, Data, Size);
|
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* Create an __orc_rt_CWrapperFunctionResult by copying the given string,
|
|
|
|
* including the null-terminator.
|
2021-05-29 08:08:23 +08:00
|
|
|
*
|
|
|
|
* This function copies the input string. The client is responsible for freeing
|
|
|
|
* the ErrMsg arg.
|
|
|
|
*/
|
2021-06-01 06:37:00 +08:00
|
|
|
static inline __orc_rt_CWrapperFunctionResult
|
|
|
|
__orc_rt_CreateCWrapperFunctionResultFromString(const char *Source) {
|
|
|
|
return __orc_rt_CreateCWrapperFunctionResultFromRange(Source,
|
|
|
|
strlen(Source) + 1);
|
2021-05-29 08:08:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* Create an __orc_rt_CWrapperFunctionResult representing an out-of-band
|
2021-05-29 08:08:23 +08:00
|
|
|
* error.
|
|
|
|
*
|
2021-08-13 04:55:16 +08:00
|
|
|
* This function copies the input string. The client is responsible for freeing
|
|
|
|
* the ErrMsg arg.
|
2021-05-29 08:08:23 +08:00
|
|
|
*/
|
2021-06-01 06:37:00 +08:00
|
|
|
static inline __orc_rt_CWrapperFunctionResult
|
|
|
|
__orc_rt_CreateCWrapperFunctionResultFromOutOfBandError(const char *ErrMsg) {
|
|
|
|
__orc_rt_CWrapperFunctionResult R;
|
2021-05-29 08:08:23 +08:00
|
|
|
R.Size = 0;
|
|
|
|
char *Tmp = (char *)malloc(strlen(ErrMsg) + 1);
|
|
|
|
strcpy(Tmp, ErrMsg);
|
|
|
|
R.Data.ValuePtr = Tmp;
|
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* This should be called to destroy __orc_rt_CWrapperFunctionResult values
|
2021-05-29 08:08:23 +08:00
|
|
|
* regardless of their state.
|
|
|
|
*/
|
|
|
|
static inline void
|
2021-06-01 06:37:00 +08:00
|
|
|
__orc_rt_DisposeCWrapperFunctionResult(__orc_rt_CWrapperFunctionResult *R) {
|
2021-05-29 08:08:23 +08:00
|
|
|
if (R->Size > sizeof(R->Data.Value) ||
|
|
|
|
(R->Size == 0 && R->Data.ValuePtr))
|
2021-07-19 19:01:14 +08:00
|
|
|
free(R->Data.ValuePtr);
|
2021-05-29 08:08:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a pointer to the data contained in the given
|
2021-06-01 06:37:00 +08:00
|
|
|
* __orc_rt_CWrapperFunctionResult.
|
2021-05-29 08:08:23 +08:00
|
|
|
*/
|
2021-08-24 15:16:20 +08:00
|
|
|
static inline char *
|
|
|
|
__orc_rt_CWrapperFunctionResultData(__orc_rt_CWrapperFunctionResult *R) {
|
2021-08-13 04:55:16 +08:00
|
|
|
assert((R->Size != 0 || R->Data.ValuePtr == NULL) &&
|
2021-05-29 08:08:23 +08:00
|
|
|
"Cannot get data for out-of-band error value");
|
|
|
|
return R->Size > sizeof(R->Data.Value) ? R->Data.ValuePtr : R->Data.Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-06-01 06:37:00 +08:00
|
|
|
* Safely get the size of the given __orc_rt_CWrapperFunctionResult.
|
2021-05-29 08:08:23 +08:00
|
|
|
*
|
|
|
|
* Asserts that we're not trying to access the size of an error value.
|
|
|
|
*/
|
|
|
|
static inline size_t
|
2021-06-01 06:37:00 +08:00
|
|
|
__orc_rt_CWrapperFunctionResultSize(const __orc_rt_CWrapperFunctionResult *R) {
|
2021-08-13 04:55:16 +08:00
|
|
|
assert((R->Size != 0 || R->Data.ValuePtr == NULL) &&
|
2021-05-29 08:08:23 +08:00
|
|
|
"Cannot get size for out-of-band error value");
|
|
|
|
return R->Size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns 1 if this value is equivalent to a value just initialized by
|
2021-06-01 06:37:00 +08:00
|
|
|
* __orc_rt_CWrapperFunctionResultInit, 0 otherwise.
|
2021-05-29 08:08:23 +08:00
|
|
|
*/
|
|
|
|
static inline size_t
|
2021-06-01 06:37:00 +08:00
|
|
|
__orc_rt_CWrapperFunctionResultEmpty(const __orc_rt_CWrapperFunctionResult *R) {
|
2021-05-29 08:08:23 +08:00
|
|
|
return R->Size == 0 && R->Data.ValuePtr == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a pointer to the out-of-band error string for this
|
2021-06-01 06:37:00 +08:00
|
|
|
* __orc_rt_CWrapperFunctionResult, or null if there is no error.
|
2021-05-29 08:08:23 +08:00
|
|
|
*
|
2021-06-01 06:37:00 +08:00
|
|
|
* The __orc_rt_CWrapperFunctionResult retains ownership of the error
|
2021-05-29 08:08:23 +08:00
|
|
|
* string, so it should be copied if the caller wishes to preserve it.
|
|
|
|
*/
|
2021-06-01 06:37:00 +08:00
|
|
|
static inline const char *__orc_rt_CWrapperFunctionResultGetOutOfBandError(
|
|
|
|
const __orc_rt_CWrapperFunctionResult *R) {
|
2021-05-29 08:08:23 +08:00
|
|
|
return R->Size == 0 ? R->Data.ValuePtr : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ORC_RT_C_EXTERN_C_END
|
|
|
|
|
|
|
|
#endif /* ORC_RT_C_API_H */
|