[llvm] [Offload] Introduce offload-tblgen and initial new API implementation (PR #108413)
Callum Fare via llvm-commits
llvm-commits at lists.llvm.org
Fri Oct 25 02:11:06 PDT 2024
https://github.com/callumfare updated https://github.com/llvm/llvm-project/pull/108413
>From 841d00229858f15476089d866c95ad3795d117a0 Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Thu, 12 Sep 2024 15:32:27 +0100
Subject: [PATCH 01/12] [Offload] Introduce offload-tblgen and initial new API
implementation
---
offload/CMakeLists.txt | 3 +
offload/new-api/API/APIDefs.td | 212 ++++++++++++++++
offload/new-api/API/Common.td | 75 ++++++
offload/new-api/API/Device.td | 101 ++++++++
offload/new-api/API/OffloadAPI.td | 15 ++
offload/new-api/API/Platform.td | 94 ++++++++
offload/new-api/API/README.md | 150 ++++++++++++
offload/new-api/CMakeLists.txt | 36 +++
offload/new-api/README.md | 8 +
offload/new-api/src/helpers.hpp | 96 ++++++++
offload/new-api/src/offload_impl.cpp | 187 +++++++++++++++
offload/new-api/src/offload_lib.cpp | 23 ++
.../common/include/PluginInterface.h | 7 +
offload/tools/offload-tblgen/APIGen.cpp | 196 +++++++++++++++
offload/tools/offload-tblgen/CMakeLists.txt | 24 ++
.../tools/offload-tblgen/EntryPointGen.cpp | 86 +++++++
offload/tools/offload-tblgen/FuncsGen.cpp | 54 +++++
offload/tools/offload-tblgen/GenCommon.hpp | 58 +++++
offload/tools/offload-tblgen/Generators.hpp | 18 ++
offload/tools/offload-tblgen/PrintGen.cpp | 206 ++++++++++++++++
offload/tools/offload-tblgen/RecordTypes.hpp | 227 ++++++++++++++++++
.../tools/offload-tblgen/offload-tblgen.cpp | 95 ++++++++
22 files changed, 1971 insertions(+)
create mode 100644 offload/new-api/API/APIDefs.td
create mode 100644 offload/new-api/API/Common.td
create mode 100644 offload/new-api/API/Device.td
create mode 100644 offload/new-api/API/OffloadAPI.td
create mode 100644 offload/new-api/API/Platform.td
create mode 100644 offload/new-api/API/README.md
create mode 100644 offload/new-api/CMakeLists.txt
create mode 100644 offload/new-api/README.md
create mode 100644 offload/new-api/src/helpers.hpp
create mode 100644 offload/new-api/src/offload_impl.cpp
create mode 100644 offload/new-api/src/offload_lib.cpp
create mode 100644 offload/tools/offload-tblgen/APIGen.cpp
create mode 100644 offload/tools/offload-tblgen/CMakeLists.txt
create mode 100644 offload/tools/offload-tblgen/EntryPointGen.cpp
create mode 100644 offload/tools/offload-tblgen/FuncsGen.cpp
create mode 100644 offload/tools/offload-tblgen/GenCommon.hpp
create mode 100644 offload/tools/offload-tblgen/Generators.hpp
create mode 100644 offload/tools/offload-tblgen/PrintGen.cpp
create mode 100644 offload/tools/offload-tblgen/RecordTypes.hpp
create mode 100644 offload/tools/offload-tblgen/offload-tblgen.cpp
diff --git a/offload/CMakeLists.txt b/offload/CMakeLists.txt
index 9ffe8f56b76e67..ba7be255cfe582 100644
--- a/offload/CMakeLists.txt
+++ b/offload/CMakeLists.txt
@@ -349,6 +349,9 @@ add_subdirectory(tools)
# Build target agnostic offloading library.
add_subdirectory(src)
+add_subdirectory(tools/offload-tblgen)
+add_subdirectory(new-api)
+
# Add tests.
add_subdirectory(test)
diff --git a/offload/new-api/API/APIDefs.td b/offload/new-api/API/APIDefs.td
new file mode 100644
index 00000000000000..410a28c4c90cfe
--- /dev/null
+++ b/offload/new-api/API/APIDefs.td
@@ -0,0 +1,212 @@
+//===-- 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 = "OFFLOAD";
+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<string Type> {
+ // 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 '*'?
+class IsPointerType<string Type> {
+ bit ret = !ne(!find(Type, "*", !sub(!size(Type), 1)), -1);
+}
+
+// Describes the valid range of a pointer parameter that reperesents an array
+class Range<string Begin, string End> {
+ 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 TypeEnum, string TypeSize> {
+ string enum = TypeEnum;
+ string size = TypeSize;
+}
+
+class Param<string Type, string Name, string Desc, bits<3> 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<type>.ret;
+ bit IsPointer = IsPointerType<type>.ret;
+}
+
+// A parameter whose range is described by other parameters in the function.
+class RangedParam<string Type, string Name, string Desc, bits<3> Flags, Range ParamRange> : Param<Type, Name, Desc, Flags> {
+ 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<string Type, string Name, string Desc, bits<3> Flags, TypeInfo ParamTypeInfo> : Param<Type, Name, Desc, Flags> {
+ let type_info = ParamTypeInfo;
+}
+
+class Return<string Value, list<string> Conditions = []> {
+ string value = Value;
+ list<string> conditions = Conditions;
+}
+
+class ShouldCheckHandle<Param P> {
+ bit ret = !and(P.IsHandle, !eq(!and(PARAM_OPTIONAL, P.flags), 0));
+}
+
+class ShouldCheckPointer<Param P> {
+ bit ret = !and(P.IsPointer, !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<list<Return> Returns, string ReturnValue,
+ list<string> Conditions> {
+ list<Return> ret =
+ !foreach(Ret, Returns,
+ !if(!eq(Ret.value, ReturnValue),
+ Return<Ret.value, Ret.conditions#Conditions>, Ret));
+}
+
+// Add null handle checks to a function's return value descriptions
+class AddHandleChecksToReturns<list<Param> Params, list<Return> Returns> {
+ list<string> handle_params =
+ !foreach(P, Params, !if(ShouldCheckHandle<P>.ret, P.name, ""));
+ list<string> handle_params_filt =
+ !filter(param, handle_params, !ne(param, ""));
+ list<string> 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#"_RESULT_ERROR_INVALID_NULL_HANDLE")));
+
+ list<Return> returns_out = !if(returns_has_inv_handle,
+ AppendConditionsToReturn<Returns, PREFIX # "_RESULT_ERROR_INVALID_NULL_HANDLE", handle_param_conds>.ret,
+ !listconcat(Returns, [Return<PREFIX # "_RESULT_ERROR_INVALID_NULL_HANDLE", handle_param_conds>])
+ );
+}
+
+// Add null pointer checks to a function's return value descriptions
+class AddPointerChecksToReturns<list<Param> Params, list<Return> Returns> {
+ list<string> ptr_params =
+ !foreach(P, Params, !if(ShouldCheckPointer<P>.ret, P.name, ""));
+ list<string> ptr_params_filt = !filter(param, ptr_params, !ne(param, ""));
+ list<string> 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#"_RESULT_ERROR_INVALID_NULL_POINTER")));
+ list<Return> returns_out = !if(returns_has_inv_ptr,
+ AppendConditionsToReturn<Returns, PREFIX # "_RESULT_ERROR_INVALID_NULL_POINTER", ptr_param_conds>.ret,
+ !listconcat(Returns, [Return<PREFIX # "_RESULT_ERROR_INVALID_NULL_POINTER", ptr_param_conds>])
+ );
+}
+
+defvar DefaultReturns = [Return<PREFIX#"_RESULT_SUCCESS">,
+ Return<PREFIX#"_RESULT_ERROR_UNINITIALIZED">,
+ Return<PREFIX#"_RESULT_ERROR_DEVICE_LOST">];
+
+class APIObject {
+ string name;
+ string desc;
+}
+
+class Function : APIObject {
+ list<Param> params;
+ list<Return> returns;
+ list<string> details = [];
+ list<string> analogues = [];
+
+ list<Return> returns_with_def = !listconcat(DefaultReturns, returns);
+ list<Return> all_returns = AddPointerChecksToReturns<params,
+ AddHandleChecksToReturns<params, returns_with_def>.returns_out>.returns_out;
+}
+
+class Etor<string Name, string Desc> {
+ string name = Name;
+ string desc = Desc;
+ string tagged_type;
+}
+
+class TaggedEtor<string Name, string Type, string Desc> : Etor<Name, Desc> {
+ 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;
+
+ list<Etor> etors = [];
+}
+
+class StructMember<string Type, string Name, string Desc> {
+ string type = Type;
+ string name = Name;
+ string desc = Desc;
+}
+
+defvar DefaultPropStructMembers =
+ [StructMember<prefix#"_structure_type_t", "stype",
+ "type of this structure">,
+ StructMember<"void*", "pNext", "pointer to extension-specific structure">];
+
+class StructHasInheritedMembers<string BaseClass> {
+ bit ret = !or(!eq(BaseClass, prefix#"_base_properties_t"),
+ !eq(BaseClass, prefix#"_base_desc_t"));
+}
+
+class Struct : APIObject {
+ string base_class = "";
+ list<StructMember> members;
+ list<StructMember> all_members =
+ !if(StructHasInheritedMembers<base_class>.ret,
+ DefaultPropStructMembers, [])#members;
+}
+
+class Typedef : APIObject { string value; }
+
+class FptrTypedef : APIObject {
+ list<Param> params;
+ list<Return> returns;
+}
+
+class Macro : APIObject {
+ string value;
+
+ string condition;
+ string alt_value;
+}
+
+class Handle : APIObject;
diff --git a/offload/new-api/API/Common.td b/offload/new-api/API/Common.td
new file mode 100644
index 00000000000000..d293e4addfef8b
--- /dev/null
+++ b/offload/new-api/API/Common.td
@@ -0,0 +1,75 @@
+def : Macro {
+ let name = "OFFLOAD_APICALL";
+ let desc = "Calling convention for all API functions";
+ let condition = "defined(_WIN32)";
+ let value = "__cdecl";
+ let alt_value = "";
+}
+
+def : Macro {
+ let name = "OFFLOAD_APIEXPORT";
+ let desc = "Microsoft-specific dllexport storage-class attribute";
+ let condition = "defined(_WIN32)";
+ let value = "__declspec(dllexport)";
+ let alt_value = "";
+}
+
+def : Macro {
+ let name = "OFFLOAD_DLLEXPORT";
+ let desc = "Microsoft-specific dllexport storage-class attribute";
+ let condition = "defined(_WIN32)";
+ let value = "__declspec(dllexport)";
+}
+
+def : Macro {
+ let name = "OFFLOAD_DLLEXPORT";
+ let desc = "GCC-specific dllexport storage-class attribute";
+ let condition = "__GNUC__ >= 4";
+ let value = "__attribute__ ((visibility (\"default\")))";
+ let alt_value = "";
+}
+
+def : Typedef {
+ let name = "offload_bool_t";
+ let value = "uint8_t";
+ let desc = "compiler-independent type";
+}
+
+def : Handle {
+ let name = "offload_platform_handle_t";
+ let desc = "Handle of a platform instance";
+}
+
+def : Handle {
+ let name = "offload_device_handle_t";
+ let desc = "Handle of platform's device object";
+}
+
+def : Handle {
+ let name = "offload_context_handle_t";
+ let desc = "Handle of context object";
+}
+
+def : Enum {
+ let name = "offload_result_t";
+ let desc = "Defines Return/Error codes";
+ let etors =[
+ Etor<"SUCCESS", "Success">,
+ Etor<"ERROR_INVALID_VALUE", "Invalid Value">,
+ Etor<"ERROR_INVALID_PLATFORM", "Invalid platform">,
+ Etor<"ERROR_DEVICE_NOT_FOUND", "Device not found">,
+ Etor<"ERROR_INVALID_DEVICE", "Invalid device">,
+ Etor<"ERROR_DEVICE_LOST", "Device hung, reset, was removed, or driver update occurred">,
+ Etor<"ERROR_UNINITIALIZED", "plugin is not initialized or specific entry-point is not implemented">,
+ Etor<"ERROR_OUT_OF_RESOURCES", "Out of resources">,
+ Etor<"ERROR_UNSUPPORTED_VERSION", "[Validation] generic error code for unsupported versions">,
+ Etor<"ERROR_UNSUPPORTED_FEATURE", "[Validation] generic error code for unsupported features">,
+ Etor<"ERROR_INVALID_ARGUMENT", "[Validation] generic error code for invalid arguments">,
+ Etor<"ERROR_INVALID_NULL_HANDLE", "[Validation] handle argument is not valid">,
+ Etor<"ERROR_INVALID_NULL_POINTER", "[Validation] pointer argument may not be nullptr">,
+ Etor<"ERROR_INVALID_SIZE", "[Validation] invalid size or dimensions (e.g., must not be zero, or is out of bounds)">,
+ Etor<"ERROR_INVALID_ENUMERATION", "[Validation] enumerator argument is not valid">,
+ Etor<"ERROR_UNSUPPORTED_ENUMERATION", "[Validation] enumerator argument is not supported by the device">,
+ Etor<"ERROR_UNKNOWN", "Unknown or internal error">
+ ];
+}
\ No newline at end of file
diff --git a/offload/new-api/API/Device.td b/offload/new-api/API/Device.td
new file mode 100644
index 00000000000000..2a330d7ef7c4cb
--- /dev/null
+++ b/offload/new-api/API/Device.td
@@ -0,0 +1,101 @@
+//===-- Device.td - Device definitions for Offload ---------*- 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 Offload API definitions related to the Device handle
+//
+//===----------------------------------------------------------------------===//
+
+def : Enum {
+ let name = "offload_device_type_t";
+ let desc = "Supported device types";
+ let etors =[
+ Etor<"DEFAULT", "The default device type as preferred by the runtime">,
+ Etor<"ALL", "Devices of all types">,
+ Etor<"GPU", "GPU device type">,
+ Etor<"CPU", "CPU device type">,
+ ];
+}
+
+def : Enum {
+ let name = "offload_device_info_t";
+ let desc = "Supported device info";
+ let is_typed = 1;
+ let etors =[
+ TaggedEtor<"TYPE", "offload_device_type_t", "type of the device">,
+ TaggedEtor<"PLATFORM", "offload_platform_handle_t", "the platform associated with the device">,
+ TaggedEtor<"NAME", "char[]", "Device name">,
+ TaggedEtor<"VENDOR", "char[]", "Device vendor">,
+ TaggedEtor<"DRIVER_VERSION", "char[]", "Driver version">
+ ];
+}
+
+def : Function {
+ let name = "offloadDeviceGet";
+ let desc = "Retrieves devices within a platform";
+ let details = [
+ "Multiple calls to this function will return identical device handles, in the same order.",
+ "The number and order of handles returned from this function can be affected by environment variables that filter devices exposed through API.",
+ "The returned devices are taken a reference of and must be released with a subsequent call to olDeviceRelease.",
+ "The application may call this function from simultaneous threads, the implementation must be thread-safe"
+ ];
+ let params = [
+ Param<"offload_platform_handle_t", "hPlatform", "handle of the platform instance", PARAM_IN>,
+ Param<"offload_device_type_t", "DeviceType", "the type of the devices.", PARAM_IN>,
+ Param<"uint32_t", "NumEntries", "the number of devices to be added to phDevices."
+ "If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE"
+ "will be returned.", PARAM_IN>,
+ RangedParam<"offload_device_handle_t*", "phDevices", "array of handle of devices."
+ "If NumEntries is less than the number of devices available, then platform shall only retrieve that number of devices.", PARAM_OUT_OPTIONAL,
+ Range<"0", "NumEntries">>,
+ Param<"uint32_t*", "pNumDevices", "pointer to the number of devices."
+ "pNumDevices will be updated with the total number of devices available.", PARAM_OUT_OPTIONAL>
+ ];
+ let returns = [
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ "`NumEntries == 0 && phDevices != NULL`"
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER", [
+ "`NumEntries > 0 && phDevices == NULL`"
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_VALUE">
+ ];
+}
+
+def : Function {
+ let name = "offloadDeviceGetInfo";
+ let desc = "Retrieves various information about device";
+ let details = [
+ "The application may call this function from simultaneous threads.",
+ "The implementation of this function should be lock-free."
+ ];
+ let params = [
+ Param<"offload_device_handle_t", "hDevice", "handle of the device instance", PARAM_IN>,
+ Param<"offload_device_info_t", "propName", "type of the info to retrieve", PARAM_IN>,
+ Param<"size_t", "propSize", "the number of bytes pointed to by pPropValue.", PARAM_IN>,
+ TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info. If propSize is not equal to or greater than the real "
+ "number of bytes needed to return the info then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and "
+ "pPropValue is not used.", PARAM_OUT_OPTIONAL, TypeInfo<"propName" , "propSize">>,
+ Param<"size_t*", "pPropSizeRet", "pointer to the actual size in bytes of the queried propName.", PARAM_OUT_OPTIONAL>
+ ];
+ let returns = [
+ Return<"OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION", [
+ "If `propName` is not supported by the adapter."
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ "`propSize == 0 && pPropValue != NULL`",
+ "If `propSize` is less than the real number of bytes needed to return the info."
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER", [
+ "`propSize != 0 && pPropValue == NULL`",
+ "`pPropValue == NULL && pPropSizeRet == NULL`"
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_DEVICE">,
+ Return<"OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES">,
+ Return<"OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY">
+ ];
+}
diff --git a/offload/new-api/API/OffloadAPI.td b/offload/new-api/API/OffloadAPI.td
new file mode 100644
index 00000000000000..8a0c3c40581223
--- /dev/null
+++ b/offload/new-api/API/OffloadAPI.td
@@ -0,0 +1,15 @@
+//===-- OffloadAPI.td - Root tablegen file for Offload -----*- 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
+//
+//===----------------------------------------------------------------------===//
+
+// Always include this file first
+include "APIDefs.td"
+
+// Add API definition files here
+include "Common.td"
+include "Platform.td"
+include "Device.td"
diff --git a/offload/new-api/API/Platform.td b/offload/new-api/API/Platform.td
new file mode 100644
index 00000000000000..71af04bb831998
--- /dev/null
+++ b/offload/new-api/API/Platform.td
@@ -0,0 +1,94 @@
+//===-- Platform.td - Platform definitions for Offload -----*- 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 Offload API definitions related to the Platform handle
+//
+//===----------------------------------------------------------------------===//
+def : Function {
+ let name = "offloadPlatformGet";
+ let desc = "Retrieves all available platforms";
+ let details = [
+ "Multiple calls to this function will return identical platforms handles, in the same order.",
+ ];
+ let params = [
+ Param<"uint32_t", "NumEntries",
+ "The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, then"
+ "NumEntries should be greater than zero, otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE"
+ "will be returned.", PARAM_IN>,
+ RangedParam<"offload_platform_handle_t*", "phPlatforms",
+ "Array of handle of platforms. If NumEntries is"
+ "less than the number of platforms available, then offloadPlatformGet"
+ "shall only retrieve that number of platforms.",
+ PARAM_OUT_OPTIONAL, Range<"0", "NumEntries">>,
+ Param<"uint32_t*",
+ "pNumPlatforms", "returns the total number of platforms available.",
+ PARAM_OUT_OPTIONAL>
+ ];
+ let returns = [
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ "`NumEntries == 0 && phPlatforms != NULL`"
+ ]>
+ ];
+}
+
+def : Enum {
+ let name = "offload_platform_info_t";
+ let desc = "Supported platform info";
+ let is_typed = 1;
+ let etors = [
+ TaggedEtor<"NAME", "char[]", "The string denoting name of the platform. The size of the info needs to be dynamically queried.">,
+ TaggedEtor<"VENDOR_NAME", "char[]","The string denoting name of the vendor of the platform. The size of the info needs to be dynamically queried.">,
+ TaggedEtor<"VERSION", "char[]", "The string denoting the version of the platform. The size of the info needs to be dynamically queried.">,
+ TaggedEtor<"BACKEND", "offload_platform_backend_t", "The backend of the platform. Identifies the native backend adapter implementing this platform.">
+ ];
+}
+
+def : Enum {
+ let name = "offload_platform_backend_t";
+ let desc = "Identifies the native backend of the platform";
+ let etors =[
+ Etor<"UNKNOWN", "The backend is not recognized">,
+ Etor<"CUDA", "The backend is CUDA">,
+ Etor<"AMDGPU", "The backend is AMDGPU">,
+ ];
+}
+
+def : Function {
+ let name = "offloadPlatformGetInfo";
+ let desc = "Retrieves various information about platform";
+ let details = [
+ "The application may call this function from simultaneous threads.",
+ "The implementation of this function should be lock-free."
+ ];
+ let params = [
+ Param<"offload_platform_handle_t", "hPlatform", "handle of the platform", PARAM_IN>,
+ Param<"offload_platform_info_t", "propName", "type of the info to retrieve", PARAM_IN>,
+ Param<"size_t", "propSize", "the number of bytes pointed to by pPlatformInfo.", PARAM_IN>,
+ TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info."
+ "If Size is not equal to or greater to the real number of bytes needed to return the info"
+ "then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is not used.", PARAM_OUT_OPTIONAL,
+ TypeInfo<"propName" , "propSize">>,
+ Param<"size_t*", "pPropSizeRet", "pointer to the actual number of bytes being queried by pPlatformInfo.", PARAM_OUT_OPTIONAL>
+ ];
+ let returns = [
+ Return<"OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION", [
+ "If `propName` is not supported by the platform."
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ "`propSize == 0 && pPropValue != NULL`",
+ "If `propSize` is less than the real number of bytes needed to return the info."
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER", [
+ "`propSize != 0 && pPropValue == NULL`",
+ "`pPropValue == NULL && pPropSizeRet == NULL`"
+ ]>,
+ Return<"OFFLOAD_RESULT_ERROR_INVALID_PLATFORM">,
+ Return<"OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES">,
+ Return<"OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY">
+ ];
+}
diff --git a/offload/new-api/API/README.md b/offload/new-api/API/README.md
new file mode 100644
index 00000000000000..ab1db704404f35
--- /dev/null
+++ b/offload/new-api/API/README.md
@@ -0,0 +1,150 @@
+# Offload API definitions
+
+**Note**: This is a work-in-progress. It is loosely based on equivalent
+tooling in Unified Runtime.
+
+The Tablegen files in this directory are used to define the Offload API. They
+are used with the `offload-tblgen` tool to generate API headers, print headers,
+and other implementation details.
+
+The root file is `OffloadAPI.td` - additional `.td` files can be included in
+this file to add them to the API.
+
+## API Objects
+The API consists of a number of objects, which always have a *name* field and
+*description* field, and are one of the following types:
+
+### Function
+Represents an API entry point function. Has a list of returns and parameters.
+Also has fields for details (representing a bullet-point list of
+information about the function that would otherwise be too detailed for the
+description), and analogues (equivalent functions in other APIs).
+
+#### Parameter
+Represents a parameter to a function, has *type*, *name*, and *desc* fields.
+Also has a *flags* field containing flags representing whether the parameter is
+in, out, or optional.
+
+The *type* field is used to infer if the parameter is a pointer or handle type.
+A *handle* type is a pointer to an opaque struct, used to abstract over
+plugin-specific implementation details.
+
+There are two special variants of a *parameter*:
+* **RangedParameter** - Represents a parameter that has a range described by other parameters. Generally these are pointers to an arbitrary number of objects. The range is used for generating validation and printing code. E.g, a range might be between `(0, NumDevices)`
+* **TypeTaggedParameter** - Represents a parameter (usually of `void*` type) that has the type and size of its pointee data described by other function parameters. The type is usually described by a type-tagged enum. This allows functions (e.g. `offloadDeviceGetInfo`) to return data of an arbitrary type.
+
+#### Return
+A return represents a possible return code from the function, and optionally a
+list of conditions in which this value may be returned. The conditions list is
+not expected to be exhaustive. A condition is considered free-form text, but
+if it is wrapped in \`backticks\` then it is treated as literal code
+representing an error condition (e.g. `someParam < 1`). These conditions are
+used to automatically create validation checks by the `offload-tblgen`
+validation generator.
+
+Returns are automatically generated for functions with pointer or handle
+parameters, so API authors do not need to exhaustively add null checks for
+these types of parameters. All functions also get a number of default return
+values automatically.
+
+
+### Struct
+Represents a struct. Contains a list of members, which each have a *type*,
+*name*, and *desc*.
+
+Also optionally takes a *base_class* field. If this is either of the special
+`offload_base_properties_t` or `offload_base_desc_t` structs, then the struct
+will inherit members from those structs. The generated struct does **not** use
+actual C++ inheritance, but instead explicitly has those members copied in,
+which preserves ABI compatibility with C.
+
+### Enum
+Represents a C-style enum. Contains a list of `etor` values, which have a name
+and description.
+
+A `TaggedEtor` record type also exists which addtionally takes a type. This type
+is used when the enum is used as a parameter to a function with a type-tagged
+function parameter (e.g. `offloadDeviceGetInfo`).
+
+All enums automatically get a `<enum_name>_FORCE_UINT32 = 0x7fffffff` value,
+which forces the underlying type to be uint32.
+
+### Handle
+Represents a pointer to an opaque struct, as described in the Parameter section.
+It does not take any extra fields.
+
+### Typedef
+Represents a typedef, contains only a *value* field.
+
+### Macro
+Represents a C preprocessor `#define`. Contains a *value* field. Optionally
+takes a *condition* field, which allows the macro to be conditionally defined,
+and an *alt_value* field, which represents the value if the condition is false.
+
+Macro arguments are presented in the *name* field (e.g. name = `mymacro(arg)`).
+
+While there may seem little point generating a macro from tablegen, doing this
+allows the entire source of the header file to be generated from the tablegen
+files, rather than requiring a mix of C source and tablegen.
+
+## Generation
+
+### API header
+```
+./offload-tblgen -I <path-to-llvm>/offload/API <path-to-llvm>/offload/API/OffloadAPI.td --gen-api
+```
+The comments in the generated header are in Doxygen format, although
+generating documentation from them hasn't been implemented yet.
+
+The entirety of this header is generated by Tablegen, rather than having a predefined header file that includes one or more `.inc` files. This is because this header is expected to be part of the installation and distributed to end-users, so should be self-contained.
+
+### Entry Points
+```
+./offload-tblgen -I <path-to-llvm>/offload/API <path-to-llvm>/offload/API/OffloadAPI.td --gen-entry-points
+```
+These functions form the actual Offload interface, and are wrappers over the
+functions that contain the actual implementation (see
+'Adding a new entry point').
+
+They implement automatically generated validation checks, and tracing of
+function calls with arguments and results. The tracing can be enabled with the
+`OFFLOAD_TRACE` environment variable.
+
+### Implementation function declarations
+```
+./offload-tblgen -I <path-to-llvm>/offload/API <path-to-llvm>/offload/API/OffloadAPI.td --gen-impl-func-decls
+```
+Generates declarations of the implementation of functions of every entry point
+in the API, e.g. `offloadDeviceFoo_impl` for `offloadDeviceFoo`.
+
+### Print header
+```
+./offload-tblgen -I <path-to-llvm>/offload/API <path-to-llvm>/offload/API/OffloadAPI.td --gen-print-header
+```
+This header contains `std::ostream &operator<<(std::ostream&)` definitions for
+various API objects, including function parameters.
+
+As with the API header, it is expected that this header is part of the installed
+package, so it is entirely generated by Tablegen.
+
+For ease of implementation, and since it is not strictly part of the API, this
+is a C++ header file. If a C version is desirable it could be added.
+
+### Future Tablegen backends
+`RecordTypes.hpp` contains wrappers for all of the API object types, which will
+allow more backends to be easily added in future.
+
+## Adding to the API
+
+A new object can be added to the API by adding to one of the existing `.td`
+files. It is also possible to add a new tablegen file to the API by adding it
+to the includes in `OffloadAPI.td`. When the offload target is rebuilt, the
+new definition will be included in the generated files.
+
+### Adding a new entry point
+
+When a new entry point is added (e.g. `offloadDeviceFoo`), the actual entry
+point is automatically generated, which contains validation and tracing code.
+It expects an implementation function (`offloadDeviceFoo_impl`) to be defined,
+which it will call into. The definition of this implementation function should
+be added to `src/offload_impl.cpp`
\ No newline at end of file
diff --git a/offload/new-api/CMakeLists.txt b/offload/new-api/CMakeLists.txt
new file mode 100644
index 00000000000000..c27591d2af3ab3
--- /dev/null
+++ b/offload/new-api/CMakeLists.txt
@@ -0,0 +1,36 @@
+
+set(LLVM_TARGET_DEFINITIONS ${CMAKE_CURRENT_SOURCE_DIR}/API/OffloadAPI.td)
+list(APPEND LLVM_TABLEGEN_FLAGS -I ${CMAKE_CURRENT_SOURCE_DIR}/API)
+
+tablegen(OFFLOAD offload_api.h -gen-api)
+tablegen(OFFLOAD offload_funcs.inc -gen-func-names)
+tablegen(OFFLOAD offload_impl_func_decls.inc -gen-impl-func-decls)
+tablegen(OFFLOAD offload_entry_points.inc -gen-entry-points)
+tablegen(OFFLOAD offload_print.hpp -gen-print-header)
+
+
+add_public_tablegen_target(OffloadHeaderGen)
+
+add_llvm_library(offload_new SHARED
+ src/offload_lib.cpp
+ src/offload_impl.cpp
+ DEPENDS OffloadHeaderGen
+ LINK_LIBS omptarget omptarget.rtl.cuda omptarget.rtl.amdgpu)
+
+target_include_directories(offload_new PUBLIC
+ ${CMAKE_CURRENT_BINARY_DIR}
+ ${CMAKE_CURRENT_BINARY_DIR}/../include
+ ${CMAKE_CURRENT_SOURCE_DIR}/../include
+ ${CMAKE_CURRENT_SOURCE_DIR}/../plugins-nextgen/common/include)
+
+# foreach(plugin IN LISTS LIBOMPTARGET_PLUGINS_TO_BUILD)
+# target_link_libraries(offload_new PRIVATE omptarget.rtl.${plugin})
+# endforeach()
+
+set_target_properties(offload_new PROPERTIES
+ POSITION_INDEPENDENT_CODE ON
+ INSTALL_RPATH "$ORIGIN"
+ BUILD_RPATH "$ORIGIN:${CMAKE_CURRENT_BINARY_DIR}/..")
+install(TARGETS offload_new LIBRARY COMPONENT offload_new DESTINATION "${OFFLOAD_INSTALL_LIBDIR}")
+
+install(FILES ${CMAKE_CURRENT_BINARY_DIR}/offload_api.h DESTINATION ${CMAKE_INSTALL_PREFIX}/include/offload)
diff --git a/offload/new-api/README.md b/offload/new-api/README.md
new file mode 100644
index 00000000000000..95c9bf54d7badd
--- /dev/null
+++ b/offload/new-api/README.md
@@ -0,0 +1,8 @@
+# Offload New API
+
+This directory contains the implementation of the experimental work-in-progress
+new API for Offload. It builds on top of the existing plugin implementations but
+provides a single level of abstraction suitable for runtimes for languages other
+than OpenMP to be built on top of.
+
+See the [API definition readme](API/README.md) for implementation details.
\ No newline at end of file
diff --git a/offload/new-api/src/helpers.hpp b/offload/new-api/src/helpers.hpp
new file mode 100644
index 00000000000000..66148d9f210083
--- /dev/null
+++ b/offload/new-api/src/helpers.hpp
@@ -0,0 +1,96 @@
+//===- helpers.hpp- GetInfo return helpers for the new LLVM/Offload API ---===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+//
+// The getInfo*/ReturnHelper facilities provide shortcut way of writing return
+// data + size for the various getInfo APIs. Based on the equivalent
+// implementations in Unified Runtime.
+//
+//===----------------------------------------------------------------------===//
+
+#include "offload_api.h"
+
+#include <cstring>
+
+
+template <typename T, typename Assign>
+offload_result_t getInfoImpl(size_t ParamValueSize, void *ParamValue,
+ size_t *ParamValueSizeRet, T Value,
+ size_t ValueSize, Assign &&AssignFunc) {
+ if (!ParamValue && !ParamValueSizeRet) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ }
+
+ if (ParamValue != nullptr) {
+ if (ParamValueSize < ValueSize) {
+ return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ }
+ AssignFunc(ParamValue, Value, ValueSize);
+ }
+
+ if (ParamValueSizeRet != nullptr) {
+ *ParamValueSizeRet = ValueSize;
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+template <typename T>
+offload_result_t getInfo(size_t ParamValueSize, void *ParamValue,
+ size_t *ParamValueSizeRet, T Value) {
+ auto Assignment = [](void *ParamValue, T Value, size_t) {
+ *static_cast<T *>(ParamValue) = Value;
+ };
+
+ return getInfoImpl(ParamValueSize, ParamValue, ParamValueSizeRet, Value,
+ sizeof(T), Assignment);
+}
+
+template <typename T>
+offload_result_t getInfoArray(size_t array_length, size_t ParamValueSize,
+ void *ParamValue, size_t *ParamValueSizeRet,
+ const T *Value) {
+ return getInfoImpl(ParamValueSize, ParamValue, ParamValueSizeRet, Value,
+ array_length * sizeof(T), memcpy);
+}
+
+template <>
+inline offload_result_t
+getInfo<const char *>(size_t ParamValueSize, void *ParamValue,
+ size_t *ParamValueSizeRet, const char *Value) {
+ return getInfoArray(strlen(Value) + 1, ParamValueSize, ParamValue,
+ ParamValueSizeRet, Value);
+}
+
+class ReturnHelper {
+public:
+ ReturnHelper(size_t ParamValueSize, void *ParamValue,
+ size_t *ParamValueSizeRet)
+ : ParamValueSize(ParamValueSize), ParamValue(ParamValue),
+ ParamValueSizeRet(ParamValueSizeRet) {}
+
+ // A version where in/out info size is represented by a single pointer
+ // to a value which is updated on return
+ ReturnHelper(size_t *ParamValueSize, void *ParamValue)
+ : ParamValueSize(*ParamValueSize), ParamValue(ParamValue),
+ ParamValueSizeRet(ParamValueSize) {}
+
+ // Scalar return Value
+ template <class T> offload_result_t operator()(const T &t) {
+ return getInfo(ParamValueSize, ParamValue, ParamValueSizeRet, t);
+ }
+
+ // Array return Value
+ template <class T> offload_result_t operator()(const T *t, size_t s) {
+ return getInfoArray(s, ParamValueSize, ParamValue, ParamValueSizeRet, t);
+ }
+
+protected:
+ size_t ParamValueSize;
+ void *ParamValue;
+ size_t *ParamValueSizeRet;
+};
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
new file mode 100644
index 00000000000000..165f2264ef7daa
--- /dev/null
+++ b/offload/new-api/src/offload_impl.cpp
@@ -0,0 +1,187 @@
+//===- offload_impl.cpp - Implementation of the new LLVM/Offload API ------===//
+//
+// 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 contains the definitions of the new LLVM/Offload API entry points. See
+// new-api/API/README.md for more information.
+//
+//===----------------------------------------------------------------------===//
+
+#include "PluginManager.h"
+#include "helpers.hpp"
+#include "llvm/Support/FormatVariadic.h"
+#include <offload_api.h>
+
+#include <mutex>
+
+using namespace llvm;
+using namespace llvm::omp::target::plugin;
+
+// Handle type definitions. Ideally these would be 1:1 with the plugins
+struct offload_device_handle_t_ {
+ int DeviceNum;
+ GenericDeviceTy &Device;
+};
+
+struct offload_platform_handle_t_ {
+ std::unique_ptr<GenericPluginTy> Plugin;
+ std::vector<offload_device_handle_t_> Devices;
+};
+
+static std::vector<offload_platform_handle_t_> Platforms;
+
+// Every plugin exports this method to create an instance of the plugin type.
+#define PLUGIN_TARGET(Name) extern "C" GenericPluginTy *createPlugin_##Name();
+#include "Shared/Targets.def"
+
+void initPlugins() {
+ // Attempt to create an instance of each supported plugin.
+#define PLUGIN_TARGET(Name) \
+ do { \
+ Platforms.emplace_back(offload_platform_handle_t_{ \
+ std::unique_ptr<GenericPluginTy>(createPlugin_##Name()), {}}); \
+ } while (false);
+#include "Shared/Targets.def"
+
+ // Preemptively initialize all devices in the plugin so we can just return
+ // them from deviceGet
+ for (auto &Platform : Platforms) {
+ auto Err = Platform.Plugin->init();
+ [[maybe_unused]] std::string InfoMsg = toString(std::move(Err));
+ for (auto DevNum = 0; DevNum < Platform.Plugin->number_of_devices();
+ DevNum++) {
+ if (Platform.Plugin->init_device(DevNum) == OFFLOAD_SUCCESS) {
+ Platform.Devices.emplace_back(offload_device_handle_t_{
+ DevNum, Platform.Plugin->getDevice(DevNum)});
+ }
+ }
+ }
+}
+
+offload_result_t offloadPlatformGet_impl(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms) {
+ // It is expected that offloadPlatformGet is the first function to be called.
+ // In future it may make sense to have a specific entry point for Offload
+ // initialization, or expose explicit initialization of plugins.
+ static std::once_flag InitFlag;
+ std::call_once(InitFlag, initPlugins);
+
+ if (NumEntries > Platforms.size()) {
+ return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ }
+
+ if (phPlatforms) {
+ for (uint32_t PlatformIndex = 0; PlatformIndex < NumEntries;
+ PlatformIndex++) {
+ phPlatforms[PlatformIndex] = &Platforms[PlatformIndex];
+ }
+ }
+
+ if (pNumPlatforms) {
+ *pNumPlatforms = Platforms.size();
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+offload_result_t
+offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName, size_t propSize,
+ void *pPropValue, size_t *pPropSizeRet) {
+ ReturnHelper ReturnValue(propSize, pPropValue, pPropSizeRet);
+
+ switch (propName) {
+ case OFFLOAD_PLATFORM_INFO_NAME:
+ return ReturnValue(hPlatform->Plugin->getName());
+ case OFFLOAD_PLATFORM_INFO_VENDOR_NAME:
+ // TODO: Implement this
+ return ReturnValue("Unknown platform vendor");
+ case OFFLOAD_PLATFORM_INFO_VERSION: {
+ // TODO: Implement this
+ return ReturnValue("v0.0.0");
+ }
+ case OFFLOAD_PLATFORM_INFO_BACKEND: {
+ auto PluginName = hPlatform->Plugin->getName();
+ if (PluginName == StringRef("CUDA")) {
+ return ReturnValue(OFFLOAD_PLATFORM_BACKEND_CUDA);
+ } else if (PluginName == StringRef("AMDGPU")) {
+ return ReturnValue(OFFLOAD_PLATFORM_BACKEND_AMDGPU);
+ } else {
+ return ReturnValue(OFFLOAD_PLATFORM_BACKEND_UNKNOWN);
+ }
+ }
+ default:
+ return OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION;
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+offload_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
+ offload_device_type_t,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices) {
+
+ if (phDevices) {
+ for (uint32_t DeviceIndex = 0; DeviceIndex < NumEntries; DeviceIndex++) {
+ phDevices[DeviceIndex] = &(hPlatform->Devices[DeviceIndex]);
+ }
+ }
+
+ if (pNumDevices) {
+ *pNumDevices = static_cast<uint32_t>(hPlatform->Devices.size());
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+offload_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t propSize, void *pPropValue,
+ size_t *pPropSizeRet) {
+
+ ReturnHelper ReturnValue(propSize, pPropValue, pPropSizeRet);
+
+ InfoQueueTy DevInfo;
+ if (auto Err = hDevice->Device.obtainInfoImpl(DevInfo))
+ return OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES;
+
+ // Find the info if it exists under any of the given names
+ auto GetInfo = [&DevInfo](std::vector<std::string> Names) {
+ for (auto Name : Names) {
+ auto InfoKeyMatches = [&](const InfoQueueTy::InfoQueueEntryTy &info) {
+ return info.Key == Name;
+ };
+ auto Item = std::find_if(DevInfo.getQueue().begin(),
+ DevInfo.getQueue().end(), InfoKeyMatches);
+
+ if (Item != std::end(DevInfo.getQueue())) {
+ return Item->Value;
+ }
+ }
+
+ return std::string("");
+ };
+
+ switch (propName) {
+ case OFFLOAD_DEVICE_INFO_TYPE:
+ return ReturnValue(OFFLOAD_DEVICE_TYPE_GPU);
+ case OFFLOAD_DEVICE_INFO_NAME:
+ return ReturnValue(GetInfo({"Device Name"}).c_str());
+ case OFFLOAD_DEVICE_INFO_VENDOR:
+ return ReturnValue(GetInfo({"Vendor Name"}).c_str());
+ case OFFLOAD_DEVICE_INFO_DRIVER_VERSION:
+ return ReturnValue(
+ GetInfo({"CUDA Driver Version", "HSA Runtime Version"}).c_str());
+ default:
+ return OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION;
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
diff --git a/offload/new-api/src/offload_lib.cpp b/offload/new-api/src/offload_lib.cpp
new file mode 100644
index 00000000000000..b2452b2cdab950
--- /dev/null
+++ b/offload/new-api/src/offload_lib.cpp
@@ -0,0 +1,23 @@
+//===- offload_lib.cpp - Entry points for the new LLVM/Offload API --------===//
+//
+// 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 pulls in the tablegen'd API entry point functions.
+//
+//===----------------------------------------------------------------------===//
+
+#include <offload_api.h>
+#include <offload_print.hpp>
+
+#include <iostream>
+
+// Pull in the declarations for the implementation funtions. The actual entry
+// points in this file wrap these.
+#include "offload_impl_func_decls.inc"
+
+// Pull in the tablegen'd entry point definitions.
+#include "offload_entry_points.inc"
diff --git a/offload/plugins-nextgen/common/include/PluginInterface.h b/offload/plugins-nextgen/common/include/PluginInterface.h
index 41cc0f286a581f..4cfe6cbf9ffe7f 100644
--- a/offload/plugins-nextgen/common/include/PluginInterface.h
+++ b/offload/plugins-nextgen/common/include/PluginInterface.h
@@ -124,6 +124,7 @@ enum InfoLevelKind { InfoLevel1 = 1, InfoLevel2, InfoLevel3 };
/// we use the level to determine the indentation of the key-value property at
/// printing time. See the enum InfoLevelKind for the list of accepted levels.
class InfoQueueTy {
+public:
struct InfoQueueEntryTy {
std::string Key;
std::string Value;
@@ -131,6 +132,8 @@ class InfoQueueTy {
uint64_t Level;
};
+
+private:
std::deque<InfoQueueEntryTy> Queue;
public:
@@ -153,6 +156,10 @@ class InfoQueueTy {
Queue.push_back({Key, Value, Units, L});
}
+ const std::deque<InfoQueueEntryTy> &getQueue() const {
+ return Queue;
+ }
+
/// Print all info entries added to the queue.
void print() const {
// We print four spances for each level.
diff --git a/offload/tools/offload-tblgen/APIGen.cpp b/offload/tools/offload-tblgen/APIGen.cpp
new file mode 100644
index 00000000000000..6fdf3025f25bb5
--- /dev/null
+++ b/offload/tools/offload-tblgen/APIGen.cpp
@@ -0,0 +1,196 @@
+//===- offload-tblgen/APIGen.cpp - Tablegen backend for Offload header ----===//
+//
+// 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 is a Tablegen backend that produces the contents of the Offload API
+// header. The generated comments are Doxygen compatible.
+//
+//===----------------------------------------------------------------------===//
+
+#include "llvm/ADT/StringExtras.h"
+#include "llvm/Support/FormatVariadic.h"
+#include "llvm/TableGen/Record.h"
+#include "llvm/TableGen/TableGenBackend.h"
+
+#include "GenCommon.hpp"
+#include "RecordTypes.hpp"
+
+using namespace llvm;
+using namespace offload::tblgen;
+
+// Produce a possibly multi-line comment from the input string
+static std::string MakeComment(StringRef in) {
+ std::string out = "";
+ size_t LineStart = 0;
+ size_t LineBreak = 0;
+ while (LineBreak < in.size()) {
+ LineBreak = in.find_first_of("\n", LineStart);
+ if (LineBreak - LineStart <= 1) {
+ break;
+ }
+ out += std::string("\t///< ") +
+ in.substr(LineStart, LineBreak - LineStart).str() + "\n";
+ LineStart = LineBreak + 1;
+ }
+
+ return out;
+}
+
+static void ProcessHandle(const HandleRec &H, raw_ostream &OS) {
+ OS << CommentsHeader;
+ OS << formatv("/// @brief {0}\n", H.getDesc());
+ OS << formatv("typedef struct {0}_ *{0};\n", H.getName());
+}
+
+static void ProcessTypedef(const TypedefRec &T, raw_ostream &OS) {
+ OS << CommentsHeader;
+ OS << formatv("/// @brief {0}\n", T.getDesc());
+ OS << formatv("typedef {0} {1};\n", T.getValue(), T.getName());
+}
+
+static void ProcessMacro(const MacroRec &M, raw_ostream &OS) {
+ OS << CommentsHeader;
+ OS << formatv("#ifndef {0}\n", M.getName());
+ if (auto Condition = M.getCondition()) {
+ OS << formatv("#if {0}\n", *Condition);
+ }
+ OS << "/// @brief " << M.getDesc() << "\n";
+ OS << formatv("#define {0} {1}\n", M.getNameWithArgs(), M.getValue());
+ if (auto AltValue = M.getAltValue()) {
+ OS << "#else\n";
+ OS << formatv("#define {0} {1}\n", M.getNameWithArgs(), *AltValue);
+ }
+ if (auto Condition = M.getCondition()) {
+ OS << formatv("#endif // {0}\n", *Condition);
+ }
+ OS << formatv("#endif // {0}\n", M.getName());
+}
+
+static void ProcessFunction(const FunctionRec &F, raw_ostream &OS) {
+ OS << CommentsHeader;
+ OS << formatv("/// @brief {0}\n", F.getDesc());
+ OS << CommentsBreak;
+
+ OS << "/// @details\n";
+ for (auto &Detail : F.getDetails()) {
+ OS << formatv("/// - {0}\n", Detail);
+ }
+ OS << CommentsBreak;
+
+ // Emit analogue remarks
+ auto Analogues = F.getAnalogues();
+ if (!Analogues.empty()) {
+ OS << "/// @remarks\n/// _Analogues_\n";
+ for (auto &Analogue : Analogues) {
+ OS << formatv("/// - **{0}**\n", Analogue);
+ }
+ OS << CommentsBreak;
+ }
+
+ OS << "/// @returns\n";
+ auto Returns = F.getReturns();
+ for (auto &Ret : Returns) {
+ OS << formatv("/// - ::{0}\n", Ret.getValue());
+ auto RetConditions = Ret.getConditions();
+ for (auto &RetCondition : RetConditions) {
+ OS << formatv("/// + {0}\n", RetCondition);
+ }
+ }
+
+ OS << formatv("{0}_APIEXPORT {1}_result_t {0}_APICALL ", PrefixUpper,
+ PrefixLower);
+ OS << F.getName();
+ OS << "(\n";
+ auto Params = F.getParams();
+ for (auto &Param : Params) {
+ OS << " " << Param.getType() << " " << Param.getName();
+ if (Param != Params.back()) {
+ OS << ", ";
+ } else {
+ OS << " ";
+ }
+ OS << MakeParamComment(Param) << "\n";
+ }
+ OS << ");\n\n";
+}
+
+static void ProcessEnum(const EnumRec &Enum, raw_ostream &OS) {
+ OS << CommentsHeader;
+ OS << formatv("/// @brief {0}\n", Enum.getDesc());
+ OS << formatv("typedef enum {0} {{\n", Enum.getName());
+
+ uint32_t EtorVal = 0;
+ for (const auto &EnumVal : Enum.getValues()) {
+ auto Desc = MakeComment(EnumVal.getDesc());
+ OS << formatv(TAB_1 "{0}_{1} = {2}, {3}", Enum.getEnumValNamePrefix(),
+ EnumVal.getName(), EtorVal++, Desc);
+ }
+
+ // Add force uint32 val
+ OS << formatv(TAB_1 "/// @cond\n" TAB_1
+ "{0}_FORCE_UINT32 = 0x7fffffff\n" TAB_1
+ "/// @endcond\n\n",
+ Enum.getEnumValNamePrefix());
+
+ OS << formatv("} {0};\n", Enum.getName());
+}
+
+static void ProcessStruct(const StructRec &Struct, raw_ostream &OS) {
+ OS << CommentsHeader;
+ OS << formatv("/// @brief {0}\n", Struct.getDesc());
+ OS << formatv("typedef struct {0} {{\n", Struct.getName());
+
+ for (const auto &Member : Struct.getMembers()) {
+ OS << formatv(TAB_1 "{0} {1}; {2}", Member.getType(), Member.getName(),
+ MakeComment(Member.getDesc()));
+ }
+
+ OS << formatv("} {0};\n\n", Struct.getName());
+}
+
+static void ProcessFuncParamStruct(const FunctionRec &Func, raw_ostream &OS) {
+ auto FuncParamStructBegin = R"(
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for {0}
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct {1} {{
+)";
+
+ OS << formatv(FuncParamStructBegin, Func.getName(),
+ Func.getParamStructName());
+ for (const auto &Param : Func.getParams()) {
+ OS << TAB_1 << Param.getType() << "* p" << Param.getName() << ";\n";
+ }
+ OS << formatv("} {0};\n", Func.getParamStructName());
+}
+
+void EmitOffloadAPI(RecordKeeper &Records, raw_ostream &OS) {
+ OS << FileHeader;
+ // Generate main API definitions
+ for (auto *R : Records.getAllDerivedDefinitions("APIObject")) {
+ if (R->isSubClassOf("Macro")) {
+ ProcessMacro(MacroRec{R}, OS);
+ } else if (R->isSubClassOf("Typedef")) {
+ ProcessTypedef(TypedefRec{R}, OS);
+ } else if (R->isSubClassOf("Handle")) {
+ ProcessHandle(HandleRec{R}, OS);
+ } else if (R->isSubClassOf("Function")) {
+ ProcessFunction(FunctionRec{R}, OS);
+ } else if (R->isSubClassOf("Enum")) {
+ ProcessEnum(EnumRec{R}, OS);
+ } else if (R->isSubClassOf("Struct")) {
+ ProcessStruct(StructRec{R}, OS);
+ }
+ }
+
+ // Generate auxiliary definitions (func param structs etc)
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ ProcessFuncParamStruct(FunctionRec{R}, OS);
+ }
+
+ OS << FileFooter;
+}
diff --git a/offload/tools/offload-tblgen/CMakeLists.txt b/offload/tools/offload-tblgen/CMakeLists.txt
new file mode 100644
index 00000000000000..52986cbbaa9187
--- /dev/null
+++ b/offload/tools/offload-tblgen/CMakeLists.txt
@@ -0,0 +1,24 @@
+##===----------------------------------------------------------------------===##
+#
+# 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
+#
+##===----------------------------------------------------------------------===##
+include(TableGen)
+
+add_tablegen(offload-tblgen OFFLOAD
+ EXPORT OFFLOAD
+ APIGen.cpp
+ EntryPointGen.cpp
+ FuncsGen.cpp
+ GenCommon.hpp
+ Generators.hpp
+ offload-tblgen.cpp
+ PrintGen.cpp
+ RecordTypes.hpp
+ )
+
+set(OFFLOAD_TABLEGEN_EXE "${OFFLOAD_TABLEGEN_EXE}" CACHE INTERNAL "")
+set(OFFLOAD_TABLEGEN_TARGET "${OFFLOAD_TABLEGEN_TARGET}" CACHE INTERNAL "")
+
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
new file mode 100644
index 00000000000000..b89592187639e3
--- /dev/null
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -0,0 +1,86 @@
+//===- offload-tblgen/EntryPointGen.cpp - Tablegen backend for Offload ----===//
+//
+// 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 is a Tablegen backend that produces the actual entry points for the
+// Offload API. It serves as a place to integrate functionality like tracing
+// and validation before dispatching to the actual implementations.
+//===----------------------------------------------------------------------===//
+
+#include "llvm/Support/FormatVariadic.h"
+#include "llvm/TableGen/Record.h"
+
+#include "GenCommon.hpp"
+#include "RecordTypes.hpp"
+
+using namespace llvm;
+using namespace offload::tblgen;
+
+static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
+ OS << CommentsHeader;
+ // Emit preamble
+ OS << formatv("{1}_APIEXPORT {0}_result_t {1}_APICALL {2}(\n ", PrefixLower,
+ PrefixUpper, F.getName());
+ // Emit arguments
+ std::string ParamNameList = "";
+ for (auto &Param : F.getParams()) {
+ OS << Param.getType() << " " << Param.getName();
+ if (Param != F.getParams().back()) {
+ OS << ", ";
+ }
+ ParamNameList += Param.getName().str() + ", ";
+ }
+ OS << ") {\n";
+
+ OS << TAB_1 "if (true /*enableParameterValidation*/) {\n";
+
+ // Emit validation checks
+ for (const auto &Return : F.getReturns()) {
+ for (auto &Condition : Return.getConditions()) {
+ if (Condition.starts_with("`") && Condition.ends_with("`")) {
+ auto ConditionString = Condition.substr(1, Condition.size() - 2);
+ OS << formatv(TAB_2 "if ({0}) {{\n", ConditionString);
+ OS << formatv(TAB_3 "return {0};\n", Return.getValue());
+ OS << TAB_2 "}\n\n";
+ }
+ }
+ }
+ OS << " }\n\n";
+
+ // Emit pre-call prints
+ OS << TAB_1 "if (std::getenv(\"OFFLOAD_TRACE\")) {\n";
+ OS << formatv(TAB_2 "std::cout << \"---> {0}\";\n", F.getName());
+ OS << TAB_1 "}\n\n";
+
+ // Perform actual function call
+ ParamNameList = ParamNameList.substr(0, ParamNameList.size() - 2);
+ OS << formatv(TAB_1 "{0}_result_t result = {1}_impl({2});\n\n", PrefixLower,
+ F.getName(), ParamNameList);
+
+ // Emit post-call prints
+ OS << TAB_1 "if (std::getenv(\"OFFLOAD_TRACE\")) {\n";
+ OS << formatv(TAB_2 "{0} Params = {{ ", F.getParamStructName());
+ for (const auto &Param : F.getParams()) {
+ OS << "&" << Param.getName();
+ if (Param != F.getParams().back()) {
+ OS << ", ";
+ }
+ }
+ OS << formatv("};\n");
+ OS << TAB_2 "std::cout << \"(\" << &Params << \")\";\n";
+ OS << TAB_2 "std::cout << \"-> \" << result << \"\\n\";\n";
+ OS << TAB_1 "}\n";
+
+ OS << TAB_1 "return result;\n";
+ OS << "}\n";
+}
+
+void EmitOffloadEntryPoints(RecordKeeper &Records, raw_ostream &OS) {
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ EmitEntryPointFunc(FunctionRec{R}, OS);
+ }
+}
diff --git a/offload/tools/offload-tblgen/FuncsGen.cpp b/offload/tools/offload-tblgen/FuncsGen.cpp
new file mode 100644
index 00000000000000..4dc74047ac49d8
--- /dev/null
+++ b/offload/tools/offload-tblgen/FuncsGen.cpp
@@ -0,0 +1,54 @@
+//===- offload-tblgen/APIGen.cpp - Tablegen backend for Offload validation ===//
+//
+// 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 is a Tablegen backend that produces validation functions for the Offload
+// API entry point functions.
+//
+//===----------------------------------------------------------------------===//
+
+#include "llvm/Support/FormatVariadic.h"
+#include "llvm/TableGen/Record.h"
+
+#include "GenCommon.hpp"
+#include "RecordTypes.hpp"
+
+using namespace llvm;
+using namespace offload::tblgen;
+
+// Emit a list of just the API function names
+void EmitOffloadFuncNames(RecordKeeper &Records, raw_ostream &OS) {
+ OS << R"(
+#ifndef OFFLOAD_FUNC
+#error Please define the macro OFFLOAD_FUNC(Function)
+#endif
+
+)";
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ FunctionRec FR{R};
+ OS << formatv("OFFLOAD_FUNC({0})", FR.getName()) << "\n";
+ }
+
+ OS << "\n#undef OFFLOAD_FUNC\n";
+}
+
+void EmitOffloadImplFuncDecls(RecordKeeper &Records, raw_ostream &OS) {
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ FunctionRec F{R};
+ OS << formatv("{0}_result_t {1}_impl(", PrefixLower, F.getName());
+ auto Params = F.getParams();
+ for (auto &Param : Params) {
+ OS << Param.getType() << " " << Param.getName();
+ if (Param != Params.back()) {
+ OS << ", ";
+ } else {
+ OS << ");";
+ }
+ }
+ OS << "\n";
+ }
+}
diff --git a/offload/tools/offload-tblgen/GenCommon.hpp b/offload/tools/offload-tblgen/GenCommon.hpp
new file mode 100644
index 00000000000000..fb24ef06f9ec8f
--- /dev/null
+++ b/offload/tools/offload-tblgen/GenCommon.hpp
@@ -0,0 +1,58 @@
+//===- offload-tblgen/GenCommon.cpp - Common defs for Offload generators --===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#pragma once
+
+#include "RecordTypes.hpp"
+#include "llvm/Support/FormatVariadic.h"
+
+// Having inline bits of tabbed code is hard to read, provide some definitions
+// so we can keep things tidier
+#define TAB_1 " "
+#define TAB_2 " "
+#define TAB_3 " "
+#define TAB_4 " "
+#define TAB_5 " "
+
+constexpr auto FileHeader = R"(
+// Auto-generated file, do not manually edit.
+
+#pragma once
+
+#include <stddef.h>
+#include <stdint.h>
+
+#if defined(__cplusplus)
+extern "C" {
+#endif
+
+)";
+
+constexpr auto FileFooter = R"(
+#if defined(__cplusplus)
+} // extern "C"
+#endif
+
+)";
+
+constexpr auto CommentsHeader = R"(
+///////////////////////////////////////////////////////////////////////////////
+)";
+
+constexpr auto CommentsBreak = "///\n";
+
+constexpr auto PrefixLower = "offload";
+constexpr auto PrefixUpper = "OFFLOAD";
+
+inline std::string
+MakeParamComment(const llvm::offload::tblgen::ParamRec &Param) {
+ return llvm::formatv("///< {0}{1}{2} {3}", (Param.isIn() ? "[in]" : ""),
+ (Param.isOut() ? "[out]" : ""),
+ (Param.isOpt() ? "[optional]" : ""), Param.getDesc());
+}
+
diff --git a/offload/tools/offload-tblgen/Generators.hpp b/offload/tools/offload-tblgen/Generators.hpp
new file mode 100644
index 00000000000000..4b300f52ddf84d
--- /dev/null
+++ b/offload/tools/offload-tblgen/Generators.hpp
@@ -0,0 +1,18 @@
+//===- offload-tblgen/Generators.hpp - Offload generator declarations -----===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#pragma once
+
+#include "llvm/TableGen/Record.h"
+
+void EmitOffloadAPI(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
+void EmitOffloadFuncNames(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
+void EmitOffloadImplFuncDecls(llvm::RecordKeeper &Records,
+ llvm::raw_ostream &OS);
+void EmitOffloadEntryPoints(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
+void EmitOffloadPrintHeader(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
diff --git a/offload/tools/offload-tblgen/PrintGen.cpp b/offload/tools/offload-tblgen/PrintGen.cpp
new file mode 100644
index 00000000000000..0c47695eca12b6
--- /dev/null
+++ b/offload/tools/offload-tblgen/PrintGen.cpp
@@ -0,0 +1,206 @@
+//===- offload-tblgen/APIGen.cpp - Tablegen backend for Offload printing --===//
+//
+// 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 is a Tablegen backend that produces print functions for the Offload API
+// entry point functions.
+//
+//===----------------------------------------------------------------------===//
+
+#include "llvm/Support/FormatVariadic.h"
+#include "llvm/TableGen/Record.h"
+
+#include "GenCommon.hpp"
+#include "RecordTypes.hpp"
+
+using namespace llvm;
+using namespace offload::tblgen;
+
+constexpr auto PrintEnumHeader =
+ R"(///////////////////////////////////////////////////////////////////////////////
+/// @brief Print operator for the {0} type
+/// @returns std::ostream &
+)";
+
+constexpr auto PrintTaggedEnumHeader =
+ R"(///////////////////////////////////////////////////////////////////////////////
+/// @brief Print type-tagged {0} enum value
+/// @returns std::ostream &
+)";
+
+static void ProcessEnum(const EnumRec &Enum, raw_ostream &OS) {
+ OS << formatv(PrintEnumHeader, Enum.getName());
+ OS << formatv(
+ "inline std::ostream &operator<<(std::ostream &os, enum {0} value) "
+ "{{\n" TAB_1 "switch (value) {{\n",
+ Enum.getName());
+
+ for (const auto &Val : Enum.getValues()) {
+ auto Name = Enum.getEnumValNamePrefix() + "_" + Val.getName();
+ OS << formatv(TAB_1 "case {0}:\n", Name);
+ OS << formatv(TAB_2 "os << \"{0}\";\n", Name);
+ OS << formatv(TAB_2 "break;\n");
+ }
+
+ OS << TAB_1 "default:\n" TAB_2 "os << \"unknown enumerator\";\n" TAB_2
+ "break;\n" TAB_1 "}\n" TAB_1 "return os;\n}\n\n";
+
+ if (!Enum.isTyped()) {
+ return;
+ }
+
+ OS << formatv(PrintTaggedEnumHeader, Enum.getName());
+
+ OS << formatv(R"""(template <>
+inline void printTagged(std::ostream &os, const void *ptr, {0} value, size_t size) {{
+ if (ptr == NULL) {{
+ printPtr(os, ptr);
+ return;
+ }
+
+ switch (value) {{
+)""",
+ Enum.getName());
+
+ for (const auto &Val : Enum.getValues()) {
+ auto Name = Enum.getEnumValNamePrefix() + "_" + Val.getName();
+ auto Type = Val.getTaggedType();
+ OS << formatv(TAB_1 "case {0}: {{\n", Name);
+ // Special case for strings
+ if (Type == "char[]") {
+ OS << formatv(TAB_2 "printPtr(os, (const char*) ptr);\n");
+ } else {
+ OS << formatv(TAB_2 "const {0} * const tptr = (const {0} * const)ptr;\n",
+ Type);
+ // TODO: Handle other cases here
+ OS << TAB_2 "os << (const void *)tptr << \" (\";\n";
+ if (Type.ends_with("*")) {
+ OS << TAB_2 "os << printPtr(os, tptr);\n";
+ } else {
+ OS << TAB_2 "os << *tptr;\n";
+ }
+ OS << TAB_2 "os << \")\";\n";
+ }
+ OS << formatv(TAB_2 "break;\n" TAB_1 "}\n");
+ }
+
+ OS << TAB_1 "default:\n" TAB_2 "os << \"unknown enumerator\";\n" TAB_2
+ "break;\n" TAB_1 "}\n";
+
+ OS << "}\n";
+}
+
+static void EmitFunctionParamStructPrint(const FunctionRec &Func,
+ raw_ostream &OS) {
+ OS << formatv(R"(
+inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct {0} *params) {{
+)",
+ Func.getParamStructName());
+
+ for (const auto &Param : Func.getParams()) {
+ OS << formatv(TAB_1 "os << \".{0} = \";\n", Param.getName());
+ if (auto Range = Param.getRange()) {
+ OS << formatv(TAB_1 "os << \"{{\";\n");
+ OS << formatv(TAB_1 "for (size_t i = {0}; i < *params->p{1}; i++){{\n",
+ Range->first, Range->second);
+ OS << TAB_2 "if (i > 0) {\n";
+ OS << TAB_3 " os << \", \";\n";
+ OS << TAB_2 "}\n";
+ OS << formatv(TAB_2 "printPtr(os, (*params->p{0})[i]);\n",
+ Param.getName());
+ OS << formatv(TAB_1 "}\n");
+ OS << formatv(TAB_1 "os << \"}\";\n");
+ } else if (auto TypeInfo = Param.getTypeInfo()) {
+ OS << formatv(
+ TAB_1
+ "printTagged(os, *params->p{0}, *params->p{1}, *params->p{2});\n",
+ Param.getName(), TypeInfo->first, TypeInfo->second);
+ } else if (Param.isPointerType() || Param.isHandleType()) {
+ OS << formatv(TAB_1 "printPtr(os, *params->p{0});\n", Param.getName());
+ } else {
+ OS << formatv(TAB_1 "os << *params->p{0};\n", Param.getName());
+ }
+ if (Param != Func.getParams().back()) {
+ OS << TAB_1 "os << \", \";\n";
+ }
+ }
+
+ OS << TAB_1 "return os;\n}\n";
+}
+
+void EmitOffloadPrintHeader(RecordKeeper &Records, raw_ostream &OS) {
+ OS << R"""(
+// Auto-generated file, do not manually edit.
+
+#pragma once
+
+#include <offload_api.h>
+#include <ostream>
+
+
+template <typename T> inline offload_result_t printPtr(std::ostream &os, const T *ptr);
+template <typename T> inline void printTagged(std::ostream &os, const void *ptr, T value, size_t size);
+)""";
+
+ // ==========
+ OS << "template <typename T> struct is_handle : std::false_type {};\n";
+ for (auto *R : Records.getAllDerivedDefinitions("Handle")) {
+ HandleRec H{R};
+ OS << formatv("template <> struct is_handle<{0}> : std::true_type {{};\n",
+ H.getName());
+ }
+ OS << "template <typename T> inline constexpr bool is_handle_v = "
+ "is_handle<T>::value;\n";
+ // =========
+
+ // Forward declare the operator<< overloads so their implementations can
+ // use each other.
+ OS << "\n";
+ for (auto *R : Records.getAllDerivedDefinitions("Enum")) {
+ OS << formatv(
+ "inline std::ostream &operator<<(std::ostream &os, enum {0} value);\n",
+ EnumRec{R}.getName());
+ }
+ OS << "\n";
+
+ // Create definitions
+ for (auto *R : Records.getAllDerivedDefinitions("Enum")) {
+ EnumRec E{R};
+ ProcessEnum(E, OS);
+ }
+
+ // Emit print functions for the function param structs
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ EmitFunctionParamStructPrint(FunctionRec{R}, OS);
+ }
+
+ OS << R"""(
+///////////////////////////////////////////////////////////////////////////////
+// @brief Print pointer value
+template <typename T> inline offload_result_t printPtr(std::ostream &os, const T *ptr) {
+ if (ptr == nullptr) {
+ os << "nullptr";
+ } else if constexpr (std::is_pointer_v<T>) {
+ os << (const void *)(ptr) << " (";
+ printPtr(os, *ptr);
+ os << ")";
+ } else if constexpr (std::is_void_v<T> || is_handle_v<T *>) {
+ os << (const void *)ptr;
+ } else if constexpr (std::is_same_v<std::remove_cv_t< T >, char>) {
+ os << (const void *)(ptr) << " (";
+ os << ptr;
+ os << ")";
+ } else {
+ os << (const void *)(ptr) << " (";
+ os << *ptr;
+ os << ")";
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+ )""";
+}
diff --git a/offload/tools/offload-tblgen/RecordTypes.hpp b/offload/tools/offload-tblgen/RecordTypes.hpp
new file mode 100644
index 00000000000000..ccd45d472cf770
--- /dev/null
+++ b/offload/tools/offload-tblgen/RecordTypes.hpp
@@ -0,0 +1,227 @@
+//===- offload-tblgen/RecordTypes.cpp - Offload record type wrappers -----===-//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#pragma once
+
+#include <string>
+
+#include "llvm/TableGen/Record.h"
+
+namespace llvm {
+namespace offload {
+namespace tblgen {
+
+class HandleRec {
+public:
+ explicit HandleRec(Record *rec) : rec(rec) {}
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+
+private:
+ Record *rec;
+};
+
+class MacroRec {
+public:
+ explicit MacroRec(Record *rec) : rec(rec) {
+ auto Name = rec->getValueAsString("name");
+ auto OpenBrace = Name.find_first_of("(");
+ nameWithoutArgs = Name.substr(0, OpenBrace);
+ }
+ StringRef getName() const { return nameWithoutArgs; }
+ StringRef getNameWithArgs() const { return rec->getValueAsString("name"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+
+ std::optional<StringRef> getCondition() const {
+ return rec->getValueAsOptionalString("condition");
+ }
+ StringRef getValue() const { return rec->getValueAsString("value"); }
+ std::optional<StringRef> getAltValue() const {
+ return rec->getValueAsOptionalString("alt_value");
+ }
+
+private:
+ Record *rec;
+ std::string nameWithoutArgs;
+};
+
+class TypedefRec {
+public:
+ explicit TypedefRec(Record *rec) : rec(rec) {}
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+ StringRef getValue() const { return rec->getValueAsString("value"); }
+
+private:
+ Record *rec;
+};
+
+class EnumValueRec {
+public:
+ explicit EnumValueRec(Record *rec) : rec(rec) {}
+ std::string getName() const { return rec->getValueAsString("name").upper(); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+ StringRef getTaggedType() const {
+ return rec->getValueAsString("tagged_type");
+ }
+
+private:
+ Record *rec;
+};
+
+class EnumRec {
+public:
+ explicit EnumRec(Record *rec) : rec(rec) {
+ for (auto *Val : rec->getValueAsListOfDefs("etors")) {
+ vals.emplace_back(EnumValueRec{Val});
+ }
+ }
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+ const std::vector<EnumValueRec> &getValues() const { return vals; }
+
+ std::string getEnumValNamePrefix() const {
+ return StringRef(getName().str().substr(0, getName().str().length() - 2))
+ .upper();
+ }
+
+ bool isTyped() const { return rec->getValueAsBit("is_typed"); }
+
+private:
+ Record *rec;
+ std::vector<EnumValueRec> vals;
+};
+
+class StructMemberRec {
+public:
+ explicit StructMemberRec(Record *rec) : rec(rec) {}
+ StringRef getType() const { return rec->getValueAsString("type"); }
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+
+private:
+ Record *rec;
+};
+
+class StructRec {
+public:
+ explicit StructRec(Record *rec) : rec(rec) {
+ for (auto *Member : rec->getValueAsListOfDefs("all_members")) {
+ members.emplace_back(StructMemberRec(Member));
+ }
+ }
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+ std::optional<StringRef> getBaseClass() const {
+ return rec->getValueAsOptionalString("base_class");
+ }
+ const std::vector<StructMemberRec> &getMembers() const { return members; }
+
+private:
+ Record *rec;
+ std::vector<StructMemberRec> members;
+};
+
+class ParamRec {
+public:
+ explicit ParamRec(Record *rec) : rec(rec) {
+ flags = rec->getValueAsBitsInit("flags");
+ auto *Range = rec->getValueAsDef("range");
+ auto RangeBegin = Range->getValueAsString("begin");
+ auto RangeEnd = Range->getValueAsString("end");
+ if (RangeBegin != "" && RangeEnd != "") {
+ range = {RangeBegin, RangeEnd};
+ } else {
+ range = std::nullopt;
+ }
+
+ auto *TypeInfo = rec->getValueAsDef("type_info");
+ auto TypeInfoEnum = TypeInfo->getValueAsString("enum");
+ auto TypeInfoSize = TypeInfo->getValueAsString("size");
+ if (TypeInfoEnum != "" && TypeInfoSize != "") {
+ typeinfo = {TypeInfoEnum, TypeInfoSize};
+ } else {
+ typeinfo = std::nullopt;
+ }
+ }
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getType() const { return rec->getValueAsString("type"); }
+ bool isPointerType() const { return getType().ends_with('*'); }
+ bool isHandleType() const { return getType().ends_with("_handle_t"); }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+ bool isIn() const { return dyn_cast<BitInit>(flags->getBit(0))->getValue(); }
+ bool isOut() const { return dyn_cast<BitInit>(flags->getBit(1))->getValue(); }
+ bool isOpt() const { return dyn_cast<BitInit>(flags->getBit(2))->getValue(); }
+
+ Record *getRec() const { return rec; }
+ std::optional<std::pair<StringRef, StringRef>> getRange() const {
+ return range;
+ }
+
+ std::optional<std::pair<StringRef, StringRef>> getTypeInfo() const {
+ return typeinfo;
+ }
+
+ // Needed to check whether we're at the back of a vector of params
+ bool operator!=(const ParamRec &p) const { return rec != p.getRec(); }
+
+private:
+ Record *rec;
+ BitsInit *flags;
+ std::optional<std::pair<StringRef, StringRef>> range;
+ std::optional<std::pair<StringRef, StringRef>> typeinfo;
+};
+
+class ReturnRec {
+public:
+ ReturnRec(Record *rec) : rec(rec) {}
+ StringRef getValue() const { return rec->getValueAsString("value"); }
+ std::vector<StringRef> getConditions() const {
+ return rec->getValueAsListOfStrings("conditions");
+ }
+
+private:
+ Record *rec;
+};
+
+class FunctionRec {
+public:
+ FunctionRec(Record *rec) : rec(rec) {
+ for (auto &Ret : rec->getValueAsListOfDefs("all_returns"))
+ rets.emplace_back(Ret);
+ for (auto &Param : rec->getValueAsListOfDefs("params"))
+ params.emplace_back(Param);
+ }
+
+ std::string getParamStructName() const {
+ return llvm::formatv("{0}_params_t",
+ llvm::convertToSnakeFromCamelCase(getName()));
+ }
+
+ StringRef getName() const { return rec->getValueAsString("name"); }
+ StringRef getClass() const { return rec->getValueAsString("api_class"); }
+ const std::vector<ReturnRec> &getReturns() const { return rets; }
+ const std::vector<ParamRec> &getParams() const { return params; }
+ StringRef getDesc() const { return rec->getValueAsString("desc"); }
+ std::vector<StringRef> getDetails() const {
+ return rec->getValueAsListOfStrings("details");
+ }
+ std::vector<StringRef> getAnalogues() const {
+ return rec->getValueAsListOfStrings("analogues");
+ }
+
+private:
+ std::vector<ReturnRec> rets;
+ std::vector<ParamRec> params;
+
+ Record *rec;
+};
+
+} // namespace tblgen
+} // namespace offload
+} // namespace llvm
diff --git a/offload/tools/offload-tblgen/offload-tblgen.cpp b/offload/tools/offload-tblgen/offload-tblgen.cpp
new file mode 100644
index 00000000000000..2e11ec08d0e93a
--- /dev/null
+++ b/offload/tools/offload-tblgen/offload-tblgen.cpp
@@ -0,0 +1,95 @@
+//===- offload-tblgen/offload-tblgen.cpp ----------------------------------===//
+//
+// 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 is a Tablegen tool that produces source files for the Offload project.
+// See offload/API/README.md for more information.
+//
+//===----------------------------------------------------------------------===//
+
+#include "llvm/Support/CommandLine.h"
+#include "llvm/Support/InitLLVM.h"
+#include "llvm/TableGen/Main.h"
+#include "llvm/TableGen/Record.h"
+
+#include "Generators.hpp"
+
+namespace llvm {
+namespace offload {
+namespace tblgen {
+
+enum ActionType {
+ PrintRecords,
+ DumpJSON,
+ GenAPI,
+ GenFuncNames,
+ GenImplFuncDecls,
+ GenEntryPoints,
+ GenPrintHeader
+};
+
+namespace {
+cl::opt<ActionType> Action(
+ cl::desc("Action to perform:"),
+ cl::values(
+ clEnumValN(PrintRecords, "print-records",
+ "Print all records to stdout (default)"),
+ clEnumValN(DumpJSON, "dump-json",
+ "Dump all records as machine-readable JSON"),
+ clEnumValN(GenAPI, "gen-api", "Generate Offload API header contents"),
+ clEnumValN(GenFuncNames, "gen-func-names",
+ "Generate a list of all Offload API function names"),
+ clEnumValN(
+ GenImplFuncDecls, "gen-impl-func-decls",
+ "Generate declarations for Offload API implementation functions"),
+ clEnumValN(GenEntryPoints, "gen-entry-points",
+ "Generate Offload API wrapper function definitions"),
+ clEnumValN(GenPrintHeader, "gen-print-header",
+ "Generate Offload API print header")));
+}
+
+static bool OffloadTableGenMain(raw_ostream &OS, RecordKeeper &Records) {
+ switch (Action) {
+ case PrintRecords:
+ OS << Records;
+ break;
+ case DumpJSON:
+ EmitJSON(Records, OS);
+ break;
+ case GenAPI:
+ EmitOffloadAPI(Records, OS);
+ break;
+ case GenFuncNames:
+ EmitOffloadFuncNames(Records, OS);
+ break;
+ case GenImplFuncDecls:
+ EmitOffloadImplFuncDecls(Records, OS);
+ break;
+ case GenEntryPoints:
+ EmitOffloadEntryPoints(Records, OS);
+ break;
+ case GenPrintHeader:
+ EmitOffloadPrintHeader(Records, OS);
+ break;
+ }
+
+ return false;
+}
+
+int OffloadTblgenMain(int argc, char **argv) {
+ InitLLVM y(argc, argv);
+ cl::ParseCommandLineOptions(argc, argv);
+ return TableGenMain(argv[0], &OffloadTableGenMain);
+}
+} // namespace tblgen
+} // namespace offload
+} // namespace llvm
+
+using namespace llvm;
+using namespace offload::tblgen;
+
+int main(int argc, char **argv) { return OffloadTblgenMain(argc, argv); }
>From 2b142440f0c4bdbf78a46394c51b18241f2046dc Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Fri, 13 Sep 2024 16:03:15 +0100
Subject: [PATCH 02/12] Fix liboffload_new linking and formatting
---
offload/new-api/CMakeLists.txt | 19 ++++++++++++-------
offload/new-api/src/helpers.hpp | 1 -
offload/new-api/src/offload_impl.cpp | 15 +++++++++------
.../common/include/PluginInterface.h | 5 +----
offload/tools/offload-tblgen/FuncsGen.cpp | 19 ++++++++++++++++---
offload/tools/offload-tblgen/GenCommon.hpp | 1 -
offload/tools/offload-tblgen/Generators.hpp | 1 +
.../tools/offload-tblgen/offload-tblgen.cpp | 10 ++++++++--
8 files changed, 47 insertions(+), 24 deletions(-)
diff --git a/offload/new-api/CMakeLists.txt b/offload/new-api/CMakeLists.txt
index c27591d2af3ab3..ac82e7db647601 100644
--- a/offload/new-api/CMakeLists.txt
+++ b/offload/new-api/CMakeLists.txt
@@ -1,4 +1,3 @@
-
set(LLVM_TARGET_DEFINITIONS ${CMAKE_CURRENT_SOURCE_DIR}/API/OffloadAPI.td)
list(APPEND LLVM_TABLEGEN_FLAGS -I ${CMAKE_CURRENT_SOURCE_DIR}/API)
@@ -7,15 +6,22 @@ tablegen(OFFLOAD offload_funcs.inc -gen-func-names)
tablegen(OFFLOAD offload_impl_func_decls.inc -gen-impl-func-decls)
tablegen(OFFLOAD offload_entry_points.inc -gen-entry-points)
tablegen(OFFLOAD offload_print.hpp -gen-print-header)
-
+tablegen(OFFLOAD offload_exports -gen-exports)
add_public_tablegen_target(OffloadHeaderGen)
add_llvm_library(offload_new SHARED
src/offload_lib.cpp
src/offload_impl.cpp
- DEPENDS OffloadHeaderGen
- LINK_LIBS omptarget omptarget.rtl.cuda omptarget.rtl.amdgpu)
+ DEPENDS OffloadHeaderGen)
+
+foreach(plugin IN LISTS LIBOMPTARGET_PLUGINS_TO_BUILD)
+ target_link_libraries(offload_new PRIVATE omptarget.rtl.${plugin})
+endforeach()
+
+if(LIBOMP_HAVE_VERSION_SCRIPT_FLAG)
+ target_link_libraries(offload_new PRIVATE "-Wl,--version-script=${CMAKE_CURRENT_BINARY_DIR}/offload_exports")
+endif()
target_include_directories(offload_new PUBLIC
${CMAKE_CURRENT_BINARY_DIR}
@@ -23,9 +29,8 @@ target_include_directories(offload_new PUBLIC
${CMAKE_CURRENT_SOURCE_DIR}/../include
${CMAKE_CURRENT_SOURCE_DIR}/../plugins-nextgen/common/include)
-# foreach(plugin IN LISTS LIBOMPTARGET_PLUGINS_TO_BUILD)
-# target_link_libraries(offload_new PRIVATE omptarget.rtl.${plugin})
-# endforeach()
+target_compile_options(offload_new PRIVATE ${offload_compile_flags})
+target_link_options(offload_new PRIVATE ${offload_link_flags})
set_target_properties(offload_new PROPERTIES
POSITION_INDEPENDENT_CODE ON
diff --git a/offload/new-api/src/helpers.hpp b/offload/new-api/src/helpers.hpp
index 66148d9f210083..246de2b678015d 100644
--- a/offload/new-api/src/helpers.hpp
+++ b/offload/new-api/src/helpers.hpp
@@ -16,7 +16,6 @@
#include <cstring>
-
template <typename T, typename Assign>
offload_result_t getInfoImpl(size_t ParamValueSize, void *ParamValue,
size_t *ParamValueSizeRet, T Value,
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
index 165f2264ef7daa..b3bf602607ac47 100644
--- a/offload/new-api/src/offload_impl.cpp
+++ b/offload/new-api/src/offload_impl.cpp
@@ -32,7 +32,10 @@ struct offload_platform_handle_t_ {
std::vector<offload_device_handle_t_> Devices;
};
-static std::vector<offload_platform_handle_t_> Platforms;
+std::vector<offload_platform_handle_t_> &Platforms() {
+ static std::vector<offload_platform_handle_t_> Platforms;
+ return Platforms;
+}
// Every plugin exports this method to create an instance of the plugin type.
#define PLUGIN_TARGET(Name) extern "C" GenericPluginTy *createPlugin_##Name();
@@ -42,14 +45,14 @@ void initPlugins() {
// Attempt to create an instance of each supported plugin.
#define PLUGIN_TARGET(Name) \
do { \
- Platforms.emplace_back(offload_platform_handle_t_{ \
+ Platforms().emplace_back(offload_platform_handle_t_{ \
std::unique_ptr<GenericPluginTy>(createPlugin_##Name()), {}}); \
} while (false);
#include "Shared/Targets.def"
// Preemptively initialize all devices in the plugin so we can just return
// them from deviceGet
- for (auto &Platform : Platforms) {
+ for (auto &Platform : Platforms()) {
auto Err = Platform.Plugin->init();
[[maybe_unused]] std::string InfoMsg = toString(std::move(Err));
for (auto DevNum = 0; DevNum < Platform.Plugin->number_of_devices();
@@ -71,19 +74,19 @@ offload_result_t offloadPlatformGet_impl(uint32_t NumEntries,
static std::once_flag InitFlag;
std::call_once(InitFlag, initPlugins);
- if (NumEntries > Platforms.size()) {
+ if (NumEntries > Platforms().size()) {
return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
}
if (phPlatforms) {
for (uint32_t PlatformIndex = 0; PlatformIndex < NumEntries;
PlatformIndex++) {
- phPlatforms[PlatformIndex] = &Platforms[PlatformIndex];
+ phPlatforms[PlatformIndex] = &(Platforms())[PlatformIndex];
}
}
if (pNumPlatforms) {
- *pNumPlatforms = Platforms.size();
+ *pNumPlatforms = Platforms().size();
}
return OFFLOAD_RESULT_SUCCESS;
diff --git a/offload/plugins-nextgen/common/include/PluginInterface.h b/offload/plugins-nextgen/common/include/PluginInterface.h
index 4cfe6cbf9ffe7f..82efcdaf5ad3dc 100644
--- a/offload/plugins-nextgen/common/include/PluginInterface.h
+++ b/offload/plugins-nextgen/common/include/PluginInterface.h
@@ -132,7 +132,6 @@ class InfoQueueTy {
uint64_t Level;
};
-
private:
std::deque<InfoQueueEntryTy> Queue;
@@ -156,9 +155,7 @@ class InfoQueueTy {
Queue.push_back({Key, Value, Units, L});
}
- const std::deque<InfoQueueEntryTy> &getQueue() const {
- return Queue;
- }
+ const std::deque<InfoQueueEntryTy> &getQueue() const { return Queue; }
/// Print all info entries added to the queue.
void print() const {
diff --git a/offload/tools/offload-tblgen/FuncsGen.cpp b/offload/tools/offload-tblgen/FuncsGen.cpp
index 4dc74047ac49d8..567d784f320010 100644
--- a/offload/tools/offload-tblgen/FuncsGen.cpp
+++ b/offload/tools/offload-tblgen/FuncsGen.cpp
@@ -1,4 +1,4 @@
-//===- offload-tblgen/APIGen.cpp - Tablegen backend for Offload validation ===//
+//===- offload-tblgen/APIGen.cpp - Tablegen backend for Offload functions -===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
@@ -6,8 +6,8 @@
//
//===----------------------------------------------------------------------===//
//
-// This is a Tablegen backend that produces validation functions for the Offload
-// API entry point functions.
+// This is a Tablegen backend that handles generation of various small files
+// pertaining to the API functions.
//
//===----------------------------------------------------------------------===//
@@ -36,6 +36,19 @@ void EmitOffloadFuncNames(RecordKeeper &Records, raw_ostream &OS) {
OS << "\n#undef OFFLOAD_FUNC\n";
}
+void EmitOffloadExports(RecordKeeper &Records, raw_ostream &OS) {
+ OS << "VERS1.0 {\n";
+ OS << TAB_1 "global:\n";
+
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ OS << formatv(TAB_2 "{0};\n", FunctionRec(R).getName());
+ }
+ OS << TAB_1 "local:\n";
+ OS << TAB_2 "*;\n";
+ OS << "};\n";
+}
+
+// Emit declarations for every implementation function
void EmitOffloadImplFuncDecls(RecordKeeper &Records, raw_ostream &OS) {
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
FunctionRec F{R};
diff --git a/offload/tools/offload-tblgen/GenCommon.hpp b/offload/tools/offload-tblgen/GenCommon.hpp
index fb24ef06f9ec8f..16ffb3a4d667c8 100644
--- a/offload/tools/offload-tblgen/GenCommon.hpp
+++ b/offload/tools/offload-tblgen/GenCommon.hpp
@@ -55,4 +55,3 @@ MakeParamComment(const llvm::offload::tblgen::ParamRec &Param) {
(Param.isOut() ? "[out]" : ""),
(Param.isOpt() ? "[optional]" : ""), Param.getDesc());
}
-
diff --git a/offload/tools/offload-tblgen/Generators.hpp b/offload/tools/offload-tblgen/Generators.hpp
index 4b300f52ddf84d..4e347ba2fb2946 100644
--- a/offload/tools/offload-tblgen/Generators.hpp
+++ b/offload/tools/offload-tblgen/Generators.hpp
@@ -16,3 +16,4 @@ void EmitOffloadImplFuncDecls(llvm::RecordKeeper &Records,
llvm::raw_ostream &OS);
void EmitOffloadEntryPoints(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
void EmitOffloadPrintHeader(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
+void EmitOffloadExports(llvm::RecordKeeper &Records, llvm::raw_ostream &OS);
diff --git a/offload/tools/offload-tblgen/offload-tblgen.cpp b/offload/tools/offload-tblgen/offload-tblgen.cpp
index 2e11ec08d0e93a..856ebebfc93249 100644
--- a/offload/tools/offload-tblgen/offload-tblgen.cpp
+++ b/offload/tools/offload-tblgen/offload-tblgen.cpp
@@ -29,7 +29,8 @@ enum ActionType {
GenFuncNames,
GenImplFuncDecls,
GenEntryPoints,
- GenPrintHeader
+ GenPrintHeader,
+ GenExports
};
namespace {
@@ -49,7 +50,9 @@ cl::opt<ActionType> Action(
clEnumValN(GenEntryPoints, "gen-entry-points",
"Generate Offload API wrapper function definitions"),
clEnumValN(GenPrintHeader, "gen-print-header",
- "Generate Offload API print header")));
+ "Generate Offload API print header"),
+ clEnumValN(GenExports, "gen-exports",
+ "Generate export file for the Offload library")));
}
static bool OffloadTableGenMain(raw_ostream &OS, RecordKeeper &Records) {
@@ -75,6 +78,9 @@ static bool OffloadTableGenMain(raw_ostream &OS, RecordKeeper &Records) {
case GenPrintHeader:
EmitOffloadPrintHeader(Records, OS);
break;
+ case GenExports:
+ EmitOffloadExports(Records, OS);
+ break;
}
return false;
>From bb17c4fae86b5f90dde584a9e5d8d4021a5b9812 Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Wed, 18 Sep 2024 11:15:00 +0100
Subject: [PATCH 03/12] Refactor entry point validation so failures are visible
in tracing output
---
.../tools/offload-tblgen/EntryPointGen.cpp | 34 +++++++++++++++----
1 file changed, 27 insertions(+), 7 deletions(-)
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
index b89592187639e3..27f8a306e40ddf 100644
--- a/offload/tools/offload-tblgen/EntryPointGen.cpp
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -20,11 +20,10 @@
using namespace llvm;
using namespace offload::tblgen;
-static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
+static void EmitValidationFunc(const FunctionRec &F, raw_ostream &OS) {
OS << CommentsHeader;
// Emit preamble
- OS << formatv("{1}_APIEXPORT {0}_result_t {1}_APICALL {2}(\n ", PrefixLower,
- PrefixUpper, F.getName());
+ OS << formatv("{0}_result_t {1}_val(\n ", PrefixLower, F.getName());
// Emit arguments
std::string ParamNameList = "";
for (auto &Param : F.getParams()) {
@@ -37,7 +36,6 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
OS << ") {\n";
OS << TAB_1 "if (true /*enableParameterValidation*/) {\n";
-
// Emit validation checks
for (const auto &Return : F.getReturns()) {
for (auto &Condition : Return.getConditions()) {
@@ -49,16 +47,37 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
}
}
}
- OS << " }\n\n";
+ OS << TAB_1 "}\n\n";
+
+ // Perform actual function call to the implementation
+ ParamNameList = ParamNameList.substr(0, ParamNameList.size() - 2);
+ OS << formatv(TAB_1 "return {0}_impl({1});\n\n", F.getName(), ParamNameList);
+ OS << "}\n";
+}
+
+static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
+ // Emit preamble
+ OS << formatv("{1}_APIEXPORT {0}_result_t {1}_APICALL {2}(\n ", PrefixLower,
+ PrefixUpper, F.getName());
+ // Emit arguments
+ std::string ParamNameList = "";
+ for (auto &Param : F.getParams()) {
+ OS << Param.getType() << " " << Param.getName();
+ if (Param != F.getParams().back()) {
+ OS << ", ";
+ }
+ ParamNameList += Param.getName().str() + ", ";
+ }
+ OS << ") {\n";
// Emit pre-call prints
OS << TAB_1 "if (std::getenv(\"OFFLOAD_TRACE\")) {\n";
OS << formatv(TAB_2 "std::cout << \"---> {0}\";\n", F.getName());
OS << TAB_1 "}\n\n";
- // Perform actual function call
+ // Perform actual function call to the validation wrapper
ParamNameList = ParamNameList.substr(0, ParamNameList.size() - 2);
- OS << formatv(TAB_1 "{0}_result_t result = {1}_impl({2});\n\n", PrefixLower,
+ OS << formatv(TAB_1 "{0}_result_t result = {1}_val({2});\n\n", PrefixLower,
F.getName(), ParamNameList);
// Emit post-call prints
@@ -81,6 +100,7 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
void EmitOffloadEntryPoints(RecordKeeper &Records, raw_ostream &OS) {
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ EmitValidationFunc(FunctionRec{R}, OS);
EmitEntryPointFunc(FunctionRec{R}, OS);
}
}
>From 2c98c64bd34bd1903de7d071cf902dbb9d1b1dee Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Thu, 26 Sep 2024 16:58:33 +0100
Subject: [PATCH 04/12] Add Offload API unittests
---
offload/new-api/src/offload_impl.cpp | 7 +-
offload/unittests/CMakeLists.txt | 3 +-
offload/unittests/OffloadAPI/CMakeLists.txt | 12 +++
.../OffloadAPI/common/environment.cpp | 92 ++++++++++++++++
.../OffloadAPI/common/environment.hpp | 17 +++
.../unittests/OffloadAPI/common/fixtures.hpp | 54 ++++++++++
.../OffloadAPI/device/offloadDeviceGet.cpp | 41 +++++++
.../device/offloadDeviceGetInfo.cpp | 102 ++++++++++++++++++
.../platform/offloadPlatformGet.cpp | 28 +++++
.../platform/offloadPlatformGetInfo.cpp | 97 +++++++++++++++++
10 files changed, 450 insertions(+), 3 deletions(-)
create mode 100644 offload/unittests/OffloadAPI/CMakeLists.txt
create mode 100644 offload/unittests/OffloadAPI/common/environment.cpp
create mode 100644 offload/unittests/OffloadAPI/common/environment.hpp
create mode 100644 offload/unittests/OffloadAPI/common/fixtures.hpp
create mode 100644 offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp
create mode 100644 offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
create mode 100644 offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
create mode 100644 offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
index b3bf602607ac47..c6c68eeba0bf72 100644
--- a/offload/new-api/src/offload_impl.cpp
+++ b/offload/new-api/src/offload_impl.cpp
@@ -25,6 +25,7 @@ using namespace llvm::omp::target::plugin;
struct offload_device_handle_t_ {
int DeviceNum;
GenericDeviceTy &Device;
+ offload_platform_handle_t Platform;
};
struct offload_platform_handle_t_ {
@@ -59,7 +60,7 @@ void initPlugins() {
DevNum++) {
if (Platform.Plugin->init_device(DevNum) == OFFLOAD_SUCCESS) {
Platform.Devices.emplace_back(offload_device_handle_t_{
- DevNum, Platform.Plugin->getDevice(DevNum)});
+ DevNum, Platform.Plugin->getDevice(DevNum), &Platform});
}
}
}
@@ -173,6 +174,8 @@ offload_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
};
switch (propName) {
+ case OFFLOAD_DEVICE_INFO_PLATFORM:
+ return ReturnValue(hDevice->Platform);
case OFFLOAD_DEVICE_INFO_TYPE:
return ReturnValue(OFFLOAD_DEVICE_TYPE_GPU);
case OFFLOAD_DEVICE_INFO_NAME:
@@ -183,7 +186,7 @@ offload_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
return ReturnValue(
GetInfo({"CUDA Driver Version", "HSA Runtime Version"}).c_str());
default:
- return OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION;
+ return OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION;
}
return OFFLOAD_RESULT_SUCCESS;
diff --git a/offload/unittests/CMakeLists.txt b/offload/unittests/CMakeLists.txt
index 73c87b708d25fd..25ac4b2fa36752 100644
--- a/offload/unittests/CMakeLists.txt
+++ b/offload/unittests/CMakeLists.txt
@@ -5,4 +5,5 @@ function(add_libompt_unittest test_dirname)
add_unittest(LibomptUnitTests ${test_dirname} ${ARGN})
endfunction()
-add_subdirectory(Plugins)
+# add_subdirectory(Plugins)
+add_subdirectory(OffloadAPI)
diff --git a/offload/unittests/OffloadAPI/CMakeLists.txt b/offload/unittests/OffloadAPI/CMakeLists.txt
new file mode 100644
index 00000000000000..0960fa5a135111
--- /dev/null
+++ b/offload/unittests/OffloadAPI/CMakeLists.txt
@@ -0,0 +1,12 @@
+set(PLUGINS_TEST_COMMON offload_new)
+set(PLUGINS_TEST_INCLUDE ${LIBOMPTARGET_INCLUDE_DIR} ${CMAKE_CURRENT_SOURCE_DIR}/common)
+
+add_libompt_unittest("offload.unittests"
+ ${CMAKE_CURRENT_SOURCE_DIR}/common/environment.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/platform/offloadPlatformGet.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/platform/offloadPlatformGetInfo.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGet.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGetInfo.cpp)
+add_dependencies("offload.unittests" ${PLUGINS_TEST_COMMON})
+target_link_libraries("offload.unittests" PRIVATE ${PLUGINS_TEST_COMMON})
+target_include_directories("offload.unittests" PRIVATE ${PLUGINS_TEST_INCLUDE})
diff --git a/offload/unittests/OffloadAPI/common/environment.cpp b/offload/unittests/OffloadAPI/common/environment.cpp
new file mode 100644
index 00000000000000..a19e09fc5767c0
--- /dev/null
+++ b/offload/unittests/OffloadAPI/common/environment.cpp
@@ -0,0 +1,92 @@
+//===------- Offload API tests - gtest environment ------------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include "environment.hpp"
+#include "fixtures.hpp"
+#include "llvm/Support/CommandLine.h"
+#include <offload_api.h>
+
+using namespace llvm;
+
+static cl::opt<std::string>
+ SelectedPlatform("platform", cl::desc("Only test the specified platform"),
+ cl::value_desc("platform"));
+
+std::ostream &operator<<(std::ostream &Out,
+ const offload_platform_handle_t &Platform) {
+ size_t Size;
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_NAME, 0, nullptr,
+ &Size);
+ std::vector<char> Name(Size);
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_NAME, Size,
+ Name.data(), nullptr);
+ Out << Name.data();
+ return Out;
+}
+
+std::ostream &
+operator<<(std::ostream &Out,
+ const std::vector<offload_platform_handle_t> &Platforms) {
+ for (auto Platform : Platforms) {
+ Out << "\n * \"" << Platform << "\"";
+ }
+ return Out;
+}
+
+const std::vector<offload_platform_handle_t> &TestEnvironment::getPlatforms() {
+ static std::vector<offload_platform_handle_t> Platforms{};
+
+ if (Platforms.empty()) {
+ uint32_t PlatformCount = 0;
+ offloadPlatformGet(0, nullptr, &PlatformCount);
+ if (PlatformCount > 0) {
+ Platforms.resize(PlatformCount);
+ offloadPlatformGet(PlatformCount, Platforms.data(), nullptr);
+ }
+ }
+
+ return Platforms;
+}
+
+// Get a single platform, which may be selected by the user.
+offload_platform_handle_t TestEnvironment::getPlatform() {
+ static offload_platform_handle_t Platform = nullptr;
+ const auto &Platforms = getPlatforms();
+
+ if (!Platform) {
+ if (SelectedPlatform != "") {
+ for (const auto CandidatePlatform : Platforms) {
+ std::stringstream PlatformName;
+ PlatformName << CandidatePlatform;
+ if (SelectedPlatform == PlatformName.str()) {
+ Platform = CandidatePlatform;
+ return Platform;
+ }
+ }
+ std::cout << "No platform found with the name \"" << SelectedPlatform
+ << "\". Choose from:" << Platforms << "\n";
+ std::exit(1);
+ } else {
+ // Pick a single platform. We prefer one that has available devices, but
+ // just pick the first initially in case none have any devices.
+ Platform = Platforms[0];
+ for (auto CandidatePlatform : Platforms) {
+ uint32_t NumDevices = 0;
+ if (offloadDeviceGet(CandidatePlatform, OFFLOAD_DEVICE_TYPE_ALL, 0,
+ nullptr, &NumDevices) == OFFLOAD_RESULT_SUCCESS) {
+ if (NumDevices > 0) {
+ Platform = CandidatePlatform;
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ return Platform;
+}
diff --git a/offload/unittests/OffloadAPI/common/environment.hpp b/offload/unittests/OffloadAPI/common/environment.hpp
new file mode 100644
index 00000000000000..3e85c54c7413eb
--- /dev/null
+++ b/offload/unittests/OffloadAPI/common/environment.hpp
@@ -0,0 +1,17 @@
+//===------- Offload API tests - gtest environment ------------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#pragma once
+
+#include <gtest/gtest.h>
+#include <offload_api.h>
+
+namespace TestEnvironment {
+const std::vector<offload_platform_handle_t> &getPlatforms();
+offload_platform_handle_t getPlatform();
+} // namespace TestEnvironment
diff --git a/offload/unittests/OffloadAPI/common/fixtures.hpp b/offload/unittests/OffloadAPI/common/fixtures.hpp
new file mode 100644
index 00000000000000..a7f5f84adb2568
--- /dev/null
+++ b/offload/unittests/OffloadAPI/common/fixtures.hpp
@@ -0,0 +1,54 @@
+//===------- Offload API tests - gtest fixtures --==-----------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include <gtest/gtest.h>
+#include <offload_api.h>
+#include <offload_print.hpp>
+
+#include "environment.hpp"
+
+#pragma once
+
+#ifndef ASSERT_SUCCESS
+#define ASSERT_SUCCESS(ACTUAL) ASSERT_EQ(OFFLOAD_RESULT_SUCCESS, ACTUAL)
+#endif
+
+#define RETURN_ON_FATAL_FAILURE(...) \
+ __VA_ARGS__; \
+ if (this->HasFatalFailure() || this->IsSkipped()) { \
+ return; \
+ } \
+ (void)0
+
+struct offloadPlatformTest : ::testing::Test {
+ void SetUp() override {
+ RETURN_ON_FATAL_FAILURE(::testing::Test::SetUp());
+
+ Platform = TestEnvironment::getPlatform();
+ ASSERT_NE(Platform, nullptr);
+ }
+
+ offload_platform_handle_t Platform;
+};
+
+struct offloadDeviceTest : offloadPlatformTest {
+ void SetUp() override {
+ RETURN_ON_FATAL_FAILURE(offloadPlatformTest::SetUp());
+
+ uint32_t NumDevices;
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 0,
+ nullptr, &NumDevices));
+ if (NumDevices == 0) {
+ GTEST_SKIP() << "No available devices on this platform.";
+ }
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 1,
+ &Device, nullptr));
+ }
+
+ offload_device_handle_t Device;
+};
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp
new file mode 100644
index 00000000000000..1442ebc98c7f6e
--- /dev/null
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp
@@ -0,0 +1,41 @@
+//===------- Offload API tests - offloadDeviceGet -------------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include "../common/fixtures.hpp"
+#include <gtest/gtest.h>
+#include <offload_api.h>
+
+using offloadDeviceGetTest = offloadPlatformTest;
+
+TEST_F(offloadDeviceGetTest, Success) {
+ uint32_t Count = 0;
+ ASSERT_SUCCESS(
+ offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 0, nullptr, &Count));
+ ASSERT_NE(Count, 0lu);
+ std::vector<offload_device_handle_t> Devices(Count);
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, Count,
+ Devices.data(), nullptr));
+ for (auto Device : Devices) {
+ ASSERT_NE(nullptr, Device);
+ }
+}
+
+TEST_F(offloadDeviceGetTest, SuccessSubsetOfDevices) {
+ uint32_t Count;
+ ASSERT_SUCCESS(
+ offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 0, nullptr, &Count));
+ if (Count < 2) {
+ GTEST_SKIP() << "Only one device is available on this platform.";
+ }
+ std::vector<offload_device_handle_t> Devices(Count - 1);
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, Count - 1,
+ Devices.data(), nullptr));
+ for (auto Device : Devices) {
+ ASSERT_NE(nullptr, Device);
+ }
+}
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
new file mode 100644
index 00000000000000..df8aecc1f0be2d
--- /dev/null
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
@@ -0,0 +1,102 @@
+//===------- Offload API tests - offloadDeviceGetInfo ---------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include "../common/fixtures.hpp"
+#include <gtest/gtest.h>
+#include <offload_api.h>
+
+struct offloadDeviceGetInfoTest
+ : offloadDeviceTest,
+ ::testing::WithParamInterface<offload_device_info_t> {
+
+ void SetUp() override { RETURN_ON_FATAL_FAILURE(offloadDeviceTest::SetUp()); }
+};
+
+// TODO: We could autogenerate the list of enum values
+INSTANTIATE_TEST_SUITE_P(
+ , offloadDeviceGetInfoTest,
+ ::testing::Values(OFFLOAD_DEVICE_INFO_TYPE, OFFLOAD_DEVICE_INFO_PLATFORM,
+ OFFLOAD_DEVICE_INFO_NAME, OFFLOAD_DEVICE_INFO_VENDOR,
+ OFFLOAD_DEVICE_INFO_DRIVER_VERSION),
+ [](const ::testing::TestParamInfo<offload_device_info_t> &info) {
+ std::stringstream ss;
+ ss << info.param;
+ return ss.str();
+ });
+
+// TODO: We could autogenerate this
+std::unordered_map<offload_device_info_t, size_t> DeviceInfoSizeMap = {
+ {OFFLOAD_DEVICE_INFO_TYPE, sizeof(offload_device_type_t)},
+ {OFFLOAD_DEVICE_INFO_PLATFORM, sizeof(offload_platform_handle_t)},
+};
+
+TEST_P(offloadDeviceGetInfoTest, Success) {
+ offload_device_info_t InfoType = GetParam();
+ size_t Size = 0;
+
+ ASSERT_SUCCESS(offloadDeviceGetInfo(Device, InfoType, 0, nullptr, &Size));
+ auto ExpectedSize = DeviceInfoSizeMap.find(InfoType);
+ if (ExpectedSize != DeviceInfoSizeMap.end()) {
+ ASSERT_EQ(Size, ExpectedSize->second);
+ } else {
+ ASSERT_NE(Size, 0lu);
+ }
+
+ std::vector<char> InfoData(Size);
+ ASSERT_SUCCESS(
+ offloadDeviceGetInfo(Device, InfoType, Size, InfoData.data(), nullptr));
+
+ if (InfoType == OFFLOAD_DEVICE_INFO_PLATFORM) {
+ auto *ReturnedPlatform =
+ reinterpret_cast<offload_platform_handle_t *>(InfoData.data());
+ ASSERT_EQ(Platform, *ReturnedPlatform);
+ }
+}
+
+TEST_F(offloadDeviceGetInfoTest, InvalidNullHandleDevice) {
+ offload_device_type_t DeviceType;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE,
+ offloadDeviceGetInfo(nullptr, OFFLOAD_DEVICE_INFO_TYPE,
+ sizeof(offload_device_type_t), &DeviceType,
+ nullptr));
+}
+
+TEST_F(offloadDeviceGetInfoTest, InvalidEnumerationInfoType) {
+ offload_device_type_t DeviceType;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_FORCE_UINT32,
+ sizeof(offload_device_type_t), &DeviceType,
+ nullptr));
+}
+
+TEST_F(offloadDeviceGetInfoTest, InvalidSizePropSize) {
+ offload_device_type_t DeviceType;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0,
+ &DeviceType, nullptr));
+}
+
+TEST_F(offloadDeviceGetInfoTest, InvalidSizePropSizeSmall) {
+ offload_device_type_t DeviceType;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
+ sizeof(DeviceType) - 1, &DeviceType, nullptr));
+}
+
+TEST_F(offloadDeviceGetInfoTest, InvalidNullPointerPropValue) {
+ offload_device_type_t DeviceType;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
+ sizeof(DeviceType), nullptr, nullptr));
+}
+
+TEST_F(offloadDeviceGetInfoTest, InvalidNullPointerPropSizeRet) {
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0, nullptr,
+ nullptr));
+}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
new file mode 100644
index 00000000000000..6ede71ff1babe6
--- /dev/null
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
@@ -0,0 +1,28 @@
+//===------- Offload API tests - offloadPlatformGet -----------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include <gtest/gtest.h>
+#include <offload_api.h>
+#include "../common/fixtures.hpp"
+
+using offloadPlatformGetTest = ::testing::Test;
+
+TEST_F(offloadPlatformGetTest, Success) {
+ uint32_t PlatformCount;
+ ASSERT_SUCCESS(offloadPlatformGet(0, nullptr, &PlatformCount));
+ std::vector<offload_platform_handle_t> Platforms(PlatformCount);
+ ASSERT_SUCCESS(offloadPlatformGet(PlatformCount, Platforms.data(), nullptr));
+}
+
+TEST_F(offloadPlatformGetTest, InvalidNumEntries) {
+ uint32_t PlatformCount;
+ ASSERT_SUCCESS(offloadPlatformGet(0, nullptr, &PlatformCount));
+ std::vector<offload_platform_handle_t> Platforms(PlatformCount);
+ ASSERT_EQ(offloadPlatformGet(0, Platforms.data(), nullptr),
+ OFFLOAD_RESULT_ERROR_INVALID_SIZE);
+}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
new file mode 100644
index 00000000000000..7a967ee674b00c
--- /dev/null
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
@@ -0,0 +1,97 @@
+//===------- Offload API tests - offloadPlatformGetInfo -------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include <offload_api.h>
+
+#include "../common/fixtures.hpp"
+
+struct offloadPlatformGetInfoTest
+ : offloadPlatformTest,
+ ::testing::WithParamInterface<offload_platform_info_t> {};
+
+// TODO: We could autogenerate the list of enum values
+INSTANTIATE_TEST_SUITE_P(
+ offloadPlatformGetInfo, offloadPlatformGetInfoTest,
+ ::testing::Values(OFFLOAD_PLATFORM_INFO_NAME,
+ OFFLOAD_PLATFORM_INFO_VENDOR_NAME,
+ OFFLOAD_PLATFORM_INFO_VERSION,
+ OFFLOAD_PLATFORM_INFO_BACKEND),
+ [](const ::testing::TestParamInfo<offload_platform_info_t> &info) {
+ std::stringstream ss;
+ ss << info.param;
+ return ss.str();
+ });
+
+// TODO: We could autogenerate this
+std::unordered_map<offload_platform_info_t, size_t> PlatformInfoSizeMap = {
+ {OFFLOAD_PLATFORM_INFO_BACKEND, sizeof(offload_platform_backend_t)},
+};
+
+TEST_P(offloadPlatformGetInfoTest, Success) {
+ size_t Size = 0;
+ offload_platform_info_t InfoType = GetParam();
+
+ ASSERT_SUCCESS(offloadPlatformGetInfo(Platform, InfoType, 0, nullptr, &Size));
+ auto ExpectedSize = PlatformInfoSizeMap.find(InfoType);
+ if (ExpectedSize != PlatformInfoSizeMap.end()) {
+ ASSERT_EQ(Size, ExpectedSize->second);
+ } else {
+ ASSERT_NE(Size, 0lu);
+ }
+
+ std::vector<char> InfoData(Size);
+ ASSERT_SUCCESS(offloadPlatformGetInfo(Platform, InfoType, Size,
+ InfoData.data(), nullptr));
+
+ // Info types with a dynamic size are all char[] so we can verify the returned
+ // string is the expected size.
+ if (ExpectedSize == PlatformInfoSizeMap.end()) {
+ ASSERT_EQ(Size, strlen(InfoData.data()) + 1);
+ }
+}
+
+TEST_F(offloadPlatformGetInfoTest, InvalidNullHandle) {
+ size_t Size = 0;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE,
+ offloadPlatformGetInfo(nullptr, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
+ nullptr, &Size));
+}
+
+TEST_F(offloadPlatformGetInfoTest, InvalidPlatformInfoEnumeration) {
+ size_t Size = 0;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_FORCE_UINT32,
+ 0, nullptr, &Size));
+}
+
+TEST_F(offloadPlatformGetInfoTest, InvalidSizeZero) {
+ offload_platform_backend_t Backend;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
+ &Backend, nullptr));
+}
+
+TEST_F(offloadPlatformGetInfoTest, InvalidSizeSmall) {
+ offload_platform_backend_t Backend;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
+ sizeof(Backend) - 1, &Backend, nullptr));
+}
+
+TEST_F(offloadPlatformGetInfoTest, InvalidNullPointerPropValue) {
+ offload_platform_backend_t Backend;
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
+ sizeof(Backend), nullptr, nullptr));
+}
+
+TEST_F(offloadPlatformGetInfoTest, InvalidNullPointerPropSizeRet) {
+ ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
+ nullptr, nullptr));
+}
>From 5d926712b062c3e1827d58602daaeaf1835e9915 Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Tue, 1 Oct 2024 09:46:03 +0100
Subject: [PATCH 05/12] Implement optional error details
---
offload/new-api/API/Common.td | 18 ++++++-
offload/new-api/CMakeLists.txt | 1 +
offload/new-api/include/offload_impl.hpp | 33 ++++++++++++
offload/new-api/src/offload_impl.cpp | 50 +++++++++++++------
offload/new-api/src/offload_lib.cpp | 9 ++++
.../tools/offload-tblgen/EntryPointGen.cpp | 4 ++
offload/tools/offload-tblgen/FuncsGen.cpp | 8 ++-
7 files changed, 107 insertions(+), 16 deletions(-)
create mode 100644 offload/new-api/include/offload_impl.hpp
diff --git a/offload/new-api/API/Common.td b/offload/new-api/API/Common.td
index d293e4addfef8b..55d03d4ff1e783 100644
--- a/offload/new-api/API/Common.td
+++ b/offload/new-api/API/Common.td
@@ -72,4 +72,20 @@ def : Enum {
Etor<"ERROR_UNSUPPORTED_ENUMERATION", "[Validation] enumerator argument is not supported by the device">,
Etor<"ERROR_UNKNOWN", "Unknown or internal error">
];
-}
\ No newline at end of file
+}
+
+def : Function {
+ let name = "offloadGetErrorDetails";
+ let desc = "Get a detailed error message for the last error that occurred on this thread, if it exists";
+ let details = [
+ "When an Offload API call returns a return value other than OFFLOAD_RESULT_SUCCESS, the implementation *may* set an additional error message.",
+ "If a further Offload call (excluding this function) is made on the same thread without checking "
+ "its detailed error message with this function, that message should be considered lost.",
+ "The returned char* is only valid until the next Offload function call on the same thread (excluding further calls to this function.)"
+ ];
+ let params = [
+ Param<"size_t*", "SizeRet", "Pointer to return the size of the available error message. A size of 0 indicates no message.", PARAM_OUT_OPTIONAL>,
+ Param<"const char**", "DetailStringRet", "Pointer to return the error message string.", PARAM_OUT_OPTIONAL>
+ ];
+ let returns = []; // Only SUCCESS is expected
+}
diff --git a/offload/new-api/CMakeLists.txt b/offload/new-api/CMakeLists.txt
index ac82e7db647601..be1e711564c244 100644
--- a/offload/new-api/CMakeLists.txt
+++ b/offload/new-api/CMakeLists.txt
@@ -26,6 +26,7 @@ endif()
target_include_directories(offload_new PUBLIC
${CMAKE_CURRENT_BINARY_DIR}
${CMAKE_CURRENT_BINARY_DIR}/../include
+ ${CMAKE_CURRENT_SOURCE_DIR}/include
${CMAKE_CURRENT_SOURCE_DIR}/../include
${CMAKE_CURRENT_SOURCE_DIR}/../plugins-nextgen/common/include)
diff --git a/offload/new-api/include/offload_impl.hpp b/offload/new-api/include/offload_impl.hpp
new file mode 100644
index 00000000000000..1e96fdebfc765e
--- /dev/null
+++ b/offload/new-api/include/offload_impl.hpp
@@ -0,0 +1,33 @@
+//===- offload_impl.hpp- Implementation helpers for the Offload library ---===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include <offload_api.h>
+#include <optional>
+#include <string>
+
+#include "llvm/ADT/StringRef.h"
+
+std::optional<std::string> &LastErrorDetails();
+
+struct offload_impl_result_t {
+ offload_impl_result_t() = delete;
+ offload_impl_result_t(offload_result_t Result) : Result(Result) {
+ LastErrorDetails() = std::nullopt;
+ }
+
+ offload_impl_result_t(offload_result_t Result, std::string Details)
+ : Result(Result) {
+ assert(Result != OFFLOAD_RESULT_SUCCESS);
+ LastErrorDetails() = Details;
+ }
+
+ operator offload_result_t() { return Result; }
+
+private:
+ offload_result_t Result;
+};
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
index c6c68eeba0bf72..b544a847a8dbd2 100644
--- a/offload/new-api/src/offload_impl.cpp
+++ b/offload/new-api/src/offload_impl.cpp
@@ -13,6 +13,7 @@
#include "PluginManager.h"
#include "helpers.hpp"
+#include "offload_impl.hpp"
#include "llvm/Support/FormatVariadic.h"
#include <offload_api.h>
@@ -66,9 +67,27 @@ void initPlugins() {
}
}
-offload_result_t offloadPlatformGet_impl(uint32_t NumEntries,
- offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms) {
+offload_result_t offloadGetErrorDetails_impl(size_t *SizeRet,
+ const char **DetailStringRet) {
+ if (auto Details = LastErrorDetails()) {
+ if (SizeRet) {
+ *SizeRet = Details->size();
+ }
+ if (DetailStringRet) {
+ *DetailStringRet = Details->c_str();
+ }
+ } else {
+ if (SizeRet) {
+ *SizeRet = 0;
+ }
+ }
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+offload_impl_result_t
+offloadPlatformGet_impl(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms) {
// It is expected that offloadPlatformGet is the first function to be called.
// In future it may make sense to have a specific entry point for Offload
// initialization, or expose explicit initialization of plugins.
@@ -76,7 +95,9 @@ offload_result_t offloadPlatformGet_impl(uint32_t NumEntries,
std::call_once(InitFlag, initPlugins);
if (NumEntries > Platforms().size()) {
- return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ return {OFFLOAD_RESULT_ERROR_INVALID_SIZE,
+ formatv("{0} platform(s) available but {1} requested.",
+ Platforms().size(), NumEntries)};
}
if (phPlatforms) {
@@ -93,7 +114,7 @@ offload_result_t offloadPlatformGet_impl(uint32_t NumEntries,
return OFFLOAD_RESULT_SUCCESS;
}
-offload_result_t
+offload_impl_result_t
offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
offload_platform_info_t propName, size_t propSize,
void *pPropValue, size_t *pPropSizeRet) {
@@ -126,11 +147,11 @@ offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
return OFFLOAD_RESULT_SUCCESS;
}
-offload_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
- offload_device_type_t,
- uint32_t NumEntries,
- offload_device_handle_t *phDevices,
- uint32_t *pNumDevices) {
+offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
+ offload_device_type_t,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices) {
if (phDevices) {
for (uint32_t DeviceIndex = 0; DeviceIndex < NumEntries; DeviceIndex++) {
@@ -145,10 +166,11 @@ offload_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
return OFFLOAD_RESULT_SUCCESS;
}
-offload_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
- offload_device_info_t propName,
- size_t propSize, void *pPropValue,
- size_t *pPropSizeRet) {
+offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t propSize,
+ void *pPropValue,
+ size_t *pPropSizeRet) {
ReturnHelper ReturnValue(propSize, pPropValue, pPropSizeRet);
diff --git a/offload/new-api/src/offload_lib.cpp b/offload/new-api/src/offload_lib.cpp
index b2452b2cdab950..a8a23e94a2fd6c 100644
--- a/offload/new-api/src/offload_lib.cpp
+++ b/offload/new-api/src/offload_lib.cpp
@@ -10,11 +10,20 @@
//
//===----------------------------------------------------------------------===//
+#include "offload_impl.hpp"
#include <offload_api.h>
#include <offload_print.hpp>
#include <iostream>
+// Store details for the last error that occurred on this thread. It MAY be set
+// when an implementation function returns a result other than
+// OFFLOAD_RESULT_SUCCESS.
+std::optional<std::string> &LastErrorDetails() {
+ thread_local std::optional<std::string> Details;
+ return Details;
+}
+
// Pull in the declarations for the implementation funtions. The actual entry
// points in this file wrap these.
#include "offload_impl_func_decls.inc"
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
index 27f8a306e40ddf..5dbb0e8c22ae04 100644
--- a/offload/tools/offload-tblgen/EntryPointGen.cpp
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -92,6 +92,10 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
OS << formatv("};\n");
OS << TAB_2 "std::cout << \"(\" << &Params << \")\";\n";
OS << TAB_2 "std::cout << \"-> \" << result << \"\\n\";\n";
+ OS << TAB_2 "if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {\n";
+ OS << TAB_3 "std::cout << \" *Error Details* \" << *LastErrorDetails() "
+ "<< \" \\n\";\n";
+ OS << TAB_2 "}\n";
OS << TAB_1 "}\n";
OS << TAB_1 "return result;\n";
diff --git a/offload/tools/offload-tblgen/FuncsGen.cpp b/offload/tools/offload-tblgen/FuncsGen.cpp
index 567d784f320010..ee651957abc703 100644
--- a/offload/tools/offload-tblgen/FuncsGen.cpp
+++ b/offload/tools/offload-tblgen/FuncsGen.cpp
@@ -52,7 +52,13 @@ void EmitOffloadExports(RecordKeeper &Records, raw_ostream &OS) {
void EmitOffloadImplFuncDecls(RecordKeeper &Records, raw_ostream &OS) {
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
FunctionRec F{R};
- OS << formatv("{0}_result_t {1}_impl(", PrefixLower, F.getName());
+ // The error details function does not set error details itself, so don't
+ // use the impl result type
+ if (F.getName() == "offloadGetErrorDetails") {
+ OS << formatv("{0}_result_t {1}_impl(", PrefixLower, F.getName());
+ } else {
+ OS << formatv("{0}_impl_result_t {1}_impl(", PrefixLower, F.getName());
+ }
auto Params = F.getParams();
for (auto &Param : Params) {
OS << Param.getType() << " " << Param.getName();
>From 35e70fbece205cef8e8754a5b6acf141a41dd8aa Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Tue, 1 Oct 2024 17:31:28 +0100
Subject: [PATCH 06/12] Check in auto-generated Offload files
---
offload/new-api/CMakeLists.txt | 17 +-
offload/new-api/include/offload_api.h | 342 ++++++++++++++++
.../new-api/include/offload_entry_points.inc | 195 +++++++++
offload/new-api/include/offload_exports | 10 +
offload/new-api/include/offload_funcs.inc | 19 +
.../include/offload_impl_func_decls.inc | 12 +
offload/new-api/include/offload_print.hpp | 380 ++++++++++++++++++
offload/tools/offload-tblgen/APIGen.cpp | 1 +
.../tools/offload-tblgen/EntryPointGen.cpp | 1 +
offload/tools/offload-tblgen/FuncsGen.cpp | 2 +
offload/tools/offload-tblgen/GenCommon.hpp | 10 +
offload/tools/offload-tblgen/PrintGen.cpp | 1 +
12 files changed, 985 insertions(+), 5 deletions(-)
create mode 100644 offload/new-api/include/offload_api.h
create mode 100644 offload/new-api/include/offload_entry_points.inc
create mode 100644 offload/new-api/include/offload_exports
create mode 100644 offload/new-api/include/offload_funcs.inc
create mode 100644 offload/new-api/include/offload_impl_func_decls.inc
create mode 100644 offload/new-api/include/offload_print.hpp
diff --git a/offload/new-api/CMakeLists.txt b/offload/new-api/CMakeLists.txt
index be1e711564c244..9c82eb841bb8b8 100644
--- a/offload/new-api/CMakeLists.txt
+++ b/offload/new-api/CMakeLists.txt
@@ -8,23 +8,30 @@ tablegen(OFFLOAD offload_entry_points.inc -gen-entry-points)
tablegen(OFFLOAD offload_print.hpp -gen-print-header)
tablegen(OFFLOAD offload_exports -gen-exports)
-add_public_tablegen_target(OffloadHeaderGen)
+foreach(itm IN LISTS TABLEGEN_OUTPUT)
+ message(${itm})
+endforeach()
+
+set(OFFLOAD_GENERATED_FILES ${TABLEGEN_OUTPUT})
+add_public_tablegen_target(OffloadGenerate)
+
+add_custom_command(TARGET OffloadGenerate POST_BUILD COMMAND ${CMAKE_COMMAND}
+ -E copy_if_different ${OFFLOAD_GENERATED_FILES} "${CMAKE_CURRENT_SOURCE_DIR}/include")
add_llvm_library(offload_new SHARED
src/offload_lib.cpp
src/offload_impl.cpp
- DEPENDS OffloadHeaderGen)
+ DEPENDS OffloadGenerate)
foreach(plugin IN LISTS LIBOMPTARGET_PLUGINS_TO_BUILD)
target_link_libraries(offload_new PRIVATE omptarget.rtl.${plugin})
endforeach()
if(LIBOMP_HAVE_VERSION_SCRIPT_FLAG)
- target_link_libraries(offload_new PRIVATE "-Wl,--version-script=${CMAKE_CURRENT_BINARY_DIR}/offload_exports")
+ target_link_libraries(offload_new PRIVATE "-Wl,--version-script=${CMAKE_CURRENT_SOURCE_DIR}/include/offload_exports")
endif()
target_include_directories(offload_new PUBLIC
- ${CMAKE_CURRENT_BINARY_DIR}
${CMAKE_CURRENT_BINARY_DIR}/../include
${CMAKE_CURRENT_SOURCE_DIR}/include
${CMAKE_CURRENT_SOURCE_DIR}/../include
@@ -39,4 +46,4 @@ set_target_properties(offload_new PROPERTIES
BUILD_RPATH "$ORIGIN:${CMAKE_CURRENT_BINARY_DIR}/..")
install(TARGETS offload_new LIBRARY COMPONENT offload_new DESTINATION "${OFFLOAD_INSTALL_LIBDIR}")
-install(FILES ${CMAKE_CURRENT_BINARY_DIR}/offload_api.h DESTINATION ${CMAKE_INSTALL_PREFIX}/include/offload)
+install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/offload_api.h DESTINATION ${CMAKE_INSTALL_PREFIX}/include/offload)
diff --git a/offload/new-api/include/offload_api.h b/offload/new-api/include/offload_api.h
new file mode 100644
index 00000000000000..c046496a2e0f9f
--- /dev/null
+++ b/offload/new-api/include/offload_api.h
@@ -0,0 +1,342 @@
+//===- Auto-generated file, part of the LLVM/Offload project --------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+// Auto-generated file, do not manually edit.
+
+#pragma once
+
+#include <stddef.h>
+#include <stdint.h>
+
+#if defined(__cplusplus)
+extern "C" {
+#endif
+
+
+///////////////////////////////////////////////////////////////////////////////
+#ifndef OFFLOAD_APICALL
+#if defined(_WIN32)
+/// @brief Calling convention for all API functions
+#define OFFLOAD_APICALL __cdecl
+#else
+#define OFFLOAD_APICALL
+#endif // defined(_WIN32)
+#endif // OFFLOAD_APICALL
+
+///////////////////////////////////////////////////////////////////////////////
+#ifndef OFFLOAD_APIEXPORT
+#if defined(_WIN32)
+/// @brief Microsoft-specific dllexport storage-class attribute
+#define OFFLOAD_APIEXPORT __declspec(dllexport)
+#else
+#define OFFLOAD_APIEXPORT
+#endif // defined(_WIN32)
+#endif // OFFLOAD_APIEXPORT
+
+///////////////////////////////////////////////////////////////////////////////
+#ifndef OFFLOAD_DLLEXPORT
+#if defined(_WIN32)
+/// @brief Microsoft-specific dllexport storage-class attribute
+#define OFFLOAD_DLLEXPORT __declspec(dllexport)
+#endif // defined(_WIN32)
+#endif // OFFLOAD_DLLEXPORT
+
+///////////////////////////////////////////////////////////////////////////////
+#ifndef OFFLOAD_DLLEXPORT
+#if __GNUC__ >= 4
+/// @brief GCC-specific dllexport storage-class attribute
+#define OFFLOAD_DLLEXPORT __attribute__ ((visibility ("default")))
+#else
+#define OFFLOAD_DLLEXPORT
+#endif // __GNUC__ >= 4
+#endif // OFFLOAD_DLLEXPORT
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief compiler-independent type
+typedef uint8_t offload_bool_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Handle of a platform instance
+typedef struct offload_platform_handle_t_ *offload_platform_handle_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Handle of platform's device object
+typedef struct offload_device_handle_t_ *offload_device_handle_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Handle of context object
+typedef struct offload_context_handle_t_ *offload_context_handle_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Defines Return/Error codes
+typedef enum offload_result_t {
+ OFFLOAD_RESULT_SUCCESS = 0, ///< Success
+ OFFLOAD_RESULT_ERROR_INVALID_VALUE = 1, ///< Invalid Value
+ OFFLOAD_RESULT_ERROR_INVALID_PLATFORM = 2, ///< Invalid platform
+ OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND = 3, ///< Device not found
+ OFFLOAD_RESULT_ERROR_INVALID_DEVICE = 4, ///< Invalid device
+ OFFLOAD_RESULT_ERROR_DEVICE_LOST = 5, ///< Device hung, reset, was removed, or driver update occurred
+ OFFLOAD_RESULT_ERROR_UNINITIALIZED = 6, ///< plugin is not initialized or specific entry-point is not implemented
+ OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES = 7, ///< Out of resources
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION = 8, ///< [Validation] generic error code for unsupported versions
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE = 9, ///< [Validation] generic error code for unsupported features
+ OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT = 10, ///< [Validation] generic error code for invalid arguments
+ OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE = 11, ///< [Validation] handle argument is not valid
+ OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER = 12, ///< [Validation] pointer argument may not be nullptr
+ OFFLOAD_RESULT_ERROR_INVALID_SIZE = 13, ///< [Validation] invalid size or dimensions (e.g., must not be zero, or is out of bounds)
+ OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION = 14, ///< [Validation] enumerator argument is not valid
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION = 15, ///< [Validation] enumerator argument is not supported by the device
+ OFFLOAD_RESULT_ERROR_UNKNOWN = 16, ///< Unknown or internal error
+ /// @cond
+ OFFLOAD_RESULT_FORCE_UINT32 = 0x7fffffff
+ /// @endcond
+
+} offload_result_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Get a detailed error message for the last error that occurred on this thread, if it exists
+///
+/// @details
+/// - When an Offload API call returns a return value other than OFFLOAD_RESULT_SUCCESS, the implementation *may* set an additional error message.
+/// - If a further Offload call (excluding this function) is made on the same thread without checking its detailed error message with this function, that message should be considered lost.
+/// - The returned char* is only valid until the next Offload function call on the same thread (excluding further calls to this function.)
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
+ size_t* SizeRet, ///< [out][optional] Pointer to return the size of the available error message. A size of 0 indicates no message.
+ const char** DetailStringRet ///< [out][optional] Pointer to return the error message string.
+);
+
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Retrieves all available platforms
+///
+/// @details
+/// - Multiple calls to this function will return identical platforms handles, in the same order.
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// + `NumEntries == 0 && phPlatforms != NULL`
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
+ uint32_t NumEntries, ///< [in] The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, thenNumEntries should be greater than zero, otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
+ offload_platform_handle_t* phPlatforms, ///< [out][optional] Array of handle of platforms. If NumEntries isless than the number of platforms available, then offloadPlatformGetshall only retrieve that number of platforms.
+ uint32_t* pNumPlatforms ///< [out][optional] returns the total number of platforms available.
+);
+
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Supported platform info
+typedef enum offload_platform_info_t {
+ OFFLOAD_PLATFORM_INFO_NAME = 0, ///< The string denoting name of the platform. The size of the info needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_VENDOR_NAME = 1, ///< The string denoting name of the vendor of the platform. The size of the info needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_VERSION = 2, ///< The string denoting the version of the platform. The size of the info needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_BACKEND = 3, ///< The backend of the platform. Identifies the native backend adapter implementing this platform.
+ /// @cond
+ OFFLOAD_PLATFORM_INFO_FORCE_UINT32 = 0x7fffffff
+ /// @endcond
+
+} offload_platform_info_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Identifies the native backend of the platform
+typedef enum offload_platform_backend_t {
+ OFFLOAD_PLATFORM_BACKEND_UNKNOWN = 0, ///< The backend is not recognized
+ OFFLOAD_PLATFORM_BACKEND_CUDA = 1, ///< The backend is CUDA
+ OFFLOAD_PLATFORM_BACKEND_AMDGPU = 2, ///< The backend is AMDGPU
+ /// @cond
+ OFFLOAD_PLATFORM_BACKEND_FORCE_UINT32 = 0x7fffffff
+ /// @endcond
+
+} offload_platform_backend_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Retrieves various information about platform
+///
+/// @details
+/// - The application may call this function from simultaneous threads.
+/// - The implementation of this function should be lock-free.
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION
+/// + If `propName` is not supported by the platform.
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// + `propSize == 0 && pPropValue != NULL`
+/// + If `propSize` is less than the real number of bytes needed to return the info.
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// + `propSize != 0 && pPropValue == NULL`
+/// + `pPropValue == NULL && pPropSizeRet == NULL`
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_PLATFORM
+/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES
+/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// + `NULL == hPlatform`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
+ offload_platform_handle_t hPlatform, ///< [in] handle of the platform
+ offload_platform_info_t propName, ///< [in] type of the info to retrieve
+ size_t propSize, ///< [in] the number of bytes pointed to by pPlatformInfo.
+ void* pPropValue, ///< [out][optional] array of bytes holding the info.If Size is not equal to or greater to the real number of bytes needed to return the infothen the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is not used.
+ size_t* pPropSizeRet ///< [out][optional] pointer to the actual number of bytes being queried by pPlatformInfo.
+);
+
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Supported device types
+typedef enum offload_device_type_t {
+ OFFLOAD_DEVICE_TYPE_DEFAULT = 0, ///< The default device type as preferred by the runtime
+ OFFLOAD_DEVICE_TYPE_ALL = 1, ///< Devices of all types
+ OFFLOAD_DEVICE_TYPE_GPU = 2, ///< GPU device type
+ OFFLOAD_DEVICE_TYPE_CPU = 3, ///< CPU device type
+ /// @cond
+ OFFLOAD_DEVICE_TYPE_FORCE_UINT32 = 0x7fffffff
+ /// @endcond
+
+} offload_device_type_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Supported device info
+typedef enum offload_device_info_t {
+ OFFLOAD_DEVICE_INFO_TYPE = 0, ///< type of the device
+ OFFLOAD_DEVICE_INFO_PLATFORM = 1, ///< the platform associated with the device
+ OFFLOAD_DEVICE_INFO_NAME = 2, ///< Device name
+ OFFLOAD_DEVICE_INFO_VENDOR = 3, ///< Device vendor
+ OFFLOAD_DEVICE_INFO_DRIVER_VERSION = 4, ///< Driver version
+ /// @cond
+ OFFLOAD_DEVICE_INFO_FORCE_UINT32 = 0x7fffffff
+ /// @endcond
+
+} offload_device_info_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Retrieves devices within a platform
+///
+/// @details
+/// - Multiple calls to this function will return identical device handles, in the same order.
+/// - The number and order of handles returned from this function can be affected by environment variables that filter devices exposed through API.
+/// - The returned devices are taken a reference of and must be released with a subsequent call to olDeviceRelease.
+/// - The application may call this function from simultaneous threads, the implementation must be thread-safe
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// + `NumEntries == 0 && phDevices != NULL`
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// + `NumEntries > 0 && phDevices == NULL`
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_VALUE
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// + `NULL == hPlatform`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
+ offload_platform_handle_t hPlatform, ///< [in] handle of the platform instance
+ offload_device_type_t DeviceType, ///< [in] the type of the devices.
+ uint32_t NumEntries, ///< [in] the number of devices to be added to phDevices.If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
+ offload_device_handle_t* phDevices, ///< [out][optional] array of handle of devices.If NumEntries is less than the number of devices available, then platform shall only retrieve that number of devices.
+ uint32_t* pNumDevices ///< [out][optional] pointer to the number of devices.pNumDevices will be updated with the total number of devices available.
+);
+
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Retrieves various information about device
+///
+/// @details
+/// - The application may call this function from simultaneous threads.
+/// - The implementation of this function should be lock-free.
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION
+/// + If `propName` is not supported by the adapter.
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// + `propSize == 0 && pPropValue != NULL`
+/// + If `propSize` is less than the real number of bytes needed to return the info.
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// + `propSize != 0 && pPropValue == NULL`
+/// + `pPropValue == NULL && pPropSizeRet == NULL`
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_DEVICE
+/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES
+/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY
+/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// + `NULL == hDevice`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
+ offload_device_handle_t hDevice, ///< [in] handle of the device instance
+ offload_device_info_t propName, ///< [in] type of the info to retrieve
+ size_t propSize, ///< [in] the number of bytes pointed to by pPropValue.
+ void* pPropValue, ///< [out][optional] array of bytes holding the info. If propSize is not equal to or greater than the real number of bytes needed to return the info then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPropValue is not used.
+ size_t* pPropSizeRet ///< [out][optional] pointer to the actual size in bytes of the queried propName.
+);
+
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadGetErrorDetails
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_get_error_details_params_t {
+ size_t** pSizeRet;
+ const char*** pDetailStringRet;
+} offload_get_error_details_params_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadPlatformGet
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_platform_get_params_t {
+ uint32_t* pNumEntries;
+ offload_platform_handle_t** pphPlatforms;
+ uint32_t** ppNumPlatforms;
+} offload_platform_get_params_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadPlatformGetInfo
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_platform_get_info_params_t {
+ offload_platform_handle_t* phPlatform;
+ offload_platform_info_t* ppropName;
+ size_t* ppropSize;
+ void** ppPropValue;
+ size_t** ppPropSizeRet;
+} offload_platform_get_info_params_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadDeviceGet
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_device_get_params_t {
+ offload_platform_handle_t* phPlatform;
+ offload_device_type_t* pDeviceType;
+ uint32_t* pNumEntries;
+ offload_device_handle_t** pphDevices;
+ uint32_t** ppNumDevices;
+} offload_device_get_params_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadDeviceGetInfo
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_device_get_info_params_t {
+ offload_device_handle_t* phDevice;
+ offload_device_info_t* ppropName;
+ size_t* ppropSize;
+ void** ppPropValue;
+ size_t** ppPropSizeRet;
+} offload_device_get_info_params_t;
+
+#if defined(__cplusplus)
+} // extern "C"
+#endif
+
diff --git a/offload/new-api/include/offload_entry_points.inc b/offload/new-api/include/offload_entry_points.inc
new file mode 100644
index 00000000000000..ce7b784f341a09
--- /dev/null
+++ b/offload/new-api/include/offload_entry_points.inc
@@ -0,0 +1,195 @@
+//===- Auto-generated file, part of the LLVM/Offload project --------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+///////////////////////////////////////////////////////////////////////////////
+offload_result_t offloadGetErrorDetails_val(
+ size_t* SizeRet, const char** DetailStringRet) {
+ if (true /*enableParameterValidation*/) {
+ }
+
+ return offloadGetErrorDetails_impl(SizeRet, DetailStringRet);
+
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
+ size_t* SizeRet, const char** DetailStringRet) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadGetErrorDetails";
+ }
+
+ offload_result_t result = offloadGetErrorDetails_val(SizeRet, DetailStringRet);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_get_error_details_params_t Params = { &SizeRet, &DetailStringRet};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
+ std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ }
+ }
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_result_t offloadPlatformGet_val(
+ uint32_t NumEntries, offload_platform_handle_t* phPlatforms, uint32_t* pNumPlatforms) {
+ if (true /*enableParameterValidation*/) {
+ if (NumEntries == 0 && phPlatforms != NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ }
+
+ }
+
+ return offloadPlatformGet_impl(NumEntries, phPlatforms, pNumPlatforms);
+
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
+ uint32_t NumEntries, offload_platform_handle_t* phPlatforms, uint32_t* pNumPlatforms) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadPlatformGet";
+ }
+
+ offload_result_t result = offloadPlatformGet_val(NumEntries, phPlatforms, pNumPlatforms);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_platform_get_params_t Params = { &NumEntries, &phPlatforms, &pNumPlatforms};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
+ std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ }
+ }
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_result_t offloadPlatformGetInfo_val(
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+ if (true /*enableParameterValidation*/) {
+ if (propSize == 0 && pPropValue != NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ }
+
+ if (propSize != 0 && pPropValue == NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ }
+
+ if (pPropValue == NULL && pPropSizeRet == NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ }
+
+ if (NULL == hPlatform) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
+ }
+
+ }
+
+ return offloadPlatformGetInfo_impl(hPlatform, propName, propSize, pPropValue, pPropSizeRet);
+
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadPlatformGetInfo";
+ }
+
+ offload_result_t result = offloadPlatformGetInfo_val(hPlatform, propName, propSize, pPropValue, pPropSizeRet);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_platform_get_info_params_t Params = { &hPlatform, &propName, &propSize, &pPropValue, &pPropSizeRet};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
+ std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ }
+ }
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_result_t offloadDeviceGet_val(
+ offload_platform_handle_t hPlatform, offload_device_type_t DeviceType, uint32_t NumEntries, offload_device_handle_t* phDevices, uint32_t* pNumDevices) {
+ if (true /*enableParameterValidation*/) {
+ if (NumEntries == 0 && phDevices != NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ }
+
+ if (NumEntries > 0 && phDevices == NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ }
+
+ if (NULL == hPlatform) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
+ }
+
+ }
+
+ return offloadDeviceGet_impl(hPlatform, DeviceType, NumEntries, phDevices, pNumDevices);
+
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
+ offload_platform_handle_t hPlatform, offload_device_type_t DeviceType, uint32_t NumEntries, offload_device_handle_t* phDevices, uint32_t* pNumDevices) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadDeviceGet";
+ }
+
+ offload_result_t result = offloadDeviceGet_val(hPlatform, DeviceType, NumEntries, phDevices, pNumDevices);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_device_get_params_t Params = { &hPlatform, &DeviceType, &NumEntries, &phDevices, &pNumDevices};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
+ std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ }
+ }
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_result_t offloadDeviceGetInfo_val(
+ offload_device_handle_t hDevice, offload_device_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+ if (true /*enableParameterValidation*/) {
+ if (propSize == 0 && pPropValue != NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ }
+
+ if (propSize != 0 && pPropValue == NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ }
+
+ if (pPropValue == NULL && pPropSizeRet == NULL) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ }
+
+ if (NULL == hDevice) {
+ return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
+ }
+
+ }
+
+ return offloadDeviceGetInfo_impl(hDevice, propName, propSize, pPropValue, pPropSizeRet);
+
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
+ offload_device_handle_t hDevice, offload_device_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadDeviceGetInfo";
+ }
+
+ offload_result_t result = offloadDeviceGetInfo_val(hDevice, propName, propSize, pPropValue, pPropSizeRet);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_device_get_info_params_t Params = { &hDevice, &propName, &propSize, &pPropValue, &pPropSizeRet};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
+ std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ }
+ }
+ return result;
+}
diff --git a/offload/new-api/include/offload_exports b/offload/new-api/include/offload_exports
new file mode 100644
index 00000000000000..ec44d7cc0c2132
--- /dev/null
+++ b/offload/new-api/include/offload_exports
@@ -0,0 +1,10 @@
+VERS1.0 {
+ global:
+ offloadGetErrorDetails;
+ offloadPlatformGet;
+ offloadPlatformGetInfo;
+ offloadDeviceGet;
+ offloadDeviceGetInfo;
+ local:
+ *;
+};
diff --git a/offload/new-api/include/offload_funcs.inc b/offload/new-api/include/offload_funcs.inc
new file mode 100644
index 00000000000000..7160b27a5332cf
--- /dev/null
+++ b/offload/new-api/include/offload_funcs.inc
@@ -0,0 +1,19 @@
+//===- Auto-generated file, part of the LLVM/Offload project --------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#ifndef OFFLOAD_FUNC
+#error Please define the macro OFFLOAD_FUNC(Function)
+#endif
+
+OFFLOAD_FUNC(offloadGetErrorDetails)
+OFFLOAD_FUNC(offloadPlatformGet)
+OFFLOAD_FUNC(offloadPlatformGetInfo)
+OFFLOAD_FUNC(offloadDeviceGet)
+OFFLOAD_FUNC(offloadDeviceGetInfo)
+
+#undef OFFLOAD_FUNC
diff --git a/offload/new-api/include/offload_impl_func_decls.inc b/offload/new-api/include/offload_impl_func_decls.inc
new file mode 100644
index 00000000000000..1405a3e38c7c8c
--- /dev/null
+++ b/offload/new-api/include/offload_impl_func_decls.inc
@@ -0,0 +1,12 @@
+//===- Auto-generated file, part of the LLVM/Offload project --------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+offload_result_t offloadGetErrorDetails_impl(size_t* SizeRet, const char** DetailStringRet);
+offload_impl_result_t offloadPlatformGet_impl(uint32_t NumEntries, offload_platform_handle_t* phPlatforms, uint32_t* pNumPlatforms);
+offload_impl_result_t offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform, offload_platform_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet);
+offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform, offload_device_type_t DeviceType, uint32_t NumEntries, offload_device_handle_t* phDevices, uint32_t* pNumDevices);
+offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice, offload_device_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet);
diff --git a/offload/new-api/include/offload_print.hpp b/offload/new-api/include/offload_print.hpp
new file mode 100644
index 00000000000000..8b14d222a7245c
--- /dev/null
+++ b/offload/new-api/include/offload_print.hpp
@@ -0,0 +1,380 @@
+//===- Auto-generated file, part of the LLVM/Offload project --------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+// Auto-generated file, do not manually edit.
+
+#pragma once
+
+#include <offload_api.h>
+#include <ostream>
+
+
+template <typename T> inline offload_result_t printPtr(std::ostream &os, const T *ptr);
+template <typename T> inline void printTagged(std::ostream &os, const void *ptr, T value, size_t size);
+template <typename T> struct is_handle : std::false_type {};
+template <> struct is_handle<offload_platform_handle_t> : std::true_type {};
+template <> struct is_handle<offload_device_handle_t> : std::true_type {};
+template <> struct is_handle<offload_context_handle_t> : std::true_type {};
+template <typename T> inline constexpr bool is_handle_v = is_handle<T>::value;
+
+inline std::ostream &operator<<(std::ostream &os, enum offload_result_t value);
+inline std::ostream &operator<<(std::ostream &os, enum offload_platform_info_t value);
+inline std::ostream &operator<<(std::ostream &os, enum offload_platform_backend_t value);
+inline std::ostream &operator<<(std::ostream &os, enum offload_device_type_t value);
+inline std::ostream &operator<<(std::ostream &os, enum offload_device_info_t value);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print operator for the offload_result_t type
+/// @returns std::ostream &
+inline std::ostream &operator<<(std::ostream &os, enum offload_result_t value) {
+ switch (value) {
+ case OFFLOAD_RESULT_SUCCESS:
+ os << "OFFLOAD_RESULT_SUCCESS";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_VALUE:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_VALUE";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_PLATFORM:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_PLATFORM";
+ break;
+ case OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND:
+ os << "OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_DEVICE:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_DEVICE";
+ break;
+ case OFFLOAD_RESULT_ERROR_DEVICE_LOST:
+ os << "OFFLOAD_RESULT_ERROR_DEVICE_LOST";
+ break;
+ case OFFLOAD_RESULT_ERROR_UNINITIALIZED:
+ os << "OFFLOAD_RESULT_ERROR_UNINITIALIZED";
+ break;
+ case OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES:
+ os << "OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES";
+ break;
+ case OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION:
+ os << "OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION";
+ break;
+ case OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE:
+ os << "OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_SIZE:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_SIZE";
+ break;
+ case OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION:
+ os << "OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION";
+ break;
+ case OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION:
+ os << "OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION";
+ break;
+ case OFFLOAD_RESULT_ERROR_UNKNOWN:
+ os << "OFFLOAD_RESULT_ERROR_UNKNOWN";
+ break;
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+ return os;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print operator for the offload_platform_info_t type
+/// @returns std::ostream &
+inline std::ostream &operator<<(std::ostream &os, enum offload_platform_info_t value) {
+ switch (value) {
+ case OFFLOAD_PLATFORM_INFO_NAME:
+ os << "OFFLOAD_PLATFORM_INFO_NAME";
+ break;
+ case OFFLOAD_PLATFORM_INFO_VENDOR_NAME:
+ os << "OFFLOAD_PLATFORM_INFO_VENDOR_NAME";
+ break;
+ case OFFLOAD_PLATFORM_INFO_VERSION:
+ os << "OFFLOAD_PLATFORM_INFO_VERSION";
+ break;
+ case OFFLOAD_PLATFORM_INFO_BACKEND:
+ os << "OFFLOAD_PLATFORM_INFO_BACKEND";
+ break;
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+ return os;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print type-tagged offload_platform_info_t enum value
+/// @returns std::ostream &
+template <>
+inline void printTagged(std::ostream &os, const void *ptr, offload_platform_info_t value, size_t size) {
+ if (ptr == NULL) {
+ printPtr(os, ptr);
+ return;
+ }
+
+ switch (value) {
+ case OFFLOAD_PLATFORM_INFO_NAME: {
+ printPtr(os, (const char*) ptr);
+ break;
+ }
+ case OFFLOAD_PLATFORM_INFO_VENDOR_NAME: {
+ printPtr(os, (const char*) ptr);
+ break;
+ }
+ case OFFLOAD_PLATFORM_INFO_VERSION: {
+ printPtr(os, (const char*) ptr);
+ break;
+ }
+ case OFFLOAD_PLATFORM_INFO_BACKEND: {
+ const offload_platform_backend_t * const tptr = (const offload_platform_backend_t * const)ptr;
+ os << (const void *)tptr << " (";
+ os << *tptr;
+ os << ")";
+ break;
+ }
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+}
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print operator for the offload_platform_backend_t type
+/// @returns std::ostream &
+inline std::ostream &operator<<(std::ostream &os, enum offload_platform_backend_t value) {
+ switch (value) {
+ case OFFLOAD_PLATFORM_BACKEND_UNKNOWN:
+ os << "OFFLOAD_PLATFORM_BACKEND_UNKNOWN";
+ break;
+ case OFFLOAD_PLATFORM_BACKEND_CUDA:
+ os << "OFFLOAD_PLATFORM_BACKEND_CUDA";
+ break;
+ case OFFLOAD_PLATFORM_BACKEND_AMDGPU:
+ os << "OFFLOAD_PLATFORM_BACKEND_AMDGPU";
+ break;
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+ return os;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print operator for the offload_device_type_t type
+/// @returns std::ostream &
+inline std::ostream &operator<<(std::ostream &os, enum offload_device_type_t value) {
+ switch (value) {
+ case OFFLOAD_DEVICE_TYPE_DEFAULT:
+ os << "OFFLOAD_DEVICE_TYPE_DEFAULT";
+ break;
+ case OFFLOAD_DEVICE_TYPE_ALL:
+ os << "OFFLOAD_DEVICE_TYPE_ALL";
+ break;
+ case OFFLOAD_DEVICE_TYPE_GPU:
+ os << "OFFLOAD_DEVICE_TYPE_GPU";
+ break;
+ case OFFLOAD_DEVICE_TYPE_CPU:
+ os << "OFFLOAD_DEVICE_TYPE_CPU";
+ break;
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+ return os;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print operator for the offload_device_info_t type
+/// @returns std::ostream &
+inline std::ostream &operator<<(std::ostream &os, enum offload_device_info_t value) {
+ switch (value) {
+ case OFFLOAD_DEVICE_INFO_TYPE:
+ os << "OFFLOAD_DEVICE_INFO_TYPE";
+ break;
+ case OFFLOAD_DEVICE_INFO_PLATFORM:
+ os << "OFFLOAD_DEVICE_INFO_PLATFORM";
+ break;
+ case OFFLOAD_DEVICE_INFO_NAME:
+ os << "OFFLOAD_DEVICE_INFO_NAME";
+ break;
+ case OFFLOAD_DEVICE_INFO_VENDOR:
+ os << "OFFLOAD_DEVICE_INFO_VENDOR";
+ break;
+ case OFFLOAD_DEVICE_INFO_DRIVER_VERSION:
+ os << "OFFLOAD_DEVICE_INFO_DRIVER_VERSION";
+ break;
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+ return os;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Print type-tagged offload_device_info_t enum value
+/// @returns std::ostream &
+template <>
+inline void printTagged(std::ostream &os, const void *ptr, offload_device_info_t value, size_t size) {
+ if (ptr == NULL) {
+ printPtr(os, ptr);
+ return;
+ }
+
+ switch (value) {
+ case OFFLOAD_DEVICE_INFO_TYPE: {
+ const offload_device_type_t * const tptr = (const offload_device_type_t * const)ptr;
+ os << (const void *)tptr << " (";
+ os << *tptr;
+ os << ")";
+ break;
+ }
+ case OFFLOAD_DEVICE_INFO_PLATFORM: {
+ const offload_platform_handle_t * const tptr = (const offload_platform_handle_t * const)ptr;
+ os << (const void *)tptr << " (";
+ os << *tptr;
+ os << ")";
+ break;
+ }
+ case OFFLOAD_DEVICE_INFO_NAME: {
+ printPtr(os, (const char*) ptr);
+ break;
+ }
+ case OFFLOAD_DEVICE_INFO_VENDOR: {
+ printPtr(os, (const char*) ptr);
+ break;
+ }
+ case OFFLOAD_DEVICE_INFO_DRIVER_VERSION: {
+ printPtr(os, (const char*) ptr);
+ break;
+ }
+ default:
+ os << "unknown enumerator";
+ break;
+ }
+}
+
+inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_get_error_details_params_t *params) {
+ os << ".SizeRet = ";
+ printPtr(os, *params->pSizeRet);
+ os << ", ";
+ os << ".DetailStringRet = ";
+ printPtr(os, *params->pDetailStringRet);
+ return os;
+}
+
+inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_platform_get_params_t *params) {
+ os << ".NumEntries = ";
+ os << *params->pNumEntries;
+ os << ", ";
+ os << ".phPlatforms = ";
+ os << "{";
+ for (size_t i = 0; i < *params->pNumEntries; i++){
+ if (i > 0) {
+ os << ", ";
+ }
+ printPtr(os, (*params->pphPlatforms)[i]);
+ }
+ os << "}";
+ os << ", ";
+ os << ".pNumPlatforms = ";
+ printPtr(os, *params->ppNumPlatforms);
+ return os;
+}
+
+inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_platform_get_info_params_t *params) {
+ os << ".hPlatform = ";
+ printPtr(os, *params->phPlatform);
+ os << ", ";
+ os << ".propName = ";
+ os << *params->ppropName;
+ os << ", ";
+ os << ".propSize = ";
+ os << *params->ppropSize;
+ os << ", ";
+ os << ".pPropValue = ";
+ printTagged(os, *params->ppPropValue, *params->ppropName, *params->ppropSize);
+ os << ", ";
+ os << ".pPropSizeRet = ";
+ printPtr(os, *params->ppPropSizeRet);
+ return os;
+}
+
+inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_device_get_params_t *params) {
+ os << ".hPlatform = ";
+ printPtr(os, *params->phPlatform);
+ os << ", ";
+ os << ".DeviceType = ";
+ os << *params->pDeviceType;
+ os << ", ";
+ os << ".NumEntries = ";
+ os << *params->pNumEntries;
+ os << ", ";
+ os << ".phDevices = ";
+ os << "{";
+ for (size_t i = 0; i < *params->pNumEntries; i++){
+ if (i > 0) {
+ os << ", ";
+ }
+ printPtr(os, (*params->pphDevices)[i]);
+ }
+ os << "}";
+ os << ", ";
+ os << ".pNumDevices = ";
+ printPtr(os, *params->ppNumDevices);
+ return os;
+}
+
+inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_device_get_info_params_t *params) {
+ os << ".hDevice = ";
+ printPtr(os, *params->phDevice);
+ os << ", ";
+ os << ".propName = ";
+ os << *params->ppropName;
+ os << ", ";
+ os << ".propSize = ";
+ os << *params->ppropSize;
+ os << ", ";
+ os << ".pPropValue = ";
+ printTagged(os, *params->ppPropValue, *params->ppropName, *params->ppropSize);
+ os << ", ";
+ os << ".pPropSizeRet = ";
+ printPtr(os, *params->ppPropSizeRet);
+ return os;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+// @brief Print pointer value
+template <typename T> inline offload_result_t printPtr(std::ostream &os, const T *ptr) {
+ if (ptr == nullptr) {
+ os << "nullptr";
+ } else if constexpr (std::is_pointer_v<T>) {
+ os << (const void *)(ptr) << " (";
+ printPtr(os, *ptr);
+ os << ")";
+ } else if constexpr (std::is_void_v<T> || is_handle_v<T *>) {
+ os << (const void *)ptr;
+ } else if constexpr (std::is_same_v<std::remove_cv_t< T >, char>) {
+ os << (const void *)(ptr) << " (";
+ os << ptr;
+ os << ")";
+ } else {
+ os << (const void *)(ptr) << " (";
+ os << *ptr;
+ os << ")";
+ }
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
\ No newline at end of file
diff --git a/offload/tools/offload-tblgen/APIGen.cpp b/offload/tools/offload-tblgen/APIGen.cpp
index 6fdf3025f25bb5..b905144abf19f2 100644
--- a/offload/tools/offload-tblgen/APIGen.cpp
+++ b/offload/tools/offload-tblgen/APIGen.cpp
@@ -169,6 +169,7 @@ typedef struct {1} {{
}
void EmitOffloadAPI(RecordKeeper &Records, raw_ostream &OS) {
+ OS << GenericHeader;
OS << FileHeader;
// Generate main API definitions
for (auto *R : Records.getAllDerivedDefinitions("APIObject")) {
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
index 5dbb0e8c22ae04..9f81d9d5582393 100644
--- a/offload/tools/offload-tblgen/EntryPointGen.cpp
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -103,6 +103,7 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
}
void EmitOffloadEntryPoints(RecordKeeper &Records, raw_ostream &OS) {
+ OS << GenericHeader;
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
EmitValidationFunc(FunctionRec{R}, OS);
EmitEntryPointFunc(FunctionRec{R}, OS);
diff --git a/offload/tools/offload-tblgen/FuncsGen.cpp b/offload/tools/offload-tblgen/FuncsGen.cpp
index ee651957abc703..babb65c20c17c5 100644
--- a/offload/tools/offload-tblgen/FuncsGen.cpp
+++ b/offload/tools/offload-tblgen/FuncsGen.cpp
@@ -22,6 +22,7 @@ using namespace offload::tblgen;
// Emit a list of just the API function names
void EmitOffloadFuncNames(RecordKeeper &Records, raw_ostream &OS) {
+ OS << GenericHeader;
OS << R"(
#ifndef OFFLOAD_FUNC
#error Please define the macro OFFLOAD_FUNC(Function)
@@ -50,6 +51,7 @@ void EmitOffloadExports(RecordKeeper &Records, raw_ostream &OS) {
// Emit declarations for every implementation function
void EmitOffloadImplFuncDecls(RecordKeeper &Records, raw_ostream &OS) {
+ OS << GenericHeader;
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
FunctionRec F{R};
// The error details function does not set error details itself, so don't
diff --git a/offload/tools/offload-tblgen/GenCommon.hpp b/offload/tools/offload-tblgen/GenCommon.hpp
index 16ffb3a4d667c8..829f9b525fea92 100644
--- a/offload/tools/offload-tblgen/GenCommon.hpp
+++ b/offload/tools/offload-tblgen/GenCommon.hpp
@@ -19,6 +19,16 @@
#define TAB_4 " "
#define TAB_5 " "
+constexpr auto GenericHeader =
+ R"(//===- Auto-generated file, part of the LLVM/Offload project --------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+)";
+
constexpr auto FileHeader = R"(
// Auto-generated file, do not manually edit.
diff --git a/offload/tools/offload-tblgen/PrintGen.cpp b/offload/tools/offload-tblgen/PrintGen.cpp
index 0c47695eca12b6..b89d45388f0bf8 100644
--- a/offload/tools/offload-tblgen/PrintGen.cpp
+++ b/offload/tools/offload-tblgen/PrintGen.cpp
@@ -133,6 +133,7 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
}
void EmitOffloadPrintHeader(RecordKeeper &Records, raw_ostream &OS) {
+ OS << GenericHeader;
OS << R"""(
// Auto-generated file, do not manually edit.
>From 6d9c1bfda249c3495311e88b8108c02aafa6006d Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Wed, 2 Oct 2024 11:46:45 +0100
Subject: [PATCH 07/12] Fix OffloadGenerate target, clang-format generated
files
---
offload/new-api/API/CMakeLists.txt | 26 ++
offload/new-api/CMakeLists.txt | 23 +-
offload/new-api/include/offload_api.h | 249 +++++++++++-------
.../new-api/include/offload_entry_points.inc | 96 ++++---
offload/new-api/include/offload_exports | 16 +-
.../include/offload_impl_func_decls.inc | 29 +-
offload/new-api/include/offload_print.hpp | 127 +++++----
offload/new-api/src/offload_impl.cpp | 2 +-
offload/tools/offload-tblgen/FuncsGen.cpp | 2 +-
9 files changed, 348 insertions(+), 222 deletions(-)
create mode 100644 offload/new-api/API/CMakeLists.txt
diff --git a/offload/new-api/API/CMakeLists.txt b/offload/new-api/API/CMakeLists.txt
new file mode 100644
index 00000000000000..2f12b48133fbcc
--- /dev/null
+++ b/offload/new-api/API/CMakeLists.txt
@@ -0,0 +1,26 @@
+# The OffloadGenerate target is used to regenerate the generated files in the
+# include directory. These files are checked in with the rest of the source,
+# therefore it is only needed when making changes to the API.
+
+find_program(CLANG_FORMAT clang-format PATHS ${LLVM_TOOLS_BINARY_DIR} NO_DEFAULT_PATH)
+if (CLANG_FORMAT)
+ set(LLVM_TARGET_DEFINITIONS ${CMAKE_CURRENT_SOURCE_DIR}/OffloadAPI.td)
+
+ tablegen(OFFLOAD offload_api.h -gen-api)
+ tablegen(OFFLOAD offload_funcs.inc -gen-func-names)
+ tablegen(OFFLOAD offload_impl_func_decls.inc -gen-impl-func-decls)
+ tablegen(OFFLOAD offload_entry_points.inc -gen-entry-points)
+ tablegen(OFFLOAD offload_print.hpp -gen-print-header)
+ tablegen(OFFLOAD offload_exports -gen-exports)
+
+ set(OFFLOAD_GENERATED_FILES ${TABLEGEN_OUTPUT})
+ add_public_tablegen_target(OffloadGenerate)
+ add_custom_command(TARGET OffloadGenerate POST_BUILD COMMAND ${CLANG_FORMAT}
+ -i ${OFFLOAD_GENERATED_FILES})
+ add_custom_command(TARGET OffloadGenerate POST_BUILD COMMAND ${CMAKE_COMMAND}
+ -E copy_if_different ${OFFLOAD_GENERATED_FILES} "${CMAKE_CURRENT_SOURCE_DIR}/../include")
+else()
+ message(WARNING "clang-format was not found, so the OffloadGenerate target\
+ will not be available. Offload will still build, but you will not be\
+ able to make changes to the API.")
+endif()
diff --git a/offload/new-api/CMakeLists.txt b/offload/new-api/CMakeLists.txt
index 9c82eb841bb8b8..5a994dbc7e2cf7 100644
--- a/offload/new-api/CMakeLists.txt
+++ b/offload/new-api/CMakeLists.txt
@@ -1,27 +1,8 @@
-set(LLVM_TARGET_DEFINITIONS ${CMAKE_CURRENT_SOURCE_DIR}/API/OffloadAPI.td)
-list(APPEND LLVM_TABLEGEN_FLAGS -I ${CMAKE_CURRENT_SOURCE_DIR}/API)
-
-tablegen(OFFLOAD offload_api.h -gen-api)
-tablegen(OFFLOAD offload_funcs.inc -gen-func-names)
-tablegen(OFFLOAD offload_impl_func_decls.inc -gen-impl-func-decls)
-tablegen(OFFLOAD offload_entry_points.inc -gen-entry-points)
-tablegen(OFFLOAD offload_print.hpp -gen-print-header)
-tablegen(OFFLOAD offload_exports -gen-exports)
-
-foreach(itm IN LISTS TABLEGEN_OUTPUT)
- message(${itm})
-endforeach()
-
-set(OFFLOAD_GENERATED_FILES ${TABLEGEN_OUTPUT})
-add_public_tablegen_target(OffloadGenerate)
-
-add_custom_command(TARGET OffloadGenerate POST_BUILD COMMAND ${CMAKE_COMMAND}
- -E copy_if_different ${OFFLOAD_GENERATED_FILES} "${CMAKE_CURRENT_SOURCE_DIR}/include")
+add_subdirectory(API)
add_llvm_library(offload_new SHARED
src/offload_lib.cpp
- src/offload_impl.cpp
- DEPENDS OffloadGenerate)
+ src/offload_impl.cpp)
foreach(plugin IN LISTS LIBOMPTARGET_PLUGINS_TO_BUILD)
target_link_libraries(offload_new PRIVATE omptarget.rtl.${plugin})
diff --git a/offload/new-api/include/offload_api.h b/offload/new-api/include/offload_api.h
index c046496a2e0f9f..99670c68902aba 100644
--- a/offload/new-api/include/offload_api.h
+++ b/offload/new-api/include/offload_api.h
@@ -17,14 +17,13 @@
extern "C" {
#endif
-
///////////////////////////////////////////////////////////////////////////////
#ifndef OFFLOAD_APICALL
#if defined(_WIN32)
/// @brief Calling convention for all API functions
#define OFFLOAD_APICALL __cdecl
#else
-#define OFFLOAD_APICALL
+#define OFFLOAD_APICALL
#endif // defined(_WIN32)
#endif // OFFLOAD_APICALL
@@ -34,7 +33,7 @@ extern "C" {
/// @brief Microsoft-specific dllexport storage-class attribute
#define OFFLOAD_APIEXPORT __declspec(dllexport)
#else
-#define OFFLOAD_APIEXPORT
+#define OFFLOAD_APIEXPORT
#endif // defined(_WIN32)
#endif // OFFLOAD_APIEXPORT
@@ -50,9 +49,9 @@ extern "C" {
#ifndef OFFLOAD_DLLEXPORT
#if __GNUC__ >= 4
/// @brief GCC-specific dllexport storage-class attribute
-#define OFFLOAD_DLLEXPORT __attribute__ ((visibility ("default")))
+#define OFFLOAD_DLLEXPORT __attribute__((visibility("default")))
#else
-#define OFFLOAD_DLLEXPORT
+#define OFFLOAD_DLLEXPORT
#endif // __GNUC__ >= 4
#endif // OFFLOAD_DLLEXPORT
@@ -75,23 +74,35 @@ typedef struct offload_context_handle_t_ *offload_context_handle_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Defines Return/Error codes
typedef enum offload_result_t {
- OFFLOAD_RESULT_SUCCESS = 0, ///< Success
- OFFLOAD_RESULT_ERROR_INVALID_VALUE = 1, ///< Invalid Value
- OFFLOAD_RESULT_ERROR_INVALID_PLATFORM = 2, ///< Invalid platform
- OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND = 3, ///< Device not found
- OFFLOAD_RESULT_ERROR_INVALID_DEVICE = 4, ///< Invalid device
- OFFLOAD_RESULT_ERROR_DEVICE_LOST = 5, ///< Device hung, reset, was removed, or driver update occurred
- OFFLOAD_RESULT_ERROR_UNINITIALIZED = 6, ///< plugin is not initialized or specific entry-point is not implemented
- OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES = 7, ///< Out of resources
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION = 8, ///< [Validation] generic error code for unsupported versions
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE = 9, ///< [Validation] generic error code for unsupported features
- OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT = 10, ///< [Validation] generic error code for invalid arguments
- OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE = 11, ///< [Validation] handle argument is not valid
- OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER = 12, ///< [Validation] pointer argument may not be nullptr
- OFFLOAD_RESULT_ERROR_INVALID_SIZE = 13, ///< [Validation] invalid size or dimensions (e.g., must not be zero, or is out of bounds)
- OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION = 14, ///< [Validation] enumerator argument is not valid
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION = 15, ///< [Validation] enumerator argument is not supported by the device
- OFFLOAD_RESULT_ERROR_UNKNOWN = 16, ///< Unknown or internal error
+ OFFLOAD_RESULT_SUCCESS = 0, ///< Success
+ OFFLOAD_RESULT_ERROR_INVALID_VALUE = 1, ///< Invalid Value
+ OFFLOAD_RESULT_ERROR_INVALID_PLATFORM = 2, ///< Invalid platform
+ OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND = 3, ///< Device not found
+ OFFLOAD_RESULT_ERROR_INVALID_DEVICE = 4, ///< Invalid device
+ OFFLOAD_RESULT_ERROR_DEVICE_LOST =
+ 5, ///< Device hung, reset, was removed, or driver update occurred
+ OFFLOAD_RESULT_ERROR_UNINITIALIZED =
+ 6, ///< plugin is not initialized or specific entry-point is not
+ ///< implemented
+ OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES = 7, ///< Out of resources
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION =
+ 8, ///< [Validation] generic error code for unsupported versions
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE =
+ 9, ///< [Validation] generic error code for unsupported features
+ OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT =
+ 10, ///< [Validation] generic error code for invalid arguments
+ OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE =
+ 11, ///< [Validation] handle argument is not valid
+ OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER =
+ 12, ///< [Validation] pointer argument may not be nullptr
+ OFFLOAD_RESULT_ERROR_INVALID_SIZE =
+ 13, ///< [Validation] invalid size or dimensions (e.g., must not be zero,
+ ///< or is out of bounds)
+ OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION =
+ 14, ///< [Validation] enumerator argument is not valid
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION =
+ 15, ///< [Validation] enumerator argument is not supported by the device
+ OFFLOAD_RESULT_ERROR_UNKNOWN = 16, ///< Unknown or internal error
/// @cond
OFFLOAD_RESULT_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -99,12 +110,18 @@ typedef enum offload_result_t {
} offload_result_t;
///////////////////////////////////////////////////////////////////////////////
-/// @brief Get a detailed error message for the last error that occurred on this thread, if it exists
+/// @brief Get a detailed error message for the last error that occurred on this
+/// thread, if it exists
///
/// @details
-/// - When an Offload API call returns a return value other than OFFLOAD_RESULT_SUCCESS, the implementation *may* set an additional error message.
-/// - If a further Offload call (excluding this function) is made on the same thread without checking its detailed error message with this function, that message should be considered lost.
-/// - The returned char* is only valid until the next Offload function call on the same thread (excluding further calls to this function.)
+/// - When an Offload API call returns a return value other than
+/// OFFLOAD_RESULT_SUCCESS, the implementation *may* set an additional error
+/// message.
+/// - If a further Offload call (excluding this function) is made on the same
+/// thread without checking its detailed error message with this function,
+/// that message should be considered lost.
+/// - The returned char* is only valid until the next Offload function call
+/// on the same thread (excluding further calls to this function.)
///
/// @returns
/// - ::OFFLOAD_RESULT_SUCCESS
@@ -113,16 +130,19 @@ typedef enum offload_result_t {
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
- size_t* SizeRet, ///< [out][optional] Pointer to return the size of the available error message. A size of 0 indicates no message.
- const char** DetailStringRet ///< [out][optional] Pointer to return the error message string.
+ size_t *
+ SizeRet, ///< [out][optional] Pointer to return the size of the
+ ///< available error message. A size of 0 indicates no message.
+ const char **DetailStringRet ///< [out][optional] Pointer to return the
+ ///< error message string.
);
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves all available platforms
///
/// @details
-/// - Multiple calls to this function will return identical platforms handles, in the same order.
+/// - Multiple calls to this function will return identical platforms
+/// handles, in the same order.
///
/// @returns
/// - ::OFFLOAD_RESULT_SUCCESS
@@ -133,19 +153,35 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
- uint32_t NumEntries, ///< [in] The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, thenNumEntries should be greater than zero, otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
- offload_platform_handle_t* phPlatforms, ///< [out][optional] Array of handle of platforms. If NumEntries isless than the number of platforms available, then offloadPlatformGetshall only retrieve that number of platforms.
- uint32_t* pNumPlatforms ///< [out][optional] returns the total number of platforms available.
+ uint32_t
+ NumEntries, ///< [in] The number of platforms to be added to
+ ///< phPlatforms. If phPlatforms is not NULL, thenNumEntries
+ ///< should be greater than zero, otherwise
+ ///< OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
+ offload_platform_handle_t
+ *phPlatforms, ///< [out][optional] Array of handle of platforms. If
+ ///< NumEntries isless than the number of platforms
+ ///< available, then offloadPlatformGetshall only retrieve
+ ///< that number of platforms.
+ uint32_t *pNumPlatforms ///< [out][optional] returns the total number of
+ ///< platforms available.
);
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Supported platform info
typedef enum offload_platform_info_t {
- OFFLOAD_PLATFORM_INFO_NAME = 0, ///< The string denoting name of the platform. The size of the info needs to be dynamically queried.
- OFFLOAD_PLATFORM_INFO_VENDOR_NAME = 1, ///< The string denoting name of the vendor of the platform. The size of the info needs to be dynamically queried.
- OFFLOAD_PLATFORM_INFO_VERSION = 2, ///< The string denoting the version of the platform. The size of the info needs to be dynamically queried.
- OFFLOAD_PLATFORM_INFO_BACKEND = 3, ///< The backend of the platform. Identifies the native backend adapter implementing this platform.
+ OFFLOAD_PLATFORM_INFO_NAME =
+ 0, ///< The string denoting name of the platform. The size of the info
+ ///< needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_VENDOR_NAME =
+ 1, ///< The string denoting name of the vendor of the platform. The size
+ ///< of the info needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_VERSION =
+ 2, ///< The string denoting the version of the platform. The size of the
+ ///< info needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_BACKEND =
+ 3, ///< The backend of the platform. Identifies the native backend adapter
+ ///< implementing this platform.
/// @cond
OFFLOAD_PLATFORM_INFO_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -155,9 +191,9 @@ typedef enum offload_platform_info_t {
///////////////////////////////////////////////////////////////////////////////
/// @brief Identifies the native backend of the platform
typedef enum offload_platform_backend_t {
- OFFLOAD_PLATFORM_BACKEND_UNKNOWN = 0, ///< The backend is not recognized
- OFFLOAD_PLATFORM_BACKEND_CUDA = 1, ///< The backend is CUDA
- OFFLOAD_PLATFORM_BACKEND_AMDGPU = 2, ///< The backend is AMDGPU
+ OFFLOAD_PLATFORM_BACKEND_UNKNOWN = 0, ///< The backend is not recognized
+ OFFLOAD_PLATFORM_BACKEND_CUDA = 1, ///< The backend is CUDA
+ OFFLOAD_PLATFORM_BACKEND_AMDGPU = 2, ///< The backend is AMDGPU
/// @cond
OFFLOAD_PLATFORM_BACKEND_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -179,7 +215,8 @@ typedef enum offload_platform_backend_t {
/// + If `propName` is not supported by the platform.
/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
/// + `propSize == 0 && pPropValue != NULL`
-/// + If `propSize` is less than the real number of bytes needed to return the info.
+/// + If `propSize` is less than the real number of bytes needed to
+/// return the info.
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
/// + `propSize != 0 && pPropValue == NULL`
/// + `pPropValue == NULL && pPropSizeRet == NULL`
@@ -189,21 +226,26 @@ typedef enum offload_platform_backend_t {
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
- offload_platform_handle_t hPlatform, ///< [in] handle of the platform
- offload_platform_info_t propName, ///< [in] type of the info to retrieve
- size_t propSize, ///< [in] the number of bytes pointed to by pPlatformInfo.
- void* pPropValue, ///< [out][optional] array of bytes holding the info.If Size is not equal to or greater to the real number of bytes needed to return the infothen the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is not used.
- size_t* pPropSizeRet ///< [out][optional] pointer to the actual number of bytes being queried by pPlatformInfo.
+ offload_platform_handle_t hPlatform, ///< [in] handle of the platform
+ offload_platform_info_t propName, ///< [in] type of the info to retrieve
+ size_t propSize, ///< [in] the number of bytes pointed to by pPlatformInfo.
+ void *pPropValue, ///< [out][optional] array of bytes holding the info.If
+ ///< Size is not equal to or greater to the real number of
+ ///< bytes needed to return the infothen the
+ ///< OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned
+ ///< and pPlatformInfo is not used.
+ size_t *pPropSizeRet ///< [out][optional] pointer to the actual number of
+ ///< bytes being queried by pPlatformInfo.
);
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Supported device types
typedef enum offload_device_type_t {
- OFFLOAD_DEVICE_TYPE_DEFAULT = 0, ///< The default device type as preferred by the runtime
- OFFLOAD_DEVICE_TYPE_ALL = 1, ///< Devices of all types
- OFFLOAD_DEVICE_TYPE_GPU = 2, ///< GPU device type
- OFFLOAD_DEVICE_TYPE_CPU = 3, ///< CPU device type
+ OFFLOAD_DEVICE_TYPE_DEFAULT =
+ 0, ///< The default device type as preferred by the runtime
+ OFFLOAD_DEVICE_TYPE_ALL = 1, ///< Devices of all types
+ OFFLOAD_DEVICE_TYPE_GPU = 2, ///< GPU device type
+ OFFLOAD_DEVICE_TYPE_CPU = 3, ///< CPU device type
/// @cond
OFFLOAD_DEVICE_TYPE_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -213,11 +255,11 @@ typedef enum offload_device_type_t {
///////////////////////////////////////////////////////////////////////////////
/// @brief Supported device info
typedef enum offload_device_info_t {
- OFFLOAD_DEVICE_INFO_TYPE = 0, ///< type of the device
- OFFLOAD_DEVICE_INFO_PLATFORM = 1, ///< the platform associated with the device
- OFFLOAD_DEVICE_INFO_NAME = 2, ///< Device name
- OFFLOAD_DEVICE_INFO_VENDOR = 3, ///< Device vendor
- OFFLOAD_DEVICE_INFO_DRIVER_VERSION = 4, ///< Driver version
+ OFFLOAD_DEVICE_INFO_TYPE = 0, ///< type of the device
+ OFFLOAD_DEVICE_INFO_PLATFORM = 1, ///< the platform associated with the device
+ OFFLOAD_DEVICE_INFO_NAME = 2, ///< Device name
+ OFFLOAD_DEVICE_INFO_VENDOR = 3, ///< Device vendor
+ OFFLOAD_DEVICE_INFO_DRIVER_VERSION = 4, ///< Driver version
/// @cond
OFFLOAD_DEVICE_INFO_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -228,10 +270,15 @@ typedef enum offload_device_info_t {
/// @brief Retrieves devices within a platform
///
/// @details
-/// - Multiple calls to this function will return identical device handles, in the same order.
-/// - The number and order of handles returned from this function can be affected by environment variables that filter devices exposed through API.
-/// - The returned devices are taken a reference of and must be released with a subsequent call to olDeviceRelease.
-/// - The application may call this function from simultaneous threads, the implementation must be thread-safe
+/// - Multiple calls to this function will return identical device handles,
+/// in the same order.
+/// - The number and order of handles returned from this function can be
+/// affected by environment variables that filter devices exposed through
+/// API.
+/// - The returned devices are taken a reference of and must be released with
+/// a subsequent call to olDeviceRelease.
+/// - The application may call this function from simultaneous threads, the
+/// implementation must be thread-safe
///
/// @returns
/// - ::OFFLOAD_RESULT_SUCCESS
@@ -245,14 +292,22 @@ typedef enum offload_device_info_t {
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
- offload_platform_handle_t hPlatform, ///< [in] handle of the platform instance
- offload_device_type_t DeviceType, ///< [in] the type of the devices.
- uint32_t NumEntries, ///< [in] the number of devices to be added to phDevices.If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
- offload_device_handle_t* phDevices, ///< [out][optional] array of handle of devices.If NumEntries is less than the number of devices available, then platform shall only retrieve that number of devices.
- uint32_t* pNumDevices ///< [out][optional] pointer to the number of devices.pNumDevices will be updated with the total number of devices available.
+ offload_platform_handle_t
+ hPlatform, ///< [in] handle of the platform instance
+ offload_device_type_t DeviceType, ///< [in] the type of the devices.
+ uint32_t NumEntries, ///< [in] the number of devices to be added to
+ ///< phDevices.If phDevices is not NULL, then
+ ///< NumEntries should be greater than zero. Otherwise
+ ///< OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
+ offload_device_handle_t *
+ phDevices, ///< [out][optional] array of handle of devices.If NumEntries
+ ///< is less than the number of devices available, then
+ ///< platform shall only retrieve that number of devices.
+ uint32_t *pNumDevices ///< [out][optional] pointer to the number of
+ ///< devices.pNumDevices will be updated with the
+ ///< total number of devices available.
);
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves various information about device
///
@@ -268,7 +323,8 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
/// + If `propName` is not supported by the adapter.
/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
/// + `propSize == 0 && pPropValue != NULL`
-/// + If `propSize` is less than the real number of bytes needed to return the info.
+/// + If `propSize` is less than the real number of bytes needed to
+/// return the info.
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
/// + `propSize != 0 && pPropValue == NULL`
/// + `pPropValue == NULL && pPropSizeRet == NULL`
@@ -278,65 +334,68 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// + `NULL == hDevice`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
- offload_device_handle_t hDevice, ///< [in] handle of the device instance
- offload_device_info_t propName, ///< [in] type of the info to retrieve
- size_t propSize, ///< [in] the number of bytes pointed to by pPropValue.
- void* pPropValue, ///< [out][optional] array of bytes holding the info. If propSize is not equal to or greater than the real number of bytes needed to return the info then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPropValue is not used.
- size_t* pPropSizeRet ///< [out][optional] pointer to the actual size in bytes of the queried propName.
+ offload_device_handle_t hDevice, ///< [in] handle of the device instance
+ offload_device_info_t propName, ///< [in] type of the info to retrieve
+ size_t propSize, ///< [in] the number of bytes pointed to by pPropValue.
+ void *pPropValue, ///< [out][optional] array of bytes holding the info. If
+ ///< propSize is not equal to or greater than the real
+ ///< number of bytes needed to return the info then the
+ ///< OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned
+ ///< and pPropValue is not used.
+ size_t *pPropSizeRet ///< [out][optional] pointer to the actual size in
+ ///< bytes of the queried propName.
);
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadGetErrorDetails
/// @details Each entry is a pointer to the parameter passed to the function;
typedef struct offload_get_error_details_params_t {
- size_t** pSizeRet;
- const char*** pDetailStringRet;
+ size_t **pSizeRet;
+ const char ***pDetailStringRet;
} offload_get_error_details_params_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadPlatformGet
/// @details Each entry is a pointer to the parameter passed to the function;
typedef struct offload_platform_get_params_t {
- uint32_t* pNumEntries;
- offload_platform_handle_t** pphPlatforms;
- uint32_t** ppNumPlatforms;
+ uint32_t *pNumEntries;
+ offload_platform_handle_t **pphPlatforms;
+ uint32_t **ppNumPlatforms;
} offload_platform_get_params_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadPlatformGetInfo
/// @details Each entry is a pointer to the parameter passed to the function;
typedef struct offload_platform_get_info_params_t {
- offload_platform_handle_t* phPlatform;
- offload_platform_info_t* ppropName;
- size_t* ppropSize;
- void** ppPropValue;
- size_t** ppPropSizeRet;
+ offload_platform_handle_t *phPlatform;
+ offload_platform_info_t *ppropName;
+ size_t *ppropSize;
+ void **ppPropValue;
+ size_t **ppPropSizeRet;
} offload_platform_get_info_params_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadDeviceGet
/// @details Each entry is a pointer to the parameter passed to the function;
typedef struct offload_device_get_params_t {
- offload_platform_handle_t* phPlatform;
- offload_device_type_t* pDeviceType;
- uint32_t* pNumEntries;
- offload_device_handle_t** pphDevices;
- uint32_t** ppNumDevices;
+ offload_platform_handle_t *phPlatform;
+ offload_device_type_t *pDeviceType;
+ uint32_t *pNumEntries;
+ offload_device_handle_t **pphDevices;
+ uint32_t **ppNumDevices;
} offload_device_get_params_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadDeviceGetInfo
/// @details Each entry is a pointer to the parameter passed to the function;
typedef struct offload_device_get_info_params_t {
- offload_device_handle_t* phDevice;
- offload_device_info_t* ppropName;
- size_t* ppropSize;
- void** ppPropValue;
- size_t** ppPropSizeRet;
+ offload_device_handle_t *phDevice;
+ offload_device_info_t *ppropName;
+ size_t *ppropSize;
+ void **ppPropValue;
+ size_t **ppPropSizeRet;
} offload_device_get_info_params_t;
#if defined(__cplusplus)
} // extern "C"
#endif
-
diff --git a/offload/new-api/include/offload_entry_points.inc b/offload/new-api/include/offload_entry_points.inc
index ce7b784f341a09..2910bc414071fc 100644
--- a/offload/new-api/include/offload_entry_points.inc
+++ b/offload/new-api/include/offload_entry_points.inc
@@ -7,24 +7,24 @@
//===----------------------------------------------------------------------===//
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadGetErrorDetails_val(
- size_t* SizeRet, const char** DetailStringRet) {
+offload_result_t offloadGetErrorDetails_val(size_t *SizeRet,
+ const char **DetailStringRet) {
if (true /*enableParameterValidation*/) {
}
return offloadGetErrorDetails_impl(SizeRet, DetailStringRet);
-
}
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
- size_t* SizeRet, const char** DetailStringRet) {
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadGetErrorDetails(size_t *SizeRet, const char **DetailStringRet) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadGetErrorDetails";
}
- offload_result_t result = offloadGetErrorDetails_val(SizeRet, DetailStringRet);
+ offload_result_t result =
+ offloadGetErrorDetails_val(SizeRet, DetailStringRet);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_get_error_details_params_t Params = { &SizeRet, &DetailStringRet};
+ offload_get_error_details_params_t Params = {&SizeRet, &DetailStringRet};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
@@ -35,28 +35,30 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadPlatformGet_val(
- uint32_t NumEntries, offload_platform_handle_t* phPlatforms, uint32_t* pNumPlatforms) {
+offload_result_t offloadPlatformGet_val(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms) {
if (true /*enableParameterValidation*/) {
if (NumEntries == 0 && phPlatforms != NULL) {
return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
}
-
}
return offloadPlatformGet_impl(NumEntries, phPlatforms, pNumPlatforms);
-
}
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
- uint32_t NumEntries, offload_platform_handle_t* phPlatforms, uint32_t* pNumPlatforms) {
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadPlatformGet(uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadPlatformGet";
}
- offload_result_t result = offloadPlatformGet_val(NumEntries, phPlatforms, pNumPlatforms);
+ offload_result_t result =
+ offloadPlatformGet_val(NumEntries, phPlatforms, pNumPlatforms);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_platform_get_params_t Params = { &NumEntries, &phPlatforms, &pNumPlatforms};
+ offload_platform_get_params_t Params = {&NumEntries, &phPlatforms,
+ &pNumPlatforms};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
@@ -67,8 +69,10 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadPlatformGetInfo_val(
- offload_platform_handle_t hPlatform, offload_platform_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+offload_result_t offloadPlatformGetInfo_val(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName,
+ size_t propSize, void *pPropValue,
+ size_t *pPropSizeRet) {
if (true /*enableParameterValidation*/) {
if (propSize == 0 && pPropValue != NULL) {
return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
@@ -85,22 +89,24 @@ offload_result_t offloadPlatformGetInfo_val(
if (NULL == hPlatform) {
return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
}
-
}
- return offloadPlatformGetInfo_impl(hPlatform, propName, propSize, pPropValue, pPropSizeRet);
-
+ return offloadPlatformGetInfo_impl(hPlatform, propName, propSize, pPropValue,
+ pPropSizeRet);
}
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
- offload_platform_handle_t hPlatform, offload_platform_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName,
+ size_t propSize, void *pPropValue, size_t *pPropSizeRet) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadPlatformGetInfo";
}
- offload_result_t result = offloadPlatformGetInfo_val(hPlatform, propName, propSize, pPropValue, pPropSizeRet);
+ offload_result_t result = offloadPlatformGetInfo_val(
+ hPlatform, propName, propSize, pPropValue, pPropSizeRet);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_platform_get_info_params_t Params = { &hPlatform, &propName, &propSize, &pPropValue, &pPropSizeRet};
+ offload_platform_get_info_params_t Params = {
+ &hPlatform, &propName, &propSize, &pPropValue, &pPropSizeRet};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
@@ -111,8 +117,11 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadDeviceGet_val(
- offload_platform_handle_t hPlatform, offload_device_type_t DeviceType, uint32_t NumEntries, offload_device_handle_t* phDevices, uint32_t* pNumDevices) {
+offload_result_t offloadDeviceGet_val(offload_platform_handle_t hPlatform,
+ offload_device_type_t DeviceType,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices) {
if (true /*enableParameterValidation*/) {
if (NumEntries == 0 && phDevices != NULL) {
return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
@@ -125,22 +134,25 @@ offload_result_t offloadDeviceGet_val(
if (NULL == hPlatform) {
return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
}
-
}
- return offloadDeviceGet_impl(hPlatform, DeviceType, NumEntries, phDevices, pNumDevices);
-
+ return offloadDeviceGet_impl(hPlatform, DeviceType, NumEntries, phDevices,
+ pNumDevices);
}
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
- offload_platform_handle_t hPlatform, offload_device_type_t DeviceType, uint32_t NumEntries, offload_device_handle_t* phDevices, uint32_t* pNumDevices) {
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadDeviceGet(offload_platform_handle_t hPlatform,
+ offload_device_type_t DeviceType, uint32_t NumEntries,
+ offload_device_handle_t *phDevices, uint32_t *pNumDevices) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadDeviceGet";
}
- offload_result_t result = offloadDeviceGet_val(hPlatform, DeviceType, NumEntries, phDevices, pNumDevices);
+ offload_result_t result = offloadDeviceGet_val(
+ hPlatform, DeviceType, NumEntries, phDevices, pNumDevices);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_device_get_params_t Params = { &hPlatform, &DeviceType, &NumEntries, &phDevices, &pNumDevices};
+ offload_device_get_params_t Params = {&hPlatform, &DeviceType, &NumEntries,
+ &phDevices, &pNumDevices};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
@@ -151,8 +163,10 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadDeviceGetInfo_val(
- offload_device_handle_t hDevice, offload_device_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+offload_result_t offloadDeviceGetInfo_val(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t propSize, void *pPropValue,
+ size_t *pPropSizeRet) {
if (true /*enableParameterValidation*/) {
if (propSize == 0 && pPropValue != NULL) {
return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
@@ -169,22 +183,24 @@ offload_result_t offloadDeviceGetInfo_val(
if (NULL == hDevice) {
return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
}
-
}
- return offloadDeviceGetInfo_impl(hDevice, propName, propSize, pPropValue, pPropSizeRet);
-
+ return offloadDeviceGetInfo_impl(hDevice, propName, propSize, pPropValue,
+ pPropSizeRet);
}
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
- offload_device_handle_t hDevice, offload_device_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet) {
+ offload_device_handle_t hDevice, offload_device_info_t propName,
+ size_t propSize, void *pPropValue, size_t *pPropSizeRet) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadDeviceGetInfo";
}
- offload_result_t result = offloadDeviceGetInfo_val(hDevice, propName, propSize, pPropValue, pPropSizeRet);
+ offload_result_t result = offloadDeviceGetInfo_val(
+ hDevice, propName, propSize, pPropValue, pPropSizeRet);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_device_get_info_params_t Params = { &hDevice, &propName, &propSize, &pPropValue, &pPropSizeRet};
+ offload_device_get_info_params_t Params = {&hDevice, &propName, &propSize,
+ &pPropValue, &pPropSizeRet};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
diff --git a/offload/new-api/include/offload_exports b/offload/new-api/include/offload_exports
index ec44d7cc0c2132..fb8103ae5e80ef 100644
--- a/offload/new-api/include/offload_exports
+++ b/offload/new-api/include/offload_exports
@@ -1,10 +1,10 @@
VERS1.0 {
- global:
- offloadGetErrorDetails;
- offloadPlatformGet;
- offloadPlatformGetInfo;
- offloadDeviceGet;
- offloadDeviceGetInfo;
- local:
- *;
+global:
+ offloadGetErrorDetails;
+ offloadPlatformGet;
+ offloadPlatformGetInfo;
+ offloadDeviceGet;
+ offloadDeviceGetInfo;
+local:
+ *;
};
diff --git a/offload/new-api/include/offload_impl_func_decls.inc b/offload/new-api/include/offload_impl_func_decls.inc
index 1405a3e38c7c8c..df40c342049549 100644
--- a/offload/new-api/include/offload_impl_func_decls.inc
+++ b/offload/new-api/include/offload_impl_func_decls.inc
@@ -5,8 +5,27 @@
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
-offload_result_t offloadGetErrorDetails_impl(size_t* SizeRet, const char** DetailStringRet);
-offload_impl_result_t offloadPlatformGet_impl(uint32_t NumEntries, offload_platform_handle_t* phPlatforms, uint32_t* pNumPlatforms);
-offload_impl_result_t offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform, offload_platform_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet);
-offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform, offload_device_type_t DeviceType, uint32_t NumEntries, offload_device_handle_t* phDevices, uint32_t* pNumDevices);
-offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice, offload_device_info_t propName, size_t propSize, void* pPropValue, size_t* pPropSizeRet);
+offload_result_t offloadGetErrorDetails_impl(size_t *SizeRet,
+ const char **DetailStringRet);
+
+offload_impl_result_t
+offloadPlatformGet_impl(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms);
+
+offload_impl_result_t
+offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName, size_t propSize,
+ void *pPropValue, size_t *pPropSizeRet);
+
+offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
+ offload_device_type_t DeviceType,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices);
+
+offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t propSize,
+ void *pPropValue,
+ size_t *pPropSizeRet);
diff --git a/offload/new-api/include/offload_print.hpp b/offload/new-api/include/offload_print.hpp
index 8b14d222a7245c..4a0f6d14f10b4e 100644
--- a/offload/new-api/include/offload_print.hpp
+++ b/offload/new-api/include/offload_print.hpp
@@ -13,9 +13,11 @@
#include <offload_api.h>
#include <ostream>
-
-template <typename T> inline offload_result_t printPtr(std::ostream &os, const T *ptr);
-template <typename T> inline void printTagged(std::ostream &os, const void *ptr, T value, size_t size);
+template <typename T>
+inline offload_result_t printPtr(std::ostream &os, const T *ptr);
+template <typename T>
+inline void printTagged(std::ostream &os, const void *ptr, T value,
+ size_t size);
template <typename T> struct is_handle : std::false_type {};
template <> struct is_handle<offload_platform_handle_t> : std::true_type {};
template <> struct is_handle<offload_device_handle_t> : std::true_type {};
@@ -23,10 +25,14 @@ template <> struct is_handle<offload_context_handle_t> : std::true_type {};
template <typename T> inline constexpr bool is_handle_v = is_handle<T>::value;
inline std::ostream &operator<<(std::ostream &os, enum offload_result_t value);
-inline std::ostream &operator<<(std::ostream &os, enum offload_platform_info_t value);
-inline std::ostream &operator<<(std::ostream &os, enum offload_platform_backend_t value);
-inline std::ostream &operator<<(std::ostream &os, enum offload_device_type_t value);
-inline std::ostream &operator<<(std::ostream &os, enum offload_device_info_t value);
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_platform_info_t value);
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_platform_backend_t value);
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_device_type_t value);
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_device_info_t value);
///////////////////////////////////////////////////////////////////////////////
/// @brief Print operator for the offload_result_t type
@@ -94,7 +100,8 @@ inline std::ostream &operator<<(std::ostream &os, enum offload_result_t value) {
///////////////////////////////////////////////////////////////////////////////
/// @brief Print operator for the offload_platform_info_t type
/// @returns std::ostream &
-inline std::ostream &operator<<(std::ostream &os, enum offload_platform_info_t value) {
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_platform_info_t value) {
switch (value) {
case OFFLOAD_PLATFORM_INFO_NAME:
os << "OFFLOAD_PLATFORM_INFO_NAME";
@@ -119,7 +126,8 @@ inline std::ostream &operator<<(std::ostream &os, enum offload_platform_info_t v
/// @brief Print type-tagged offload_platform_info_t enum value
/// @returns std::ostream &
template <>
-inline void printTagged(std::ostream &os, const void *ptr, offload_platform_info_t value, size_t size) {
+inline void printTagged(std::ostream &os, const void *ptr,
+ offload_platform_info_t value, size_t size) {
if (ptr == NULL) {
printPtr(os, ptr);
return;
@@ -127,19 +135,20 @@ inline void printTagged(std::ostream &os, const void *ptr, offload_platform_info
switch (value) {
case OFFLOAD_PLATFORM_INFO_NAME: {
- printPtr(os, (const char*) ptr);
+ printPtr(os, (const char *)ptr);
break;
}
case OFFLOAD_PLATFORM_INFO_VENDOR_NAME: {
- printPtr(os, (const char*) ptr);
+ printPtr(os, (const char *)ptr);
break;
}
case OFFLOAD_PLATFORM_INFO_VERSION: {
- printPtr(os, (const char*) ptr);
+ printPtr(os, (const char *)ptr);
break;
}
case OFFLOAD_PLATFORM_INFO_BACKEND: {
- const offload_platform_backend_t * const tptr = (const offload_platform_backend_t * const)ptr;
+ const offload_platform_backend_t *const tptr =
+ (const offload_platform_backend_t *const)ptr;
os << (const void *)tptr << " (";
os << *tptr;
os << ")";
@@ -153,7 +162,8 @@ inline void printTagged(std::ostream &os, const void *ptr, offload_platform_info
///////////////////////////////////////////////////////////////////////////////
/// @brief Print operator for the offload_platform_backend_t type
/// @returns std::ostream &
-inline std::ostream &operator<<(std::ostream &os, enum offload_platform_backend_t value) {
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_platform_backend_t value) {
switch (value) {
case OFFLOAD_PLATFORM_BACKEND_UNKNOWN:
os << "OFFLOAD_PLATFORM_BACKEND_UNKNOWN";
@@ -174,7 +184,8 @@ inline std::ostream &operator<<(std::ostream &os, enum offload_platform_backend_
///////////////////////////////////////////////////////////////////////////////
/// @brief Print operator for the offload_device_type_t type
/// @returns std::ostream &
-inline std::ostream &operator<<(std::ostream &os, enum offload_device_type_t value) {
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_device_type_t value) {
switch (value) {
case OFFLOAD_DEVICE_TYPE_DEFAULT:
os << "OFFLOAD_DEVICE_TYPE_DEFAULT";
@@ -198,7 +209,8 @@ inline std::ostream &operator<<(std::ostream &os, enum offload_device_type_t val
///////////////////////////////////////////////////////////////////////////////
/// @brief Print operator for the offload_device_info_t type
/// @returns std::ostream &
-inline std::ostream &operator<<(std::ostream &os, enum offload_device_info_t value) {
+inline std::ostream &operator<<(std::ostream &os,
+ enum offload_device_info_t value) {
switch (value) {
case OFFLOAD_DEVICE_INFO_TYPE:
os << "OFFLOAD_DEVICE_INFO_TYPE";
@@ -226,7 +238,8 @@ inline std::ostream &operator<<(std::ostream &os, enum offload_device_info_t val
/// @brief Print type-tagged offload_device_info_t enum value
/// @returns std::ostream &
template <>
-inline void printTagged(std::ostream &os, const void *ptr, offload_device_info_t value, size_t size) {
+inline void printTagged(std::ostream &os, const void *ptr,
+ offload_device_info_t value, size_t size) {
if (ptr == NULL) {
printPtr(os, ptr);
return;
@@ -234,29 +247,31 @@ inline void printTagged(std::ostream &os, const void *ptr, offload_device_info_t
switch (value) {
case OFFLOAD_DEVICE_INFO_TYPE: {
- const offload_device_type_t * const tptr = (const offload_device_type_t * const)ptr;
+ const offload_device_type_t *const tptr =
+ (const offload_device_type_t *const)ptr;
os << (const void *)tptr << " (";
os << *tptr;
os << ")";
break;
}
case OFFLOAD_DEVICE_INFO_PLATFORM: {
- const offload_platform_handle_t * const tptr = (const offload_platform_handle_t * const)ptr;
+ const offload_platform_handle_t *const tptr =
+ (const offload_platform_handle_t *const)ptr;
os << (const void *)tptr << " (";
os << *tptr;
os << ")";
break;
}
case OFFLOAD_DEVICE_INFO_NAME: {
- printPtr(os, (const char*) ptr);
+ printPtr(os, (const char *)ptr);
break;
}
case OFFLOAD_DEVICE_INFO_VENDOR: {
- printPtr(os, (const char*) ptr);
+ printPtr(os, (const char *)ptr);
break;
}
case OFFLOAD_DEVICE_INFO_DRIVER_VERSION: {
- printPtr(os, (const char*) ptr);
+ printPtr(os, (const char *)ptr);
break;
}
default:
@@ -265,7 +280,9 @@ inline void printTagged(std::ostream &os, const void *ptr, offload_device_info_t
}
}
-inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_get_error_details_params_t *params) {
+inline std::ostream &operator<<(
+ std::ostream &os,
+ [[maybe_unused]] const struct offload_get_error_details_params_t *params) {
os << ".SizeRet = ";
printPtr(os, *params->pSizeRet);
os << ", ";
@@ -274,15 +291,17 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
return os;
}
-inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_platform_get_params_t *params) {
+inline std::ostream &operator<<(
+ std::ostream &os,
+ [[maybe_unused]] const struct offload_platform_get_params_t *params) {
os << ".NumEntries = ";
os << *params->pNumEntries;
os << ", ";
os << ".phPlatforms = ";
os << "{";
- for (size_t i = 0; i < *params->pNumEntries; i++){
+ for (size_t i = 0; i < *params->pNumEntries; i++) {
if (i > 0) {
- os << ", ";
+ os << ", ";
}
printPtr(os, (*params->pphPlatforms)[i]);
}
@@ -293,7 +312,9 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
return os;
}
-inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_platform_get_info_params_t *params) {
+inline std::ostream &operator<<(
+ std::ostream &os,
+ [[maybe_unused]] const struct offload_platform_get_info_params_t *params) {
os << ".hPlatform = ";
printPtr(os, *params->phPlatform);
os << ", ";
@@ -311,7 +332,9 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
return os;
}
-inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_device_get_params_t *params) {
+inline std::ostream &
+operator<<(std::ostream &os,
+ [[maybe_unused]] const struct offload_device_get_params_t *params) {
os << ".hPlatform = ";
printPtr(os, *params->phPlatform);
os << ", ";
@@ -323,9 +346,9 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
os << ", ";
os << ".phDevices = ";
os << "{";
- for (size_t i = 0; i < *params->pNumEntries; i++){
+ for (size_t i = 0; i < *params->pNumEntries; i++) {
if (i > 0) {
- os << ", ";
+ os << ", ";
}
printPtr(os, (*params->pphDevices)[i]);
}
@@ -336,7 +359,9 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
return os;
}
-inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct offload_device_get_info_params_t *params) {
+inline std::ostream &operator<<(
+ std::ostream &os,
+ [[maybe_unused]] const struct offload_device_get_info_params_t *params) {
os << ".hDevice = ";
printPtr(os, *params->phDevice);
os << ", ";
@@ -356,25 +381,25 @@ inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct
///////////////////////////////////////////////////////////////////////////////
// @brief Print pointer value
-template <typename T> inline offload_result_t printPtr(std::ostream &os, const T *ptr) {
- if (ptr == nullptr) {
- os << "nullptr";
- } else if constexpr (std::is_pointer_v<T>) {
- os << (const void *)(ptr) << " (";
- printPtr(os, *ptr);
- os << ")";
- } else if constexpr (std::is_void_v<T> || is_handle_v<T *>) {
- os << (const void *)ptr;
- } else if constexpr (std::is_same_v<std::remove_cv_t< T >, char>) {
- os << (const void *)(ptr) << " (";
- os << ptr;
- os << ")";
- } else {
- os << (const void *)(ptr) << " (";
- os << *ptr;
- os << ")";
- }
+template <typename T>
+inline offload_result_t printPtr(std::ostream &os, const T *ptr) {
+ if (ptr == nullptr) {
+ os << "nullptr";
+ } else if constexpr (std::is_pointer_v<T>) {
+ os << (const void *)(ptr) << " (";
+ printPtr(os, *ptr);
+ os << ")";
+ } else if constexpr (std::is_void_v<T> || is_handle_v<T *>) {
+ os << (const void *)ptr;
+ } else if constexpr (std::is_same_v<std::remove_cv_t<T>, char>) {
+ os << (const void *)(ptr) << " (";
+ os << ptr;
+ os << ")";
+ } else {
+ os << (const void *)(ptr) << " (";
+ os << *ptr;
+ os << ")";
+ }
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
}
-
\ No newline at end of file
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
index b544a847a8dbd2..43112bba448045 100644
--- a/offload/new-api/src/offload_impl.cpp
+++ b/offload/new-api/src/offload_impl.cpp
@@ -11,9 +11,9 @@
//
//===----------------------------------------------------------------------===//
+#include "offload_impl.hpp"
#include "PluginManager.h"
#include "helpers.hpp"
-#include "offload_impl.hpp"
#include "llvm/Support/FormatVariadic.h"
#include <offload_api.h>
diff --git a/offload/tools/offload-tblgen/FuncsGen.cpp b/offload/tools/offload-tblgen/FuncsGen.cpp
index babb65c20c17c5..033ed8db3240ad 100644
--- a/offload/tools/offload-tblgen/FuncsGen.cpp
+++ b/offload/tools/offload-tblgen/FuncsGen.cpp
@@ -70,6 +70,6 @@ void EmitOffloadImplFuncDecls(RecordKeeper &Records, raw_ostream &OS) {
OS << ");";
}
}
- OS << "\n";
+ OS << "\n\n";
}
}
>From dd266549e7ea66b553c51c12bd1ee21b91fe2e4f Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Wed, 2 Oct 2024 16:15:53 +0100
Subject: [PATCH 08/12] Fix offload header install location
---
offload/new-api/CMakeLists.txt | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/offload/new-api/CMakeLists.txt b/offload/new-api/CMakeLists.txt
index 5a994dbc7e2cf7..a79b88303bb4d7 100644
--- a/offload/new-api/CMakeLists.txt
+++ b/offload/new-api/CMakeLists.txt
@@ -27,4 +27,4 @@ set_target_properties(offload_new PROPERTIES
BUILD_RPATH "$ORIGIN:${CMAKE_CURRENT_BINARY_DIR}/..")
install(TARGETS offload_new LIBRARY COMPONENT offload_new DESTINATION "${OFFLOAD_INSTALL_LIBDIR}")
-install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/offload_api.h DESTINATION ${CMAKE_INSTALL_PREFIX}/include/offload)
+install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/include/offload_api.h DESTINATION ${CMAKE_INSTALL_PREFIX}/include/offload)
>From 585a239a1222851a4d97c58733ed643d9edb3cbd Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Wed, 2 Oct 2024 17:19:52 +0100
Subject: [PATCH 09/12] Tidy generated comments etc
---
offload/new-api/API/Common.td | 22 +-
offload/new-api/API/Device.td | 8 +-
offload/new-api/API/Platform.td | 14 +-
offload/new-api/include/offload_api.h | 242 +++++++++++----------
offload/tools/offload-tblgen/APIGen.cpp | 14 +-
offload/tools/offload-tblgen/GenCommon.hpp | 2 +-
6 files changed, 155 insertions(+), 147 deletions(-)
diff --git a/offload/new-api/API/Common.td b/offload/new-api/API/Common.td
index 55d03d4ff1e783..8de5475af99371 100644
--- a/offload/new-api/API/Common.td
+++ b/offload/new-api/API/Common.td
@@ -29,12 +29,6 @@ def : Macro {
let alt_value = "";
}
-def : Typedef {
- let name = "offload_bool_t";
- let value = "uint8_t";
- let desc = "compiler-independent type";
-}
-
def : Handle {
let name = "offload_platform_handle_t";
let desc = "Handle of a platform instance";
@@ -62,14 +56,14 @@ def : Enum {
Etor<"ERROR_DEVICE_LOST", "Device hung, reset, was removed, or driver update occurred">,
Etor<"ERROR_UNINITIALIZED", "plugin is not initialized or specific entry-point is not implemented">,
Etor<"ERROR_OUT_OF_RESOURCES", "Out of resources">,
- Etor<"ERROR_UNSUPPORTED_VERSION", "[Validation] generic error code for unsupported versions">,
- Etor<"ERROR_UNSUPPORTED_FEATURE", "[Validation] generic error code for unsupported features">,
- Etor<"ERROR_INVALID_ARGUMENT", "[Validation] generic error code for invalid arguments">,
- Etor<"ERROR_INVALID_NULL_HANDLE", "[Validation] handle argument is not valid">,
- Etor<"ERROR_INVALID_NULL_POINTER", "[Validation] pointer argument may not be nullptr">,
- Etor<"ERROR_INVALID_SIZE", "[Validation] invalid size or dimensions (e.g., must not be zero, or is out of bounds)">,
- Etor<"ERROR_INVALID_ENUMERATION", "[Validation] enumerator argument is not valid">,
- Etor<"ERROR_UNSUPPORTED_ENUMERATION", "[Validation] enumerator argument is not supported by the device">,
+ Etor<"ERROR_UNSUPPORTED_VERSION", "generic error code for unsupported versions">,
+ Etor<"ERROR_UNSUPPORTED_FEATURE", "generic error code for unsupported features">,
+ Etor<"ERROR_INVALID_ARGUMENT", "generic error code for invalid arguments">,
+ Etor<"ERROR_INVALID_NULL_HANDLE", "handle argument is not valid">,
+ Etor<"ERROR_INVALID_NULL_POINTER", "pointer argument may not be nullptr">,
+ Etor<"ERROR_INVALID_SIZE", "invalid size or dimensions (e.g., must not be zero, or is out of bounds)">,
+ Etor<"ERROR_INVALID_ENUMERATION", "enumerator argument is not valid">,
+ Etor<"ERROR_UNSUPPORTED_ENUMERATION", "enumerator argument is not supported by the device">,
Etor<"ERROR_UNKNOWN", "Unknown or internal error">
];
}
diff --git a/offload/new-api/API/Device.td b/offload/new-api/API/Device.td
index 2a330d7ef7c4cb..837d7071853f5c 100644
--- a/offload/new-api/API/Device.td
+++ b/offload/new-api/API/Device.td
@@ -46,13 +46,13 @@ def : Function {
let params = [
Param<"offload_platform_handle_t", "hPlatform", "handle of the platform instance", PARAM_IN>,
Param<"offload_device_type_t", "DeviceType", "the type of the devices.", PARAM_IN>,
- Param<"uint32_t", "NumEntries", "the number of devices to be added to phDevices."
- "If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE"
+ Param<"uint32_t", "NumEntries", "the number of devices to be added to phDevices. "
+ "If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE "
"will be returned.", PARAM_IN>,
- RangedParam<"offload_device_handle_t*", "phDevices", "array of handle of devices."
+ RangedParam<"offload_device_handle_t*", "phDevices", "Array of device handles. "
"If NumEntries is less than the number of devices available, then platform shall only retrieve that number of devices.", PARAM_OUT_OPTIONAL,
Range<"0", "NumEntries">>,
- Param<"uint32_t*", "pNumDevices", "pointer to the number of devices."
+ Param<"uint32_t*", "pNumDevices", "pointer to the number of devices. "
"pNumDevices will be updated with the total number of devices available.", PARAM_OUT_OPTIONAL>
];
let returns = [
diff --git a/offload/new-api/API/Platform.td b/offload/new-api/API/Platform.td
index 71af04bb831998..d280da1156c798 100644
--- a/offload/new-api/API/Platform.td
+++ b/offload/new-api/API/Platform.td
@@ -17,13 +17,13 @@ def : Function {
];
let params = [
Param<"uint32_t", "NumEntries",
- "The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, then"
- "NumEntries should be greater than zero, otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE"
+ "The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, then "
+ "NumEntries should be greater than zero, otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE "
"will be returned.", PARAM_IN>,
RangedParam<"offload_platform_handle_t*", "phPlatforms",
- "Array of handle of platforms. If NumEntries is"
- "less than the number of platforms available, then offloadPlatformGet"
- "shall only retrieve that number of platforms.",
+ "Array of handle of platforms. If NumEntries is less than the number of "
+ "platforms available, then offloadPlatformGet shall only retrieve that "
+ "number of platforms.",
PARAM_OUT_OPTIONAL, Range<"0", "NumEntries">>,
Param<"uint32_t*",
"pNumPlatforms", "returns the total number of platforms available.",
@@ -69,8 +69,8 @@ def : Function {
Param<"offload_platform_handle_t", "hPlatform", "handle of the platform", PARAM_IN>,
Param<"offload_platform_info_t", "propName", "type of the info to retrieve", PARAM_IN>,
Param<"size_t", "propSize", "the number of bytes pointed to by pPlatformInfo.", PARAM_IN>,
- TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info."
- "If Size is not equal to or greater to the real number of bytes needed to return the info"
+ TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info. "
+ "If Size is not equal to or greater to the real number of bytes needed to return the info "
"then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is not used.", PARAM_OUT_OPTIONAL,
TypeInfo<"propName" , "propSize">>,
Param<"size_t*", "pPropSizeRet", "pointer to the actual number of bytes being queried by pPlatformInfo.", PARAM_OUT_OPTIONAL>
diff --git a/offload/new-api/include/offload_api.h b/offload/new-api/include/offload_api.h
index 99670c68902aba..8e70efa27774d1 100644
--- a/offload/new-api/include/offload_api.h
+++ b/offload/new-api/include/offload_api.h
@@ -55,10 +55,6 @@ extern "C" {
#endif // __GNUC__ >= 4
#endif // OFFLOAD_DLLEXPORT
-///////////////////////////////////////////////////////////////////////////////
-/// @brief compiler-independent type
-typedef uint8_t offload_bool_t;
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Handle of a platform instance
typedef struct offload_platform_handle_t_ *offload_platform_handle_t;
@@ -74,35 +70,40 @@ typedef struct offload_context_handle_t_ *offload_context_handle_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Defines Return/Error codes
typedef enum offload_result_t {
- OFFLOAD_RESULT_SUCCESS = 0, ///< Success
- OFFLOAD_RESULT_ERROR_INVALID_VALUE = 1, ///< Invalid Value
- OFFLOAD_RESULT_ERROR_INVALID_PLATFORM = 2, ///< Invalid platform
- OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND = 3, ///< Device not found
- OFFLOAD_RESULT_ERROR_INVALID_DEVICE = 4, ///< Invalid device
- OFFLOAD_RESULT_ERROR_DEVICE_LOST =
- 5, ///< Device hung, reset, was removed, or driver update occurred
- OFFLOAD_RESULT_ERROR_UNINITIALIZED =
- 6, ///< plugin is not initialized or specific entry-point is not
- ///< implemented
- OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES = 7, ///< Out of resources
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION =
- 8, ///< [Validation] generic error code for unsupported versions
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE =
- 9, ///< [Validation] generic error code for unsupported features
- OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT =
- 10, ///< [Validation] generic error code for invalid arguments
- OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE =
- 11, ///< [Validation] handle argument is not valid
- OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER =
- 12, ///< [Validation] pointer argument may not be nullptr
- OFFLOAD_RESULT_ERROR_INVALID_SIZE =
- 13, ///< [Validation] invalid size or dimensions (e.g., must not be zero,
- ///< or is out of bounds)
- OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION =
- 14, ///< [Validation] enumerator argument is not valid
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION =
- 15, ///< [Validation] enumerator argument is not supported by the device
- OFFLOAD_RESULT_ERROR_UNKNOWN = 16, ///< Unknown or internal error
+ /// Success
+ OFFLOAD_RESULT_SUCCESS = 0,
+ /// Invalid Value
+ OFFLOAD_RESULT_ERROR_INVALID_VALUE = 1,
+ /// Invalid platform
+ OFFLOAD_RESULT_ERROR_INVALID_PLATFORM = 2,
+ /// Device not found
+ OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND = 3,
+ /// Invalid device
+ OFFLOAD_RESULT_ERROR_INVALID_DEVICE = 4,
+ /// Device hung, reset, was removed, or driver update occurred
+ OFFLOAD_RESULT_ERROR_DEVICE_LOST = 5,
+ /// plugin is not initialized or specific entry-point is not implemented
+ OFFLOAD_RESULT_ERROR_UNINITIALIZED = 6,
+ /// Out of resources
+ OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES = 7,
+ /// generic error code for unsupported versions
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION = 8,
+ /// generic error code for unsupported features
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE = 9,
+ /// generic error code for invalid arguments
+ OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT = 10,
+ /// handle argument is not valid
+ OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE = 11,
+ /// pointer argument may not be nullptr
+ OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER = 12,
+ /// invalid size or dimensions (e.g., must not be zero, or is out of bounds)
+ OFFLOAD_RESULT_ERROR_INVALID_SIZE = 13,
+ /// enumerator argument is not valid
+ OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION = 14,
+ /// enumerator argument is not supported by the device
+ OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION = 15,
+ /// Unknown or internal error
+ OFFLOAD_RESULT_ERROR_UNKNOWN = 16,
/// @cond
OFFLOAD_RESULT_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -130,12 +131,11 @@ typedef enum offload_result_t {
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
- size_t *
- SizeRet, ///< [out][optional] Pointer to return the size of the
- ///< available error message. A size of 0 indicates no message.
- const char **DetailStringRet ///< [out][optional] Pointer to return the
- ///< error message string.
-);
+ // [out][optional] Pointer to return the size of the available error
+ // message. A size of 0 indicates no message.
+ size_t *SizeRet,
+ // [out][optional] Pointer to return the error message string.
+ const char **DetailStringRet);
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves all available platforms
@@ -153,35 +153,32 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
- uint32_t
- NumEntries, ///< [in] The number of platforms to be added to
- ///< phPlatforms. If phPlatforms is not NULL, thenNumEntries
- ///< should be greater than zero, otherwise
- ///< OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
- offload_platform_handle_t
- *phPlatforms, ///< [out][optional] Array of handle of platforms. If
- ///< NumEntries isless than the number of platforms
- ///< available, then offloadPlatformGetshall only retrieve
- ///< that number of platforms.
- uint32_t *pNumPlatforms ///< [out][optional] returns the total number of
- ///< platforms available.
-);
+ // [in] The number of platforms to be added to phPlatforms. If phPlatforms
+ // is not NULL, then NumEntries should be greater than zero, otherwise
+ // OFFLOAD_RESULT_ERROR_INVALID_SIZE will be returned.
+ uint32_t NumEntries,
+ // [out][optional] Array of handle of platforms. If NumEntries is less than
+ // the number of platforms available, then offloadPlatformGet shall only
+ // retrieve that number of platforms.
+ offload_platform_handle_t *phPlatforms,
+ // [out][optional] returns the total number of platforms available.
+ uint32_t *pNumPlatforms);
///////////////////////////////////////////////////////////////////////////////
/// @brief Supported platform info
typedef enum offload_platform_info_t {
- OFFLOAD_PLATFORM_INFO_NAME =
- 0, ///< The string denoting name of the platform. The size of the info
- ///< needs to be dynamically queried.
- OFFLOAD_PLATFORM_INFO_VENDOR_NAME =
- 1, ///< The string denoting name of the vendor of the platform. The size
- ///< of the info needs to be dynamically queried.
- OFFLOAD_PLATFORM_INFO_VERSION =
- 2, ///< The string denoting the version of the platform. The size of the
- ///< info needs to be dynamically queried.
- OFFLOAD_PLATFORM_INFO_BACKEND =
- 3, ///< The backend of the platform. Identifies the native backend adapter
- ///< implementing this platform.
+ /// The string denoting name of the platform. The size of the info needs to be
+ /// dynamically queried.
+ OFFLOAD_PLATFORM_INFO_NAME = 0,
+ /// The string denoting name of the vendor of the platform. The size of the
+ /// info needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_VENDOR_NAME = 1,
+ /// The string denoting the version of the platform. The size of the info
+ /// needs to be dynamically queried.
+ OFFLOAD_PLATFORM_INFO_VERSION = 2,
+ /// The backend of the platform. Identifies the native backend adapter
+ /// implementing this platform.
+ OFFLOAD_PLATFORM_INFO_BACKEND = 3,
/// @cond
OFFLOAD_PLATFORM_INFO_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -191,9 +188,12 @@ typedef enum offload_platform_info_t {
///////////////////////////////////////////////////////////////////////////////
/// @brief Identifies the native backend of the platform
typedef enum offload_platform_backend_t {
- OFFLOAD_PLATFORM_BACKEND_UNKNOWN = 0, ///< The backend is not recognized
- OFFLOAD_PLATFORM_BACKEND_CUDA = 1, ///< The backend is CUDA
- OFFLOAD_PLATFORM_BACKEND_AMDGPU = 2, ///< The backend is AMDGPU
+ /// The backend is not recognized
+ OFFLOAD_PLATFORM_BACKEND_UNKNOWN = 0,
+ /// The backend is CUDA
+ OFFLOAD_PLATFORM_BACKEND_CUDA = 1,
+ /// The backend is AMDGPU
+ OFFLOAD_PLATFORM_BACKEND_AMDGPU = 2,
/// @cond
OFFLOAD_PLATFORM_BACKEND_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -226,26 +226,32 @@ typedef enum offload_platform_backend_t {
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
- offload_platform_handle_t hPlatform, ///< [in] handle of the platform
- offload_platform_info_t propName, ///< [in] type of the info to retrieve
- size_t propSize, ///< [in] the number of bytes pointed to by pPlatformInfo.
- void *pPropValue, ///< [out][optional] array of bytes holding the info.If
- ///< Size is not equal to or greater to the real number of
- ///< bytes needed to return the infothen the
- ///< OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned
- ///< and pPlatformInfo is not used.
- size_t *pPropSizeRet ///< [out][optional] pointer to the actual number of
- ///< bytes being queried by pPlatformInfo.
-);
+ // [in] handle of the platform
+ offload_platform_handle_t hPlatform,
+ // [in] type of the info to retrieve
+ offload_platform_info_t propName,
+ // [in] the number of bytes pointed to by pPlatformInfo.
+ size_t propSize,
+ // [out][optional] array of bytes holding the info. If Size is not equal to
+ // or greater to the real number of bytes needed to return the info then the
+ // OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is
+ // not used.
+ void *pPropValue,
+ // [out][optional] pointer to the actual number of bytes being queried by
+ // pPlatformInfo.
+ size_t *pPropSizeRet);
///////////////////////////////////////////////////////////////////////////////
/// @brief Supported device types
typedef enum offload_device_type_t {
- OFFLOAD_DEVICE_TYPE_DEFAULT =
- 0, ///< The default device type as preferred by the runtime
- OFFLOAD_DEVICE_TYPE_ALL = 1, ///< Devices of all types
- OFFLOAD_DEVICE_TYPE_GPU = 2, ///< GPU device type
- OFFLOAD_DEVICE_TYPE_CPU = 3, ///< CPU device type
+ /// The default device type as preferred by the runtime
+ OFFLOAD_DEVICE_TYPE_DEFAULT = 0,
+ /// Devices of all types
+ OFFLOAD_DEVICE_TYPE_ALL = 1,
+ /// GPU device type
+ OFFLOAD_DEVICE_TYPE_GPU = 2,
+ /// CPU device type
+ OFFLOAD_DEVICE_TYPE_CPU = 3,
/// @cond
OFFLOAD_DEVICE_TYPE_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -255,11 +261,16 @@ typedef enum offload_device_type_t {
///////////////////////////////////////////////////////////////////////////////
/// @brief Supported device info
typedef enum offload_device_info_t {
- OFFLOAD_DEVICE_INFO_TYPE = 0, ///< type of the device
- OFFLOAD_DEVICE_INFO_PLATFORM = 1, ///< the platform associated with the device
- OFFLOAD_DEVICE_INFO_NAME = 2, ///< Device name
- OFFLOAD_DEVICE_INFO_VENDOR = 3, ///< Device vendor
- OFFLOAD_DEVICE_INFO_DRIVER_VERSION = 4, ///< Driver version
+ /// type of the device
+ OFFLOAD_DEVICE_INFO_TYPE = 0,
+ /// the platform associated with the device
+ OFFLOAD_DEVICE_INFO_PLATFORM = 1,
+ /// Device name
+ OFFLOAD_DEVICE_INFO_NAME = 2,
+ /// Device vendor
+ OFFLOAD_DEVICE_INFO_VENDOR = 3,
+ /// Driver version
+ OFFLOAD_DEVICE_INFO_DRIVER_VERSION = 4,
/// @cond
OFFLOAD_DEVICE_INFO_FORCE_UINT32 = 0x7fffffff
/// @endcond
@@ -292,21 +303,21 @@ typedef enum offload_device_info_t {
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
- offload_platform_handle_t
- hPlatform, ///< [in] handle of the platform instance
- offload_device_type_t DeviceType, ///< [in] the type of the devices.
- uint32_t NumEntries, ///< [in] the number of devices to be added to
- ///< phDevices.If phDevices is not NULL, then
- ///< NumEntries should be greater than zero. Otherwise
- ///< OFFLOAD_RESULT_ERROR_INVALID_SIZEwill be returned.
- offload_device_handle_t *
- phDevices, ///< [out][optional] array of handle of devices.If NumEntries
- ///< is less than the number of devices available, then
- ///< platform shall only retrieve that number of devices.
- uint32_t *pNumDevices ///< [out][optional] pointer to the number of
- ///< devices.pNumDevices will be updated with the
- ///< total number of devices available.
-);
+ // [in] handle of the platform instance
+ offload_platform_handle_t hPlatform,
+ // [in] the type of the devices.
+ offload_device_type_t DeviceType,
+ // [in] the number of devices to be added to phDevices. If phDevices is not
+ // NULL, then NumEntries should be greater than zero. Otherwise
+ // OFFLOAD_RESULT_ERROR_INVALID_SIZE will be returned.
+ uint32_t NumEntries,
+ // [out][optional] Array of device handles. If NumEntries is less than the
+ // number of devices available, then platform shall only retrieve that
+ // number of devices.
+ offload_device_handle_t *phDevices,
+ // [out][optional] pointer to the number of devices. pNumDevices will be
+ // updated with the total number of devices available.
+ uint32_t *pNumDevices);
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves various information about device
@@ -334,17 +345,20 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
/// + `NULL == hDevice`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
- offload_device_handle_t hDevice, ///< [in] handle of the device instance
- offload_device_info_t propName, ///< [in] type of the info to retrieve
- size_t propSize, ///< [in] the number of bytes pointed to by pPropValue.
- void *pPropValue, ///< [out][optional] array of bytes holding the info. If
- ///< propSize is not equal to or greater than the real
- ///< number of bytes needed to return the info then the
- ///< OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned
- ///< and pPropValue is not used.
- size_t *pPropSizeRet ///< [out][optional] pointer to the actual size in
- ///< bytes of the queried propName.
-);
+ // [in] handle of the device instance
+ offload_device_handle_t hDevice,
+ // [in] type of the info to retrieve
+ offload_device_info_t propName,
+ // [in] the number of bytes pointed to by pPropValue.
+ size_t propSize,
+ // [out][optional] array of bytes holding the info. If propSize is not equal
+ // to or greater than the real number of bytes needed to return the info
+ // then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and
+ // pPropValue is not used.
+ void *pPropValue,
+ // [out][optional] pointer to the actual size in bytes of the queried
+ // propName.
+ size_t *pPropSizeRet);
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadGetErrorDetails
diff --git a/offload/tools/offload-tblgen/APIGen.cpp b/offload/tools/offload-tblgen/APIGen.cpp
index b905144abf19f2..9395f589952594 100644
--- a/offload/tools/offload-tblgen/APIGen.cpp
+++ b/offload/tools/offload-tblgen/APIGen.cpp
@@ -32,7 +32,7 @@ static std::string MakeComment(StringRef in) {
if (LineBreak - LineStart <= 1) {
break;
}
- out += std::string("\t///< ") +
+ out += std::string("/// ") +
in.substr(LineStart, LineBreak - LineStart).str() + "\n";
LineStart = LineBreak + 1;
}
@@ -107,13 +107,13 @@ static void ProcessFunction(const FunctionRec &F, raw_ostream &OS) {
OS << "(\n";
auto Params = F.getParams();
for (auto &Param : Params) {
+ OS << MakeParamComment(Param) << "\n";
OS << " " << Param.getType() << " " << Param.getName();
if (Param != Params.back()) {
- OS << ", ";
+ OS << ",\n";
} else {
- OS << " ";
+ OS << "\n";
}
- OS << MakeParamComment(Param) << "\n";
}
OS << ");\n\n";
}
@@ -125,9 +125,9 @@ static void ProcessEnum(const EnumRec &Enum, raw_ostream &OS) {
uint32_t EtorVal = 0;
for (const auto &EnumVal : Enum.getValues()) {
- auto Desc = MakeComment(EnumVal.getDesc());
- OS << formatv(TAB_1 "{0}_{1} = {2}, {3}", Enum.getEnumValNamePrefix(),
- EnumVal.getName(), EtorVal++, Desc);
+ OS << TAB_1 << MakeComment(EnumVal.getDesc());
+ OS << formatv(TAB_1 "{0}_{1} = {2},\n", Enum.getEnumValNamePrefix(),
+ EnumVal.getName(), EtorVal++);
}
// Add force uint32 val
diff --git a/offload/tools/offload-tblgen/GenCommon.hpp b/offload/tools/offload-tblgen/GenCommon.hpp
index 829f9b525fea92..64df6be8442207 100644
--- a/offload/tools/offload-tblgen/GenCommon.hpp
+++ b/offload/tools/offload-tblgen/GenCommon.hpp
@@ -61,7 +61,7 @@ constexpr auto PrefixUpper = "OFFLOAD";
inline std::string
MakeParamComment(const llvm::offload::tblgen::ParamRec &Param) {
- return llvm::formatv("///< {0}{1}{2} {3}", (Param.isIn() ? "[in]" : ""),
+ return llvm::formatv("// {0}{1}{2} {3}", (Param.isIn() ? "[in]" : ""),
(Param.isOut() ? "[out]" : ""),
(Param.isOpt() ? "[optional]" : ""), Param.getDesc());
}
>From 147e39ee62b881f6abb999dd885f2dc8549ee782 Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Fri, 4 Oct 2024 16:16:01 +0100
Subject: [PATCH 10/12] Rework Offload API errors
---
offload/new-api/API/APIDefs.td | 12 +-
offload/new-api/API/Common.td | 66 ++++----
offload/new-api/API/Device.td | 22 +--
offload/new-api/API/Platform.td | 18 +--
offload/new-api/include/offload_api.h | 147 ++++++++----------
.../new-api/include/offload_entry_points.inc | 102 +++++-------
offload/new-api/include/offload_exports | 1 -
offload/new-api/include/offload_funcs.inc | 1 -
offload/new-api/include/offload_impl.hpp | 71 ++++++++-
.../include/offload_impl_func_decls.inc | 3 -
offload/new-api/include/offload_print.hpp | 91 ++++++-----
offload/new-api/src/helpers.hpp | 28 ++--
offload/new-api/src/offload_impl.cpp | 37 ++---
offload/new-api/src/offload_lib.cpp | 14 +-
.../tools/offload-tblgen/EntryPointGen.cpp | 6 +-
offload/tools/offload-tblgen/PrintGen.cpp | 17 +-
.../OffloadAPI/common/environment.cpp | 2 +-
.../unittests/OffloadAPI/common/fixtures.hpp | 11 +-
.../device/offloadDeviceGetInfo.cpp | 41 ++---
.../platform/offloadPlatformGet.cpp | 5 +-
.../platform/offloadPlatformGetInfo.cpp | 37 ++---
21 files changed, 380 insertions(+), 352 deletions(-)
diff --git a/offload/new-api/API/APIDefs.td b/offload/new-api/API/APIDefs.td
index 410a28c4c90cfe..a74332363369ac 100644
--- a/offload/new-api/API/APIDefs.td
+++ b/offload/new-api/API/APIDefs.td
@@ -106,11 +106,11 @@ class AddHandleChecksToReturns<list<Param> Params, list<Return> Returns> {
// 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#"_RESULT_ERROR_INVALID_NULL_HANDLE")));
+ !or(HasErr, !eq(Ret.value, PREFIX#"_ERRC_INVALID_NULL_HANDLE")));
list<Return> returns_out = !if(returns_has_inv_handle,
- AppendConditionsToReturn<Returns, PREFIX # "_RESULT_ERROR_INVALID_NULL_HANDLE", handle_param_conds>.ret,
- !listconcat(Returns, [Return<PREFIX # "_RESULT_ERROR_INVALID_NULL_HANDLE", handle_param_conds>])
+ AppendConditionsToReturn<Returns, PREFIX # "_ERRC_INVALID_NULL_HANDLE", handle_param_conds>.ret,
+ !listconcat(Returns, [Return<PREFIX # "_ERRC_INVALID_NULL_HANDLE", handle_param_conds>])
);
}
@@ -125,10 +125,10 @@ class AddPointerChecksToReturns<list<Param> Params, list<Return> Returns> {
// 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#"_RESULT_ERROR_INVALID_NULL_POINTER")));
+ !or(HasErr, !eq(Ret.value, PREFIX#"_ERROR_CODE_INVALID_NULL_POINTER")));
list<Return> returns_out = !if(returns_has_inv_ptr,
- AppendConditionsToReturn<Returns, PREFIX # "_RESULT_ERROR_INVALID_NULL_POINTER", ptr_param_conds>.ret,
- !listconcat(Returns, [Return<PREFIX # "_RESULT_ERROR_INVALID_NULL_POINTER", ptr_param_conds>])
+ AppendConditionsToReturn<Returns, PREFIX # "_ERROR_CODE_INVALID_NULL_POINTER", ptr_param_conds>.ret,
+ !listconcat(Returns, [Return<PREFIX # "_ERROR_CODE_INVALID_NULL_POINTER", ptr_param_conds>])
);
}
diff --git a/offload/new-api/API/Common.td b/offload/new-api/API/Common.td
index 8de5475af99371..f2fe4b672fbfe5 100644
--- a/offload/new-api/API/Common.td
+++ b/offload/new-api/API/Common.td
@@ -45,41 +45,47 @@ def : Handle {
}
def : Enum {
- let name = "offload_result_t";
+ let name = "offload_errc_t";
let desc = "Defines Return/Error codes";
let etors =[
Etor<"SUCCESS", "Success">,
- Etor<"ERROR_INVALID_VALUE", "Invalid Value">,
- Etor<"ERROR_INVALID_PLATFORM", "Invalid platform">,
- Etor<"ERROR_DEVICE_NOT_FOUND", "Device not found">,
- Etor<"ERROR_INVALID_DEVICE", "Invalid device">,
- Etor<"ERROR_DEVICE_LOST", "Device hung, reset, was removed, or driver update occurred">,
- Etor<"ERROR_UNINITIALIZED", "plugin is not initialized or specific entry-point is not implemented">,
- Etor<"ERROR_OUT_OF_RESOURCES", "Out of resources">,
- Etor<"ERROR_UNSUPPORTED_VERSION", "generic error code for unsupported versions">,
- Etor<"ERROR_UNSUPPORTED_FEATURE", "generic error code for unsupported features">,
- Etor<"ERROR_INVALID_ARGUMENT", "generic error code for invalid arguments">,
- Etor<"ERROR_INVALID_NULL_HANDLE", "handle argument is not valid">,
- Etor<"ERROR_INVALID_NULL_POINTER", "pointer argument may not be nullptr">,
- Etor<"ERROR_INVALID_SIZE", "invalid size or dimensions (e.g., must not be zero, or is out of bounds)">,
- Etor<"ERROR_INVALID_ENUMERATION", "enumerator argument is not valid">,
- Etor<"ERROR_UNSUPPORTED_ENUMERATION", "enumerator argument is not supported by the device">,
- Etor<"ERROR_UNKNOWN", "Unknown or internal error">
+ Etor<"INVALID_VALUE", "Invalid Value">,
+ Etor<"INVALID_PLATFORM", "Invalid platform">,
+ Etor<"DEVICE_NOT_FOUND", "Device not found">,
+ Etor<"INVALID_DEVICE", "Invalid device">,
+ Etor<"DEVICE_LOST", "Device hung, reset, was removed, or driver update occurred">,
+ Etor<"UNINITIALIZED", "plugin is not initialized or specific entry-point is not implemented">,
+ Etor<"OUT_OF_RESOURCES", "Out of resources">,
+ Etor<"UNSUPPORTED_VERSION", "generic error code for unsupported versions">,
+ Etor<"UNSUPPORTED_FEATURE", "generic error code for unsupported features">,
+ Etor<"INVALID_ARGUMENT", "generic error code for invalid arguments">,
+ Etor<"INVALID_NULL_HANDLE", "handle argument is not valid">,
+ Etor<"INVALID_NULL_POINTER", "pointer argument may not be nullptr">,
+ Etor<"INVALID_SIZE", "invalid size or dimensions (e.g., must not be zero, or is out of bounds)">,
+ Etor<"INVALID_ENUMERATION", "enumerator argument is not valid">,
+ Etor<"UNSUPPORTED_ENUMERATION", "enumerator argument is not supported by the device">,
+ Etor<"UNKNOWN", "Unknown or internal error">
];
}
-def : Function {
- let name = "offloadGetErrorDetails";
- let desc = "Get a detailed error message for the last error that occurred on this thread, if it exists";
- let details = [
- "When an Offload API call returns a return value other than OFFLOAD_RESULT_SUCCESS, the implementation *may* set an additional error message.",
- "If a further Offload call (excluding this function) is made on the same thread without checking "
- "its detailed error message with this function, that message should be considered lost.",
- "The returned char* is only valid until the next Offload function call on the same thread (excluding further calls to this function.)"
- ];
- let params = [
- Param<"size_t*", "SizeRet", "Pointer to return the size of the available error message. A size of 0 indicates no message.", PARAM_OUT_OPTIONAL>,
- Param<"const char**", "DetailStringRet", "Pointer to return the error message string.", PARAM_OUT_OPTIONAL>
+def : Struct {
+ let name = "offload_error_struct_t";
+ let desc = "Details of the error condition returned by an API call";
+ let members = [
+ StructMember<"offload_errc_t", "code", "The error code">,
+ StructMember<"const char*", "details", "String containing error details">
];
- let returns = []; // Only SUCCESS is expected
+}
+
+def : Typedef {
+ let name = "offload_result_t";
+ let desc = "Result type returned by all entry points.";
+ let value = "const offload_error_struct_t*";
+}
+
+def : Macro {
+ let name = "OFFLOAD_SUCCESS";
+ let condition = "!defined(OFFLOAD_SUCCESS)";
+ let desc = "Success condition";
+ let value = "NULL";
}
diff --git a/offload/new-api/API/Device.td b/offload/new-api/API/Device.td
index 837d7071853f5c..da1141cf1986e0 100644
--- a/offload/new-api/API/Device.td
+++ b/offload/new-api/API/Device.td
@@ -47,7 +47,7 @@ def : Function {
Param<"offload_platform_handle_t", "hPlatform", "handle of the platform instance", PARAM_IN>,
Param<"offload_device_type_t", "DeviceType", "the type of the devices.", PARAM_IN>,
Param<"uint32_t", "NumEntries", "the number of devices to be added to phDevices. "
- "If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE "
+ "If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_ERRC_INVALID_SIZE "
"will be returned.", PARAM_IN>,
RangedParam<"offload_device_handle_t*", "phDevices", "Array of device handles. "
"If NumEntries is less than the number of devices available, then platform shall only retrieve that number of devices.", PARAM_OUT_OPTIONAL,
@@ -56,13 +56,13 @@ def : Function {
"pNumDevices will be updated with the total number of devices available.", PARAM_OUT_OPTIONAL>
];
let returns = [
- Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ Return<"OFFLOAD_ERRC_INVALID_SIZE", [
"`NumEntries == 0 && phDevices != NULL`"
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER", [
+ Return<"OFFLOAD_ERRC_INVALID_NULL_POINTER", [
"`NumEntries > 0 && phDevices == NULL`"
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_VALUE">
+ Return<"OFFLOAD_ERRC_INVALID_VALUE">
];
}
@@ -78,24 +78,24 @@ def : Function {
Param<"offload_device_info_t", "propName", "type of the info to retrieve", PARAM_IN>,
Param<"size_t", "propSize", "the number of bytes pointed to by pPropValue.", PARAM_IN>,
TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info. If propSize is not equal to or greater than the real "
- "number of bytes needed to return the info then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and "
+ "number of bytes needed to return the info then the OFFLOAD_ERRC_INVALID_SIZE error is returned and "
"pPropValue is not used.", PARAM_OUT_OPTIONAL, TypeInfo<"propName" , "propSize">>,
Param<"size_t*", "pPropSizeRet", "pointer to the actual size in bytes of the queried propName.", PARAM_OUT_OPTIONAL>
];
let returns = [
- Return<"OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION", [
+ Return<"OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION", [
"If `propName` is not supported by the adapter."
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ Return<"OFFLOAD_ERRC_INVALID_SIZE", [
"`propSize == 0 && pPropValue != NULL`",
"If `propSize` is less than the real number of bytes needed to return the info."
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER", [
+ Return<"OFFLOAD_ERRC_INVALID_NULL_POINTER", [
"`propSize != 0 && pPropValue == NULL`",
"`pPropValue == NULL && pPropSizeRet == NULL`"
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_DEVICE">,
- Return<"OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES">,
- Return<"OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY">
+ Return<"OFFLOAD_ERRC_INVALID_DEVICE">,
+ Return<"OFFLOAD_ERRC_OUT_OF_RESOURCES">,
+ Return<"OFFLOAD_ERRC_OUT_OF_HOST_MEMORY">
];
}
diff --git a/offload/new-api/API/Platform.td b/offload/new-api/API/Platform.td
index d280da1156c798..0bfcbdce8d2d5c 100644
--- a/offload/new-api/API/Platform.td
+++ b/offload/new-api/API/Platform.td
@@ -18,7 +18,7 @@ def : Function {
let params = [
Param<"uint32_t", "NumEntries",
"The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, then "
- "NumEntries should be greater than zero, otherwise OFFLOAD_RESULT_ERROR_INVALID_SIZE "
+ "NumEntries should be greater than zero, otherwise OFFLOAD_ERRC_INVALID_SIZE "
"will be returned.", PARAM_IN>,
RangedParam<"offload_platform_handle_t*", "phPlatforms",
"Array of handle of platforms. If NumEntries is less than the number of "
@@ -30,7 +30,7 @@ def : Function {
PARAM_OUT_OPTIONAL>
];
let returns = [
- Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ Return<"OFFLOAD_ERRC_INVALID_SIZE", [
"`NumEntries == 0 && phPlatforms != NULL`"
]>
];
@@ -71,24 +71,24 @@ def : Function {
Param<"size_t", "propSize", "the number of bytes pointed to by pPlatformInfo.", PARAM_IN>,
TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info. "
"If Size is not equal to or greater to the real number of bytes needed to return the info "
- "then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is not used.", PARAM_OUT_OPTIONAL,
+ "then the OFFLOAD_ERRC_INVALID_SIZE error is returned and pPlatformInfo is not used.", PARAM_OUT_OPTIONAL,
TypeInfo<"propName" , "propSize">>,
Param<"size_t*", "pPropSizeRet", "pointer to the actual number of bytes being queried by pPlatformInfo.", PARAM_OUT_OPTIONAL>
];
let returns = [
- Return<"OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION", [
+ Return<"OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION", [
"If `propName` is not supported by the platform."
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_SIZE", [
+ Return<"OFFLOAD_ERRC_INVALID_SIZE", [
"`propSize == 0 && pPropValue != NULL`",
"If `propSize` is less than the real number of bytes needed to return the info."
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER", [
+ Return<"OFFLOAD_ERRC_INVALID_NULL_POINTER", [
"`propSize != 0 && pPropValue == NULL`",
"`pPropValue == NULL && pPropSizeRet == NULL`"
]>,
- Return<"OFFLOAD_RESULT_ERROR_INVALID_PLATFORM">,
- Return<"OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES">,
- Return<"OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY">
+ Return<"OFFLOAD_ERRC_INVALID_PLATFORM">,
+ Return<"OFFLOAD_ERRC_OUT_OF_RESOURCES">,
+ Return<"OFFLOAD_ERRC_OUT_OF_HOST_MEMORY">
];
}
diff --git a/offload/new-api/include/offload_api.h b/offload/new-api/include/offload_api.h
index 8e70efa27774d1..a4d72b9a36a7c7 100644
--- a/offload/new-api/include/offload_api.h
+++ b/offload/new-api/include/offload_api.h
@@ -69,73 +69,65 @@ typedef struct offload_context_handle_t_ *offload_context_handle_t;
///////////////////////////////////////////////////////////////////////////////
/// @brief Defines Return/Error codes
-typedef enum offload_result_t {
+typedef enum offload_errc_t {
/// Success
- OFFLOAD_RESULT_SUCCESS = 0,
+ OFFLOAD_ERRC_SUCCESS = 0,
/// Invalid Value
- OFFLOAD_RESULT_ERROR_INVALID_VALUE = 1,
+ OFFLOAD_ERRC_INVALID_VALUE = 1,
/// Invalid platform
- OFFLOAD_RESULT_ERROR_INVALID_PLATFORM = 2,
+ OFFLOAD_ERRC_INVALID_PLATFORM = 2,
/// Device not found
- OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND = 3,
+ OFFLOAD_ERRC_DEVICE_NOT_FOUND = 3,
/// Invalid device
- OFFLOAD_RESULT_ERROR_INVALID_DEVICE = 4,
+ OFFLOAD_ERRC_INVALID_DEVICE = 4,
/// Device hung, reset, was removed, or driver update occurred
- OFFLOAD_RESULT_ERROR_DEVICE_LOST = 5,
+ OFFLOAD_ERRC_DEVICE_LOST = 5,
/// plugin is not initialized or specific entry-point is not implemented
- OFFLOAD_RESULT_ERROR_UNINITIALIZED = 6,
+ OFFLOAD_ERRC_UNINITIALIZED = 6,
/// Out of resources
- OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES = 7,
+ OFFLOAD_ERRC_OUT_OF_RESOURCES = 7,
/// generic error code for unsupported versions
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION = 8,
+ OFFLOAD_ERRC_UNSUPPORTED_VERSION = 8,
/// generic error code for unsupported features
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE = 9,
+ OFFLOAD_ERRC_UNSUPPORTED_FEATURE = 9,
/// generic error code for invalid arguments
- OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT = 10,
+ OFFLOAD_ERRC_INVALID_ARGUMENT = 10,
/// handle argument is not valid
- OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE = 11,
+ OFFLOAD_ERRC_INVALID_NULL_HANDLE = 11,
/// pointer argument may not be nullptr
- OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER = 12,
+ OFFLOAD_ERRC_INVALID_NULL_POINTER = 12,
/// invalid size or dimensions (e.g., must not be zero, or is out of bounds)
- OFFLOAD_RESULT_ERROR_INVALID_SIZE = 13,
+ OFFLOAD_ERRC_INVALID_SIZE = 13,
/// enumerator argument is not valid
- OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION = 14,
+ OFFLOAD_ERRC_INVALID_ENUMERATION = 14,
/// enumerator argument is not supported by the device
- OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION = 15,
+ OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION = 15,
/// Unknown or internal error
- OFFLOAD_RESULT_ERROR_UNKNOWN = 16,
+ OFFLOAD_ERRC_UNKNOWN = 16,
/// @cond
- OFFLOAD_RESULT_FORCE_UINT32 = 0x7fffffff
+ OFFLOAD_ERRC_FORCE_UINT32 = 0x7fffffff
/// @endcond
-} offload_result_t;
+} offload_errc_t;
///////////////////////////////////////////////////////////////////////////////
-/// @brief Get a detailed error message for the last error that occurred on this
-/// thread, if it exists
-///
-/// @details
-/// - When an Offload API call returns a return value other than
-/// OFFLOAD_RESULT_SUCCESS, the implementation *may* set an additional error
-/// message.
-/// - If a further Offload call (excluding this function) is made on the same
-/// thread without checking its detailed error message with this function,
-/// that message should be considered lost.
-/// - The returned char* is only valid until the next Offload function call
-/// on the same thread (excluding further calls to this function.)
-///
-/// @returns
-/// - ::OFFLOAD_RESULT_SUCCESS
-/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
-/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
- // [out][optional] Pointer to return the size of the available error
- // message. A size of 0 indicates no message.
- size_t *SizeRet,
- // [out][optional] Pointer to return the error message string.
- const char **DetailStringRet);
+/// @brief Details of the error condition returned by an API call
+typedef struct offload_error_struct_t {
+ offload_errc_t code; /// The error code
+ const char *details; /// String containing error details
+} offload_error_struct_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Result type returned by all entry points.
+typedef const offload_error_struct_t *offload_result_t;
+
+///////////////////////////////////////////////////////////////////////////////
+#ifndef OFFLOAD_SUCCESS
+#if !defined(OFFLOAD_SUCCESS)
+/// @brief Success condition
+#define OFFLOAD_SUCCESS NULL
+#endif // !defined(OFFLOAD_SUCCESS)
+#endif // OFFLOAD_SUCCESS
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves all available platforms
@@ -148,14 +140,14 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadGetErrorDetails(
/// - ::OFFLOAD_RESULT_SUCCESS
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// - ::OFFLOAD_ERRC_INVALID_SIZE
/// + `NumEntries == 0 && phPlatforms != NULL`
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
// [in] The number of platforms to be added to phPlatforms. If phPlatforms
// is not NULL, then NumEntries should be greater than zero, otherwise
- // OFFLOAD_RESULT_ERROR_INVALID_SIZE will be returned.
+ // OFFLOAD_ERRC_INVALID_SIZE will be returned.
uint32_t NumEntries,
// [out][optional] Array of handle of platforms. If NumEntries is less than
// the number of platforms available, then offloadPlatformGet shall only
@@ -211,20 +203,21 @@ typedef enum offload_platform_backend_t {
/// - ::OFFLOAD_RESULT_SUCCESS
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
-/// - ::OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION
+/// - ::OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION
/// + If `propName` is not supported by the platform.
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// - ::OFFLOAD_ERRC_INVALID_SIZE
/// + `propSize == 0 && pPropValue != NULL`
/// + If `propSize` is less than the real number of bytes needed to
/// return the info.
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
/// + `propSize != 0 && pPropValue == NULL`
/// + `pPropValue == NULL && pPropSizeRet == NULL`
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_PLATFORM
-/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES
-/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERRC_INVALID_PLATFORM
+/// - ::OFFLOAD_ERRC_OUT_OF_RESOURCES
+/// - ::OFFLOAD_ERRC_OUT_OF_HOST_MEMORY
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
+/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
// [in] handle of the platform
offload_platform_handle_t hPlatform,
@@ -234,8 +227,8 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
size_t propSize,
// [out][optional] array of bytes holding the info. If Size is not equal to
// or greater to the real number of bytes needed to return the info then the
- // OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and pPlatformInfo is
- // not used.
+ // OFFLOAD_ERRC_INVALID_SIZE error is returned and pPlatformInfo is not
+ // used.
void *pPropValue,
// [out][optional] pointer to the actual number of bytes being queried by
// pPlatformInfo.
@@ -295,13 +288,14 @@ typedef enum offload_device_info_t {
/// - ::OFFLOAD_RESULT_SUCCESS
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// - ::OFFLOAD_ERRC_INVALID_SIZE
/// + `NumEntries == 0 && phDevices != NULL`
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
/// + `NumEntries > 0 && phDevices == NULL`
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_VALUE
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERRC_INVALID_VALUE
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
+/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
// [in] handle of the platform instance
offload_platform_handle_t hPlatform,
@@ -309,7 +303,7 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
offload_device_type_t DeviceType,
// [in] the number of devices to be added to phDevices. If phDevices is not
// NULL, then NumEntries should be greater than zero. Otherwise
- // OFFLOAD_RESULT_ERROR_INVALID_SIZE will be returned.
+ // OFFLOAD_ERRC_INVALID_SIZE will be returned.
uint32_t NumEntries,
// [out][optional] Array of device handles. If NumEntries is less than the
// number of devices available, then platform shall only retrieve that
@@ -330,20 +324,21 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
/// - ::OFFLOAD_RESULT_SUCCESS
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
-/// - ::OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION
+/// - ::OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION
/// + If `propName` is not supported by the adapter.
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_SIZE
+/// - ::OFFLOAD_ERRC_INVALID_SIZE
/// + `propSize == 0 && pPropValue != NULL`
/// + If `propSize` is less than the real number of bytes needed to
/// return the info.
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
/// + `propSize != 0 && pPropValue == NULL`
/// + `pPropValue == NULL && pPropSizeRet == NULL`
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_DEVICE
-/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES
-/// - ::OFFLOAD_RESULT_ERROR_OUT_OF_HOST_MEMORY
-/// - ::OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERRC_INVALID_DEVICE
+/// - ::OFFLOAD_ERRC_OUT_OF_RESOURCES
+/// - ::OFFLOAD_ERRC_OUT_OF_HOST_MEMORY
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
/// + `NULL == hDevice`
+/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
// [in] handle of the device instance
offload_device_handle_t hDevice,
@@ -353,21 +348,13 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
size_t propSize,
// [out][optional] array of bytes holding the info. If propSize is not equal
// to or greater than the real number of bytes needed to return the info
- // then the OFFLOAD_RESULT_ERROR_INVALID_SIZE error is returned and
- // pPropValue is not used.
+ // then the OFFLOAD_ERRC_INVALID_SIZE error is returned and pPropValue is
+ // not used.
void *pPropValue,
// [out][optional] pointer to the actual size in bytes of the queried
// propName.
size_t *pPropSizeRet);
-///////////////////////////////////////////////////////////////////////////////
-/// @brief Function parameters for offloadGetErrorDetails
-/// @details Each entry is a pointer to the parameter passed to the function;
-typedef struct offload_get_error_details_params_t {
- size_t **pSizeRet;
- const char ***pDetailStringRet;
-} offload_get_error_details_params_t;
-
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadPlatformGet
/// @details Each entry is a pointer to the parameter passed to the function;
diff --git a/offload/new-api/include/offload_entry_points.inc b/offload/new-api/include/offload_entry_points.inc
index 2910bc414071fc..0bdfabe7fefa9c 100644
--- a/offload/new-api/include/offload_entry_points.inc
+++ b/offload/new-api/include/offload_entry_points.inc
@@ -7,40 +7,13 @@
//===----------------------------------------------------------------------===//
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadGetErrorDetails_val(size_t *SizeRet,
- const char **DetailStringRet) {
- if (true /*enableParameterValidation*/) {
- }
-
- return offloadGetErrorDetails_impl(SizeRet, DetailStringRet);
-}
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
-offloadGetErrorDetails(size_t *SizeRet, const char **DetailStringRet) {
- if (std::getenv("OFFLOAD_TRACE")) {
- std::cout << "---> offloadGetErrorDetails";
- }
-
- offload_result_t result =
- offloadGetErrorDetails_val(SizeRet, DetailStringRet);
-
- if (std::getenv("OFFLOAD_TRACE")) {
- offload_get_error_details_params_t Params = {&SizeRet, &DetailStringRet};
- std::cout << "(" << &Params << ")";
- std::cout << "-> " << result << "\n";
- if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
- std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
- }
- }
- return result;
-}
-
-///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadPlatformGet_val(uint32_t NumEntries,
- offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms) {
+offload_impl_result_t
+offloadPlatformGet_val(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms) {
if (true /*enableParameterValidation*/) {
if (NumEntries == 0 && phPlatforms != NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ return OFFLOAD_ERRC_INVALID_SIZE;
}
}
@@ -61,33 +34,33 @@ offloadPlatformGet(uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
&pNumPlatforms};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
- if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
- std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
}
}
return result;
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadPlatformGetInfo_val(offload_platform_handle_t hPlatform,
- offload_platform_info_t propName,
- size_t propSize, void *pPropValue,
- size_t *pPropSizeRet) {
+offload_impl_result_t
+offloadPlatformGetInfo_val(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName, size_t propSize,
+ void *pPropValue, size_t *pPropSizeRet) {
if (true /*enableParameterValidation*/) {
if (propSize == 0 && pPropValue != NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ return OFFLOAD_ERRC_INVALID_SIZE;
}
if (propSize != 0 && pPropValue == NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
if (pPropValue == NULL && pPropSizeRet == NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
if (NULL == hPlatform) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
}
@@ -109,30 +82,30 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
&hPlatform, &propName, &propSize, &pPropValue, &pPropSizeRet};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
- if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
- std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
}
}
return result;
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadDeviceGet_val(offload_platform_handle_t hPlatform,
- offload_device_type_t DeviceType,
- uint32_t NumEntries,
- offload_device_handle_t *phDevices,
- uint32_t *pNumDevices) {
+offload_impl_result_t offloadDeviceGet_val(offload_platform_handle_t hPlatform,
+ offload_device_type_t DeviceType,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices) {
if (true /*enableParameterValidation*/) {
if (NumEntries == 0 && phDevices != NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ return OFFLOAD_ERRC_INVALID_SIZE;
}
if (NumEntries > 0 && phDevices == NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
if (NULL == hPlatform) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
}
@@ -155,33 +128,34 @@ offloadDeviceGet(offload_platform_handle_t hPlatform,
&phDevices, &pNumDevices};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
- if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
- std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
}
}
return result;
}
///////////////////////////////////////////////////////////////////////////////
-offload_result_t offloadDeviceGetInfo_val(offload_device_handle_t hDevice,
- offload_device_info_t propName,
- size_t propSize, void *pPropValue,
- size_t *pPropSizeRet) {
+offload_impl_result_t offloadDeviceGetInfo_val(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t propSize,
+ void *pPropValue,
+ size_t *pPropSizeRet) {
if (true /*enableParameterValidation*/) {
if (propSize == 0 && pPropValue != NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ return OFFLOAD_ERRC_INVALID_SIZE;
}
if (propSize != 0 && pPropValue == NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
if (pPropValue == NULL && pPropSizeRet == NULL) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
if (NULL == hDevice) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE;
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
}
@@ -203,8 +177,8 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
&pPropValue, &pPropSizeRet};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
- if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {
- std::cout << " *Error Details* " << *LastErrorDetails() << " \n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
}
}
return result;
diff --git a/offload/new-api/include/offload_exports b/offload/new-api/include/offload_exports
index fb8103ae5e80ef..cb256fd34770d5 100644
--- a/offload/new-api/include/offload_exports
+++ b/offload/new-api/include/offload_exports
@@ -1,6 +1,5 @@
VERS1.0 {
global:
- offloadGetErrorDetails;
offloadPlatformGet;
offloadPlatformGetInfo;
offloadDeviceGet;
diff --git a/offload/new-api/include/offload_funcs.inc b/offload/new-api/include/offload_funcs.inc
index 7160b27a5332cf..14e878b9ed5a27 100644
--- a/offload/new-api/include/offload_funcs.inc
+++ b/offload/new-api/include/offload_funcs.inc
@@ -10,7 +10,6 @@
#error Please define the macro OFFLOAD_FUNC(Function)
#endif
-OFFLOAD_FUNC(offloadGetErrorDetails)
OFFLOAD_FUNC(offloadPlatformGet)
OFFLOAD_FUNC(offloadPlatformGetInfo)
OFFLOAD_FUNC(offloadDeviceGet)
diff --git a/offload/new-api/include/offload_impl.hpp b/offload/new-api/include/offload_impl.hpp
index 1e96fdebfc765e..b0bdf6c2221ae4 100644
--- a/offload/new-api/include/offload_impl.hpp
+++ b/offload/new-api/include/offload_impl.hpp
@@ -5,25 +5,80 @@
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
+#pragma once
+#include <iostream>
+#include <memory>
#include <offload_api.h>
#include <optional>
+#include <set>
#include <string>
+#include <unordered_set>
+#include <vector>
+#include "llvm/ADT/DenseSet.h"
#include "llvm/ADT/StringRef.h"
+#include "llvm/ADT/StringSet.h"
-std::optional<std::string> &LastErrorDetails();
+// Use the StringSet container to efficiently deduplicate repeated error
+// strings (e.g. if the same error is hit constantly in a long running program)
+llvm::StringSet<> &ErrorStrs();
+
+// Use an unordered_set to avoid duplicates of error structs themselves.
+// We cannot store the structs directly as returned pointers to them must always
+// be valid, and a rehash of the set may invalidate them. This requires
+// custom hash and equal_to function objects.
+using ErrPtrT = std::unique_ptr<offload_error_struct_t>;
+struct ErrPtrEqual {
+ bool operator()(const ErrPtrT &lhs, const ErrPtrT &rhs) const {
+ if (!lhs && !rhs) {
+ return true;
+ }
+ if (!lhs || !rhs) {
+ return false;
+ }
+
+ bool StrsEqual = false;
+ if (lhs->details == NULL && rhs->details == NULL) {
+ StrsEqual = true;
+ } else if (lhs->details != NULL && rhs->details != NULL) {
+ StrsEqual = (std::strcmp(lhs->details, rhs->details) == 0);
+ }
+ return (lhs->code == rhs->code) && StrsEqual;
+ }
+};
+struct ErrPtrHash {
+ size_t operator()(const ErrPtrT &e) const {
+ if (!e) {
+ // We shouldn't store empty errors (i.e. success), but just in case
+ return 0lu;
+ } else {
+ return std::hash<int>{}(e->code);
+ }
+ }
+};
+using ErrSetT = std::unordered_set<ErrPtrT, ErrPtrHash, ErrPtrEqual>;
+ErrSetT &Errors();
struct offload_impl_result_t {
- offload_impl_result_t() = delete;
- offload_impl_result_t(offload_result_t Result) : Result(Result) {
- LastErrorDetails() = std::nullopt;
+ offload_impl_result_t(std::nullptr_t) : Result(OFFLOAD_SUCCESS) {}
+ offload_impl_result_t(offload_errc_t Code) {
+ if (Code == OFFLOAD_ERRC_SUCCESS) {
+ Result = nullptr;
+ } else {
+ auto Err = std::unique_ptr<offload_error_struct_t>(
+ new offload_error_struct_t{Code, nullptr});
+ Result = Errors().emplace(std::move(Err)).first->get();
+ }
}
- offload_impl_result_t(offload_result_t Result, std::string Details)
- : Result(Result) {
- assert(Result != OFFLOAD_RESULT_SUCCESS);
- LastErrorDetails() = Details;
+ offload_impl_result_t(offload_errc_t Code, llvm::StringRef Details) {
+ assert(Code != OFFLOAD_ERRC_SUCCESS);
+ Result = nullptr;
+ auto DetailsStr = ErrorStrs().insert(Details).first->getKeyData();
+ auto Err = std::unique_ptr<offload_error_struct_t>(
+ new offload_error_struct_t{Code, DetailsStr});
+ Result = Errors().emplace(std::move(Err)).first->get();
}
operator offload_result_t() { return Result; }
diff --git a/offload/new-api/include/offload_impl_func_decls.inc b/offload/new-api/include/offload_impl_func_decls.inc
index df40c342049549..1160ee1c6e2549 100644
--- a/offload/new-api/include/offload_impl_func_decls.inc
+++ b/offload/new-api/include/offload_impl_func_decls.inc
@@ -5,9 +5,6 @@
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
-offload_result_t offloadGetErrorDetails_impl(size_t *SizeRet,
- const char **DetailStringRet);
-
offload_impl_result_t
offloadPlatformGet_impl(uint32_t NumEntries,
offload_platform_handle_t *phPlatforms,
diff --git a/offload/new-api/include/offload_print.hpp b/offload/new-api/include/offload_print.hpp
index 4a0f6d14f10b4e..c3a6c0a702e5b7 100644
--- a/offload/new-api/include/offload_print.hpp
+++ b/offload/new-api/include/offload_print.hpp
@@ -24,7 +24,7 @@ template <> struct is_handle<offload_device_handle_t> : std::true_type {};
template <> struct is_handle<offload_context_handle_t> : std::true_type {};
template <typename T> inline constexpr bool is_handle_v = is_handle<T>::value;
-inline std::ostream &operator<<(std::ostream &os, enum offload_result_t value);
+inline std::ostream &operator<<(std::ostream &os, enum offload_errc_t value);
inline std::ostream &operator<<(std::ostream &os,
enum offload_platform_info_t value);
inline std::ostream &operator<<(std::ostream &os,
@@ -35,60 +35,60 @@ inline std::ostream &operator<<(std::ostream &os,
enum offload_device_info_t value);
///////////////////////////////////////////////////////////////////////////////
-/// @brief Print operator for the offload_result_t type
+/// @brief Print operator for the offload_errc_t type
/// @returns std::ostream &
-inline std::ostream &operator<<(std::ostream &os, enum offload_result_t value) {
+inline std::ostream &operator<<(std::ostream &os, enum offload_errc_t value) {
switch (value) {
- case OFFLOAD_RESULT_SUCCESS:
- os << "OFFLOAD_RESULT_SUCCESS";
+ case OFFLOAD_ERRC_SUCCESS:
+ os << "OFFLOAD_ERRC_SUCCESS";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_VALUE:
- os << "OFFLOAD_RESULT_ERROR_INVALID_VALUE";
+ case OFFLOAD_ERRC_INVALID_VALUE:
+ os << "OFFLOAD_ERRC_INVALID_VALUE";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_PLATFORM:
- os << "OFFLOAD_RESULT_ERROR_INVALID_PLATFORM";
+ case OFFLOAD_ERRC_INVALID_PLATFORM:
+ os << "OFFLOAD_ERRC_INVALID_PLATFORM";
break;
- case OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND:
- os << "OFFLOAD_RESULT_ERROR_DEVICE_NOT_FOUND";
+ case OFFLOAD_ERRC_DEVICE_NOT_FOUND:
+ os << "OFFLOAD_ERRC_DEVICE_NOT_FOUND";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_DEVICE:
- os << "OFFLOAD_RESULT_ERROR_INVALID_DEVICE";
+ case OFFLOAD_ERRC_INVALID_DEVICE:
+ os << "OFFLOAD_ERRC_INVALID_DEVICE";
break;
- case OFFLOAD_RESULT_ERROR_DEVICE_LOST:
- os << "OFFLOAD_RESULT_ERROR_DEVICE_LOST";
+ case OFFLOAD_ERRC_DEVICE_LOST:
+ os << "OFFLOAD_ERRC_DEVICE_LOST";
break;
- case OFFLOAD_RESULT_ERROR_UNINITIALIZED:
- os << "OFFLOAD_RESULT_ERROR_UNINITIALIZED";
+ case OFFLOAD_ERRC_UNINITIALIZED:
+ os << "OFFLOAD_ERRC_UNINITIALIZED";
break;
- case OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES:
- os << "OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES";
+ case OFFLOAD_ERRC_OUT_OF_RESOURCES:
+ os << "OFFLOAD_ERRC_OUT_OF_RESOURCES";
break;
- case OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION:
- os << "OFFLOAD_RESULT_ERROR_UNSUPPORTED_VERSION";
+ case OFFLOAD_ERRC_UNSUPPORTED_VERSION:
+ os << "OFFLOAD_ERRC_UNSUPPORTED_VERSION";
break;
- case OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE:
- os << "OFFLOAD_RESULT_ERROR_UNSUPPORTED_FEATURE";
+ case OFFLOAD_ERRC_UNSUPPORTED_FEATURE:
+ os << "OFFLOAD_ERRC_UNSUPPORTED_FEATURE";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT:
- os << "OFFLOAD_RESULT_ERROR_INVALID_ARGUMENT";
+ case OFFLOAD_ERRC_INVALID_ARGUMENT:
+ os << "OFFLOAD_ERRC_INVALID_ARGUMENT";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE:
- os << "OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE";
+ case OFFLOAD_ERRC_INVALID_NULL_HANDLE:
+ os << "OFFLOAD_ERRC_INVALID_NULL_HANDLE";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER:
- os << "OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER";
+ case OFFLOAD_ERRC_INVALID_NULL_POINTER:
+ os << "OFFLOAD_ERRC_INVALID_NULL_POINTER";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_SIZE:
- os << "OFFLOAD_RESULT_ERROR_INVALID_SIZE";
+ case OFFLOAD_ERRC_INVALID_SIZE:
+ os << "OFFLOAD_ERRC_INVALID_SIZE";
break;
- case OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION:
- os << "OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION";
+ case OFFLOAD_ERRC_INVALID_ENUMERATION:
+ os << "OFFLOAD_ERRC_INVALID_ENUMERATION";
break;
- case OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION:
- os << "OFFLOAD_RESULT_ERROR_UNSUPPORTED_ENUMERATION";
+ case OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION:
+ os << "OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION";
break;
- case OFFLOAD_RESULT_ERROR_UNKNOWN:
- os << "OFFLOAD_RESULT_ERROR_UNKNOWN";
+ case OFFLOAD_ERRC_UNKNOWN:
+ os << "OFFLOAD_ERRC_UNKNOWN";
break;
default:
os << "unknown enumerator";
@@ -280,14 +280,13 @@ inline void printTagged(std::ostream &os, const void *ptr,
}
}
-inline std::ostream &operator<<(
- std::ostream &os,
- [[maybe_unused]] const struct offload_get_error_details_params_t *params) {
- os << ".SizeRet = ";
- printPtr(os, *params->pSizeRet);
- os << ", ";
- os << ".DetailStringRet = ";
- printPtr(os, *params->pDetailStringRet);
+inline std::ostream &operator<<(std::ostream &os,
+ const offload_error_struct_t *err) {
+ if (err == nullptr) {
+ os << "OFFLOAD_SUCCESS";
+ } else {
+ os << err->code;
+ }
return os;
}
@@ -401,5 +400,5 @@ inline offload_result_t printPtr(std::ostream &os, const T *ptr) {
os << ")";
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_SUCCESS;
}
diff --git a/offload/new-api/src/helpers.hpp b/offload/new-api/src/helpers.hpp
index 246de2b678015d..e24bd58bc11111 100644
--- a/offload/new-api/src/helpers.hpp
+++ b/offload/new-api/src/helpers.hpp
@@ -17,16 +17,16 @@
#include <cstring>
template <typename T, typename Assign>
-offload_result_t getInfoImpl(size_t ParamValueSize, void *ParamValue,
- size_t *ParamValueSizeRet, T Value,
- size_t ValueSize, Assign &&AssignFunc) {
+offload_errc_t getInfoImpl(size_t ParamValueSize, void *ParamValue,
+ size_t *ParamValueSizeRet, T Value, size_t ValueSize,
+ Assign &&AssignFunc) {
if (!ParamValue && !ParamValueSizeRet) {
- return OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER;
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
if (ParamValue != nullptr) {
if (ParamValueSize < ValueSize) {
- return OFFLOAD_RESULT_ERROR_INVALID_SIZE;
+ return OFFLOAD_ERRC_INVALID_SIZE;
}
AssignFunc(ParamValue, Value, ValueSize);
}
@@ -35,12 +35,12 @@ offload_result_t getInfoImpl(size_t ParamValueSize, void *ParamValue,
*ParamValueSizeRet = ValueSize;
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_ERRC_SUCCESS;
}
template <typename T>
-offload_result_t getInfo(size_t ParamValueSize, void *ParamValue,
- size_t *ParamValueSizeRet, T Value) {
+offload_errc_t getInfo(size_t ParamValueSize, void *ParamValue,
+ size_t *ParamValueSizeRet, T Value) {
auto Assignment = [](void *ParamValue, T Value, size_t) {
*static_cast<T *>(ParamValue) = Value;
};
@@ -50,15 +50,15 @@ offload_result_t getInfo(size_t ParamValueSize, void *ParamValue,
}
template <typename T>
-offload_result_t getInfoArray(size_t array_length, size_t ParamValueSize,
- void *ParamValue, size_t *ParamValueSizeRet,
- const T *Value) {
+offload_errc_t getInfoArray(size_t array_length, size_t ParamValueSize,
+ void *ParamValue, size_t *ParamValueSizeRet,
+ const T *Value) {
return getInfoImpl(ParamValueSize, ParamValue, ParamValueSizeRet, Value,
array_length * sizeof(T), memcpy);
}
template <>
-inline offload_result_t
+inline offload_errc_t
getInfo<const char *>(size_t ParamValueSize, void *ParamValue,
size_t *ParamValueSizeRet, const char *Value) {
return getInfoArray(strlen(Value) + 1, ParamValueSize, ParamValue,
@@ -79,12 +79,12 @@ class ReturnHelper {
ParamValueSizeRet(ParamValueSize) {}
// Scalar return Value
- template <class T> offload_result_t operator()(const T &t) {
+ template <class T> offload_errc_t operator()(const T &t) {
return getInfo(ParamValueSize, ParamValue, ParamValueSizeRet, t);
}
// Array return Value
- template <class T> offload_result_t operator()(const T *t, size_t s) {
+ template <class T> offload_errc_t operator()(const T *t, size_t s) {
return getInfoArray(s, ParamValueSize, ParamValue, ParamValueSizeRet, t);
}
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
index 43112bba448045..31329d2aea683d 100644
--- a/offload/new-api/src/offload_impl.cpp
+++ b/offload/new-api/src/offload_impl.cpp
@@ -67,23 +67,6 @@ void initPlugins() {
}
}
-offload_result_t offloadGetErrorDetails_impl(size_t *SizeRet,
- const char **DetailStringRet) {
- if (auto Details = LastErrorDetails()) {
- if (SizeRet) {
- *SizeRet = Details->size();
- }
- if (DetailStringRet) {
- *DetailStringRet = Details->c_str();
- }
- } else {
- if (SizeRet) {
- *SizeRet = 0;
- }
- }
- return OFFLOAD_RESULT_SUCCESS;
-}
-
offload_impl_result_t
offloadPlatformGet_impl(uint32_t NumEntries,
offload_platform_handle_t *phPlatforms,
@@ -95,9 +78,9 @@ offloadPlatformGet_impl(uint32_t NumEntries,
std::call_once(InitFlag, initPlugins);
if (NumEntries > Platforms().size()) {
- return {OFFLOAD_RESULT_ERROR_INVALID_SIZE,
- formatv("{0} platform(s) available but {1} requested.",
- Platforms().size(), NumEntries)};
+ return {OFFLOAD_ERRC_INVALID_SIZE,
+ std::string{formatv("{0} platform(s) available but {1} requested.",
+ Platforms().size(), NumEntries)}};
}
if (phPlatforms) {
@@ -111,7 +94,7 @@ offloadPlatformGet_impl(uint32_t NumEntries,
*pNumPlatforms = Platforms().size();
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_SUCCESS;
}
offload_impl_result_t
@@ -141,10 +124,10 @@ offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
}
}
default:
- return OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION;
+ return OFFLOAD_ERRC_INVALID_ENUMERATION;
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_SUCCESS;
}
offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
@@ -163,7 +146,7 @@ offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
*pNumDevices = static_cast<uint32_t>(hPlatform->Devices.size());
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_SUCCESS;
}
offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
@@ -176,7 +159,7 @@ offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
InfoQueueTy DevInfo;
if (auto Err = hDevice->Device.obtainInfoImpl(DevInfo))
- return OFFLOAD_RESULT_ERROR_OUT_OF_RESOURCES;
+ return OFFLOAD_ERRC_OUT_OF_RESOURCES;
// Find the info if it exists under any of the given names
auto GetInfo = [&DevInfo](std::vector<std::string> Names) {
@@ -208,8 +191,8 @@ offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
return ReturnValue(
GetInfo({"CUDA Driver Version", "HSA Runtime Version"}).c_str());
default:
- return OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION;
+ return OFFLOAD_ERRC_INVALID_ENUMERATION;
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_SUCCESS;
}
diff --git a/offload/new-api/src/offload_lib.cpp b/offload/new-api/src/offload_lib.cpp
index a8a23e94a2fd6c..7ad0ff1d6e35c3 100644
--- a/offload/new-api/src/offload_lib.cpp
+++ b/offload/new-api/src/offload_lib.cpp
@@ -16,12 +16,14 @@
#include <iostream>
-// Store details for the last error that occurred on this thread. It MAY be set
-// when an implementation function returns a result other than
-// OFFLOAD_RESULT_SUCCESS.
-std::optional<std::string> &LastErrorDetails() {
- thread_local std::optional<std::string> Details;
- return Details;
+llvm::StringSet<> &ErrorStrs() {
+ static llvm::StringSet<> ErrorStrs;
+ return ErrorStrs;
+}
+
+ErrSetT &Errors() {
+ static ErrSetT Errors{};
+ return Errors;
}
// Pull in the declarations for the implementation funtions. The actual entry
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
index 9f81d9d5582393..e36252e482c426 100644
--- a/offload/tools/offload-tblgen/EntryPointGen.cpp
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -23,7 +23,7 @@ using namespace offload::tblgen;
static void EmitValidationFunc(const FunctionRec &F, raw_ostream &OS) {
OS << CommentsHeader;
// Emit preamble
- OS << formatv("{0}_result_t {1}_val(\n ", PrefixLower, F.getName());
+ OS << formatv("{0}_impl_result_t {1}_val(\n ", PrefixLower, F.getName());
// Emit arguments
std::string ParamNameList = "";
for (auto &Param : F.getParams()) {
@@ -92,8 +92,8 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
OS << formatv("};\n");
OS << TAB_2 "std::cout << \"(\" << &Params << \")\";\n";
OS << TAB_2 "std::cout << \"-> \" << result << \"\\n\";\n";
- OS << TAB_2 "if (result != OFFLOAD_RESULT_SUCCESS && LastErrorDetails()) {\n";
- OS << TAB_3 "std::cout << \" *Error Details* \" << *LastErrorDetails() "
+ OS << TAB_2 "if (result && result->details) {\n";
+ OS << TAB_3 "std::cout << \" *Error Details* \" << result->details "
"<< \" \\n\";\n";
OS << TAB_2 "}\n";
OS << TAB_1 "}\n";
diff --git a/offload/tools/offload-tblgen/PrintGen.cpp b/offload/tools/offload-tblgen/PrintGen.cpp
index b89d45388f0bf8..0e73c29c338a1b 100644
--- a/offload/tools/offload-tblgen/PrintGen.cpp
+++ b/offload/tools/offload-tblgen/PrintGen.cpp
@@ -94,6 +94,20 @@ inline void printTagged(std::ostream &os, const void *ptr, {0} value, size_t siz
OS << "}\n";
}
+static void EmitResultPrint(raw_ostream &OS) {
+ OS << R""(
+inline std::ostream &operator<<(std::ostream &os,
+ const offload_error_struct_t *err) {
+ if (err == nullptr) {
+ os << "OFFLOAD_SUCCESS";
+ } else {
+ os << err->code;
+ }
+ return os;
+}
+)"";
+}
+
static void EmitFunctionParamStructPrint(const FunctionRec &Func,
raw_ostream &OS) {
OS << formatv(R"(
@@ -173,6 +187,7 @@ template <typename T> inline void printTagged(std::ostream &os, const void *ptr,
EnumRec E{R};
ProcessEnum(E, OS);
}
+ EmitResultPrint(OS);
// Emit print functions for the function param structs
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
@@ -201,7 +216,7 @@ template <typename T> inline offload_result_t printPtr(std::ostream &os, const T
os << ")";
}
- return OFFLOAD_RESULT_SUCCESS;
+ return OFFLOAD_SUCCESS;
}
)""";
}
diff --git a/offload/unittests/OffloadAPI/common/environment.cpp b/offload/unittests/OffloadAPI/common/environment.cpp
index a19e09fc5767c0..01973ec117e300 100644
--- a/offload/unittests/OffloadAPI/common/environment.cpp
+++ b/offload/unittests/OffloadAPI/common/environment.cpp
@@ -78,7 +78,7 @@ offload_platform_handle_t TestEnvironment::getPlatform() {
for (auto CandidatePlatform : Platforms) {
uint32_t NumDevices = 0;
if (offloadDeviceGet(CandidatePlatform, OFFLOAD_DEVICE_TYPE_ALL, 0,
- nullptr, &NumDevices) == OFFLOAD_RESULT_SUCCESS) {
+ nullptr, &NumDevices) == OFFLOAD_SUCCESS) {
if (NumDevices > 0) {
Platform = CandidatePlatform;
break;
diff --git a/offload/unittests/OffloadAPI/common/fixtures.hpp b/offload/unittests/OffloadAPI/common/fixtures.hpp
index a7f5f84adb2568..2366f8687a8e93 100644
--- a/offload/unittests/OffloadAPI/common/fixtures.hpp
+++ b/offload/unittests/OffloadAPI/common/fixtures.hpp
@@ -15,7 +15,16 @@
#pragma once
#ifndef ASSERT_SUCCESS
-#define ASSERT_SUCCESS(ACTUAL) ASSERT_EQ(OFFLOAD_RESULT_SUCCESS, ACTUAL)
+#define ASSERT_SUCCESS(ACTUAL) ASSERT_EQ(OFFLOAD_SUCCESS, ACTUAL)
+#endif
+
+// TODO: rework this so the EXPECTED/ACTUAL results are readable
+#ifndef ASSERT_ERROR
+#define ASSERT_ERROR(EXPECTED, ACTUAL) \
+ do { \
+ offload_result_t Res = ACTUAL; \
+ ASSERT_TRUE(Res && (Res->code == EXPECTED)); \
+ } while (0)
#endif
#define RETURN_ON_FATAL_FAILURE(...) \
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
index df8aecc1f0be2d..e9f5615b35d3ef 100644
--- a/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
@@ -60,43 +60,44 @@ TEST_P(offloadDeviceGetInfoTest, Success) {
TEST_F(offloadDeviceGetInfoTest, InvalidNullHandleDevice) {
offload_device_type_t DeviceType;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE,
- offloadDeviceGetInfo(nullptr, OFFLOAD_DEVICE_INFO_TYPE,
- sizeof(offload_device_type_t), &DeviceType,
- nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_HANDLE,
+ offloadDeviceGetInfo(nullptr, OFFLOAD_DEVICE_INFO_TYPE,
+ sizeof(offload_device_type_t), &DeviceType,
+ nullptr));
}
TEST_F(offloadDeviceGetInfoTest, InvalidEnumerationInfoType) {
offload_device_type_t DeviceType;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_FORCE_UINT32,
- sizeof(offload_device_type_t), &DeviceType,
- nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_ENUMERATION,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_FORCE_UINT32,
+ sizeof(offload_device_type_t), &DeviceType,
+ nullptr));
}
TEST_F(offloadDeviceGetInfoTest, InvalidSizePropSize) {
offload_device_type_t DeviceType;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0,
- &DeviceType, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0,
+ &DeviceType, nullptr));
}
TEST_F(offloadDeviceGetInfoTest, InvalidSizePropSizeSmall) {
offload_device_type_t DeviceType;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
- sizeof(DeviceType) - 1, &DeviceType, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
+ sizeof(DeviceType) - 1, &DeviceType,
+ nullptr));
}
TEST_F(offloadDeviceGetInfoTest, InvalidNullPointerPropValue) {
offload_device_type_t DeviceType;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
- sizeof(DeviceType), nullptr, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
+ sizeof(DeviceType), nullptr, nullptr));
}
TEST_F(offloadDeviceGetInfoTest, InvalidNullPointerPropSizeRet) {
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0, nullptr,
- nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0,
+ nullptr, nullptr));
}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
index 6ede71ff1babe6..14b7b9bba6b4d4 100644
--- a/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
@@ -23,6 +23,7 @@ TEST_F(offloadPlatformGetTest, InvalidNumEntries) {
uint32_t PlatformCount;
ASSERT_SUCCESS(offloadPlatformGet(0, nullptr, &PlatformCount));
std::vector<offload_platform_handle_t> Platforms(PlatformCount);
- ASSERT_EQ(offloadPlatformGet(0, Platforms.data(), nullptr),
- OFFLOAD_RESULT_ERROR_INVALID_SIZE);
+ ASSERT_ERROR(
+ OFFLOAD_ERRC_INVALID_SIZE,
+ offloadPlatformGet(PlatformCount + 1, Platforms.data(), nullptr));
}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
index 7a967ee674b00c..a5c86a7ba1933f 100644
--- a/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
@@ -57,41 +57,42 @@ TEST_P(offloadPlatformGetInfoTest, Success) {
TEST_F(offloadPlatformGetInfoTest, InvalidNullHandle) {
size_t Size = 0;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_HANDLE,
- offloadPlatformGetInfo(nullptr, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
- nullptr, &Size));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_HANDLE,
+ offloadPlatformGetInfo(nullptr, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
+ nullptr, &Size));
}
TEST_F(offloadPlatformGetInfoTest, InvalidPlatformInfoEnumeration) {
size_t Size = 0;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_ENUMERATION,
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_FORCE_UINT32,
- 0, nullptr, &Size));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_ENUMERATION,
+ offloadPlatformGetInfo(Platform,
+ OFFLOAD_PLATFORM_INFO_FORCE_UINT32, 0,
+ nullptr, &Size));
}
TEST_F(offloadPlatformGetInfoTest, InvalidSizeZero) {
offload_platform_backend_t Backend;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
- &Backend, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
+ 0, &Backend, nullptr));
}
TEST_F(offloadPlatformGetInfoTest, InvalidSizeSmall) {
offload_platform_backend_t Backend;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_SIZE,
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
- sizeof(Backend) - 1, &Backend, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
+ sizeof(Backend) - 1, &Backend, nullptr));
}
TEST_F(offloadPlatformGetInfoTest, InvalidNullPointerPropValue) {
offload_platform_backend_t Backend;
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
- sizeof(Backend), nullptr, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
+ sizeof(Backend), nullptr, nullptr));
}
TEST_F(offloadPlatformGetInfoTest, InvalidNullPointerPropSizeRet) {
- ASSERT_EQ(OFFLOAD_RESULT_ERROR_INVALID_NULL_POINTER,
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
- nullptr, nullptr));
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
+ 0, nullptr, nullptr));
}
>From fb6d7758eba23e496507c09e5fc40628266c9aae Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Wed, 23 Oct 2024 16:48:53 +0100
Subject: [PATCH 11/12] Add optional code location entry point variants
---
offload/new-api/API/Common.td | 11 ++++
offload/new-api/include/offload_api.h | 50 +++++++++++++++++++
.../new-api/include/offload_entry_points.inc | 44 ++++++++++++++++
offload/new-api/src/offload_lib.cpp | 5 ++
offload/tools/offload-tblgen/APIGen.cpp | 22 ++++++++
.../tools/offload-tblgen/EntryPointGen.cpp | 23 +++++++++
6 files changed, 155 insertions(+)
diff --git a/offload/new-api/API/Common.td b/offload/new-api/API/Common.td
index f2fe4b672fbfe5..c956ea9014596e 100644
--- a/offload/new-api/API/Common.td
+++ b/offload/new-api/API/Common.td
@@ -89,3 +89,14 @@ def : Macro {
let desc = "Success condition";
let value = "NULL";
}
+
+def : Struct {
+ let name = "offload_code_location_t";
+ let desc = "Code location information that can optionally be associated with an API call";
+ let members = [
+ StructMember<"const char*", "FunctionName", "Function name">,
+ StructMember<"const char*", "SourceFile", "Source code file">,
+ StructMember<"uint32_t", "LineNumber", "Source code line number">,
+ StructMember<"uint32_t", "ColumnNumber", "Source code column number">
+ ];
+}
diff --git a/offload/new-api/include/offload_api.h b/offload/new-api/include/offload_api.h
index a4d72b9a36a7c7..2e827655f16e7a 100644
--- a/offload/new-api/include/offload_api.h
+++ b/offload/new-api/include/offload_api.h
@@ -129,6 +129,16 @@ typedef const offload_error_struct_t *offload_result_t;
#endif // !defined(OFFLOAD_SUCCESS)
#endif // OFFLOAD_SUCCESS
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Code location information that can optionally be associated with an
+/// API call
+typedef struct offload_code_location_t {
+ const char *FunctionName; /// Function name
+ const char *SourceFile; /// Source code file
+ uint32_t LineNumber; /// Source code line number
+ uint32_t ColumnNumber; /// Source code column number
+} offload_code_location_t;
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves all available platforms
///
@@ -397,6 +407,46 @@ typedef struct offload_device_get_info_params_t {
size_t **ppPropSizeRet;
} offload_device_get_info_params_t;
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadPlatformGet that also sets source code location
+/// information
+/// @details See also ::offloadPlatformGet
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadPlatformGetWithCodeLoc(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms,
+ offload_code_location_t *pCodeLocation);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadPlatformGetInfo that also sets source code location
+/// information
+/// @details See also ::offloadPlatformGetInfo
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadPlatformGetInfoWithCodeLoc(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName,
+ size_t propSize, void *pPropValue,
+ size_t *pPropSizeRet,
+ offload_code_location_t *pCodeLocation);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadDeviceGet that also sets source code location
+/// information
+/// @details See also ::offloadDeviceGet
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetWithCodeLoc(
+ offload_platform_handle_t hPlatform, offload_device_type_t DeviceType,
+ uint32_t NumEntries, offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices, offload_code_location_t *pCodeLocation);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadDeviceGetInfo that also sets source code location
+/// information
+/// @details See also ::offloadDeviceGetInfo
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadDeviceGetInfoWithCodeLoc(offload_device_handle_t hDevice,
+ offload_device_info_t propName, size_t propSize,
+ void *pPropValue, size_t *pPropSizeRet,
+ offload_code_location_t *pCodeLocation);
+
#if defined(__cplusplus)
} // extern "C"
#endif
diff --git a/offload/new-api/include/offload_entry_points.inc b/offload/new-api/include/offload_entry_points.inc
index 0bdfabe7fefa9c..5c29f7cbe1ce44 100644
--- a/offload/new-api/include/offload_entry_points.inc
+++ b/offload/new-api/include/offload_entry_points.inc
@@ -40,6 +40,16 @@ offloadPlatformGet(uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
}
return result;
}
+offload_result_t offloadPlatformGetWithCodeLoc(
+ uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
+ uint32_t *pNumPlatforms, offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result =
+ offloadPlatformGet(NumEntries, phPlatforms, pNumPlatforms);
+
+ CodeLocation() = nullptr;
+ return result;
+}
///////////////////////////////////////////////////////////////////////////////
offload_impl_result_t
@@ -88,6 +98,17 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
}
return result;
}
+offload_result_t offloadPlatformGetInfoWithCodeLoc(
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName,
+ size_t propSize, void *pPropValue, size_t *pPropSizeRet,
+ offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadPlatformGetInfo(
+ hPlatform, propName, propSize, pPropValue, pPropSizeRet);
+
+ CodeLocation() = nullptr;
+ return result;
+}
///////////////////////////////////////////////////////////////////////////////
offload_impl_result_t offloadDeviceGet_val(offload_platform_handle_t hPlatform,
@@ -134,6 +155,17 @@ offloadDeviceGet(offload_platform_handle_t hPlatform,
}
return result;
}
+offload_result_t offloadDeviceGetWithCodeLoc(
+ offload_platform_handle_t hPlatform, offload_device_type_t DeviceType,
+ uint32_t NumEntries, offload_device_handle_t *phDevices,
+ uint32_t *pNumDevices, offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadDeviceGet(hPlatform, DeviceType, NumEntries,
+ phDevices, pNumDevices);
+
+ CodeLocation() = nullptr;
+ return result;
+}
///////////////////////////////////////////////////////////////////////////////
offload_impl_result_t offloadDeviceGetInfo_val(offload_device_handle_t hDevice,
@@ -183,3 +215,15 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
}
return result;
}
+offload_result_t
+offloadDeviceGetInfoWithCodeLoc(offload_device_handle_t hDevice,
+ offload_device_info_t propName, size_t propSize,
+ void *pPropValue, size_t *pPropSizeRet,
+ offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadDeviceGetInfo(hDevice, propName, propSize,
+ pPropValue, pPropSizeRet);
+
+ CodeLocation() = nullptr;
+ return result;
+}
diff --git a/offload/new-api/src/offload_lib.cpp b/offload/new-api/src/offload_lib.cpp
index 7ad0ff1d6e35c3..2635152c0910a8 100644
--- a/offload/new-api/src/offload_lib.cpp
+++ b/offload/new-api/src/offload_lib.cpp
@@ -26,6 +26,11 @@ ErrSetT &Errors() {
return Errors;
}
+offload_code_location_t *&CodeLocation() {
+ thread_local offload_code_location_t *CodeLoc = nullptr;
+ return CodeLoc;
+}
+
// Pull in the declarations for the implementation funtions. The actual entry
// points in this file wrap these.
#include "offload_impl_func_decls.inc"
diff --git a/offload/tools/offload-tblgen/APIGen.cpp b/offload/tools/offload-tblgen/APIGen.cpp
index 9395f589952594..cd9fe54085f1a8 100644
--- a/offload/tools/offload-tblgen/APIGen.cpp
+++ b/offload/tools/offload-tblgen/APIGen.cpp
@@ -168,6 +168,24 @@ typedef struct {1} {{
OS << formatv("} {0};\n", Func.getParamStructName());
}
+static void ProcessFuncWithCodeLocVariant(const FunctionRec &Func,
+ raw_ostream &OS) {
+
+ auto FuncWithCodeLocBegin = R"(
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of {0} that also sets source code location information
+/// @details See also ::{0}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL {0}WithCodeLoc(
+)";
+ OS << formatv(FuncWithCodeLocBegin, Func.getName());
+ auto Params = Func.getParams();
+ for (auto &Param : Params) {
+ OS << " " << Param.getType() << " " << Param.getName();
+ OS << ",\n";
+ }
+ OS << "offload_code_location_t *pCodeLocation);\n\n";
+}
+
void EmitOffloadAPI(RecordKeeper &Records, raw_ostream &OS) {
OS << GenericHeader;
OS << FileHeader;
@@ -193,5 +211,9 @@ void EmitOffloadAPI(RecordKeeper &Records, raw_ostream &OS) {
ProcessFuncParamStruct(FunctionRec{R}, OS);
}
+ for (auto *R : Records.getAllDerivedDefinitions("Function")) {
+ ProcessFuncWithCodeLocVariant(FunctionRec{R}, OS);
+ }
+
OS << FileFooter;
}
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
index e36252e482c426..c5c823f68cb56e 100644
--- a/offload/tools/offload-tblgen/EntryPointGen.cpp
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -102,10 +102,33 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
OS << "}\n";
}
+static void EmitCodeLocWrapper(const FunctionRec &F, raw_ostream &OS) {
+ // Emit preamble
+ OS << formatv("{0}_result_t {1}WithCodeLoc(\n ", PrefixLower, F.getName());
+ // Emit arguments
+ std::string ParamNameList = "";
+ for (auto &Param : F.getParams()) {
+ OS << Param.getType() << " " << Param.getName() << ", ";
+ ParamNameList += Param.getName().str();
+ if (Param != F.getParams().back()) {
+ ParamNameList += ", ";
+ }
+ }
+ OS << "offload_code_location_t *pCodeLocation";
+ OS << ") {\n";
+ OS << TAB_1 "CodeLocation() = pCodeLocation;\n";
+ OS << formatv(TAB_1 "{0}_result_t result = {1}({2});\n\n", PrefixLower,
+ F.getName(), ParamNameList);
+ OS << TAB_1 "CodeLocation() = nullptr;\n";
+ OS << TAB_1 "return result;\n";
+ OS << "}\n";
+}
+
void EmitOffloadEntryPoints(RecordKeeper &Records, raw_ostream &OS) {
OS << GenericHeader;
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
EmitValidationFunc(FunctionRec{R}, OS);
EmitEntryPointFunc(FunctionRec{R}, OS);
+ EmitCodeLocWrapper(FunctionRec{R}, OS);
}
}
>From 55740e022772e3b315d51bd1a6b08641efa8b7e3 Mon Sep 17 00:00:00 2001
From: Callum Fare <callum at codeplay.com>
Date: Fri, 25 Oct 2024 10:10:23 +0100
Subject: [PATCH 12/12] Rework API to avoid multiple returns, add
init/shutdown, general refactor
---
offload/new-api/API/APIDefs.td | 6 +-
offload/new-api/API/Common.td | 25 +-
offload/new-api/API/Device.td | 66 +--
offload/new-api/API/Platform.td | 61 ++-
offload/new-api/include/offload_api.h | 321 +++++++++++----
.../new-api/include/offload_entry_points.inc | 388 ++++++++++++++----
offload/new-api/include/offload_exports | 6 +
offload/new-api/include/offload_funcs.inc | 6 +
offload/new-api/include/offload_impl.hpp | 2 +-
.../include/offload_impl_func_decls.inc | 32 +-
offload/new-api/include/offload_print.hpp | 52 ++-
offload/new-api/src/offload_impl.cpp | 116 ++++--
offload/tools/offload-tblgen/APIGen.cpp | 4 +
.../tools/offload-tblgen/EntryPointGen.cpp | 18 +-
offload/tools/offload-tblgen/FuncsGen.cpp | 12 +-
offload/tools/offload-tblgen/PrintGen.cpp | 8 +-
offload/unittests/OffloadAPI/CMakeLists.txt | 6 +-
.../OffloadAPI/common/environment.cpp | 21 +-
.../unittests/OffloadAPI/common/fixtures.hpp | 16 +-
.../OffloadAPI/device/offloadDeviceGet.cpp | 12 +-
.../device/offloadDeviceGetCount.cpp | 30 ++
.../device/offloadDeviceGetInfo.cpp | 49 +--
.../device/offloadDeviceGetInfoSize.cpp | 61 +++
.../OffloadAPI/device/offloadDeviceInfo.hpp | 22 +
.../platform/offloadPlatformGet.cpp | 12 +-
.../platform/offloadPlatformGetCount.cpp | 23 ++
.../platform/offloadPlatformGetInfo.cpp | 50 +--
.../platform/offloadPlatformGetInfoSize.cpp | 58 +++
.../platform/offloadPlatformInfo.hpp | 21 +
29 files changed, 1118 insertions(+), 386 deletions(-)
create mode 100644 offload/unittests/OffloadAPI/device/offloadDeviceGetCount.cpp
create mode 100644 offload/unittests/OffloadAPI/device/offloadDeviceGetInfoSize.cpp
create mode 100644 offload/unittests/OffloadAPI/device/offloadDeviceInfo.hpp
create mode 100644 offload/unittests/OffloadAPI/platform/offloadPlatformGetCount.cpp
create mode 100644 offload/unittests/OffloadAPI/platform/offloadPlatformGetInfoSize.cpp
create mode 100644 offload/unittests/OffloadAPI/platform/offloadPlatformInfo.hpp
diff --git a/offload/new-api/API/APIDefs.td b/offload/new-api/API/APIDefs.td
index a74332363369ac..64785217b48849 100644
--- a/offload/new-api/API/APIDefs.td
+++ b/offload/new-api/API/APIDefs.td
@@ -125,10 +125,10 @@ class AddPointerChecksToReturns<list<Param> Params, list<Return> Returns> {
// 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#"_ERROR_CODE_INVALID_NULL_POINTER")));
+ !or(HasErr, !eq(Ret.value, PREFIX#"_ERRC_INVALID_NULL_POINTER")));
list<Return> returns_out = !if(returns_has_inv_ptr,
- AppendConditionsToReturn<Returns, PREFIX # "_ERROR_CODE_INVALID_NULL_POINTER", ptr_param_conds>.ret,
- !listconcat(Returns, [Return<PREFIX # "_ERROR_CODE_INVALID_NULL_POINTER", ptr_param_conds>])
+ AppendConditionsToReturn<Returns, PREFIX # "_ERRC_INVALID_NULL_POINTER", ptr_param_conds>.ret,
+ !listconcat(Returns, [Return<PREFIX # "_ERRC_INVALID_NULL_POINTER", ptr_param_conds>])
);
}
diff --git a/offload/new-api/API/Common.td b/offload/new-api/API/Common.td
index c956ea9014596e..c2f03bbc0ae355 100644
--- a/offload/new-api/API/Common.td
+++ b/offload/new-api/API/Common.td
@@ -84,8 +84,7 @@ def : Typedef {
}
def : Macro {
- let name = "OFFLOAD_SUCCESS";
- let condition = "!defined(OFFLOAD_SUCCESS)";
+ let name = "OFFLOAD_RESULT_SUCCESS";
let desc = "Success condition";
let value = "NULL";
}
@@ -100,3 +99,25 @@ def : Struct {
StructMember<"uint32_t", "ColumnNumber", "Source code column number">
];
}
+
+def : Function {
+ let name = "offloadInit";
+ let desc = "Perform initialization of the Offload library and plugins";
+ let details = [
+ "This must be the first API call made by a user of the Offload library",
+ "Each call will increment an internal reference count that is decremented by `offloadShutDown`"
+ ];
+ let params = [];
+ let returns = [];
+}
+
+def : Function {
+ let name = "offloadShutDown";
+ let desc = "Release the resources in use by Offload";
+ let details = [
+ "This decrements an internal reference count. When this reaches 0, all resources will be released",
+ "Subsequent API calls made after this are not valid"
+ ];
+ let params = [];
+ let returns = [];
+}
diff --git a/offload/new-api/API/Device.td b/offload/new-api/API/Device.td
index da1141cf1986e0..2700f3c9f639f9 100644
--- a/offload/new-api/API/Device.td
+++ b/offload/new-api/API/Device.td
@@ -34,41 +34,40 @@ def : Enum {
];
}
+def : Function {
+ let name = "offloadDeviceGetCount";
+ let desc = "Retrieves the number of available devices within a platform";
+ let params = [
+ Param<"offload_platform_handle_t", "hPlatform", "handle of the platform instance", PARAM_IN>,
+ Param<"uint32_t*", "pNumDevices", "pointer to the number of devices.", PARAM_OUT>
+ ];
+ let returns = [];
+}
+
def : Function {
let name = "offloadDeviceGet";
let desc = "Retrieves devices within a platform";
let details = [
"Multiple calls to this function will return identical device handles, in the same order.",
- "The number and order of handles returned from this function can be affected by environment variables that filter devices exposed through API.",
- "The returned devices are taken a reference of and must be released with a subsequent call to olDeviceRelease.",
"The application may call this function from simultaneous threads, the implementation must be thread-safe"
];
let params = [
Param<"offload_platform_handle_t", "hPlatform", "handle of the platform instance", PARAM_IN>,
- Param<"offload_device_type_t", "DeviceType", "the type of the devices.", PARAM_IN>,
- Param<"uint32_t", "NumEntries", "the number of devices to be added to phDevices. "
- "If phDevices is not NULL, then NumEntries should be greater than zero. Otherwise OFFLOAD_ERRC_INVALID_SIZE "
- "will be returned.", PARAM_IN>,
+ Param<"uint32_t", "NumEntries", "the number of devices to be added to phDevices, which must be greater than zero", PARAM_IN>,
RangedParam<"offload_device_handle_t*", "phDevices", "Array of device handles. "
- "If NumEntries is less than the number of devices available, then platform shall only retrieve that number of devices.", PARAM_OUT_OPTIONAL,
- Range<"0", "NumEntries">>,
- Param<"uint32_t*", "pNumDevices", "pointer to the number of devices. "
- "pNumDevices will be updated with the total number of devices available.", PARAM_OUT_OPTIONAL>
+ "If NumEntries is less than the number of devices available, then this function shall only retrieve that number of devices.", PARAM_OUT,
+ Range<"0", "NumEntries">>
];
let returns = [
Return<"OFFLOAD_ERRC_INVALID_SIZE", [
- "`NumEntries == 0 && phDevices != NULL`"
- ]>,
- Return<"OFFLOAD_ERRC_INVALID_NULL_POINTER", [
- "`NumEntries > 0 && phDevices == NULL`"
- ]>,
- Return<"OFFLOAD_ERRC_INVALID_VALUE">
+ "`NumEntries == 0`"
+ ]>
];
}
def : Function {
let name = "offloadDeviceGetInfo";
- let desc = "Retrieves various information about device";
+ let desc = "Queries the given property of the device";
let details = [
"The application may call this function from simultaneous threads.",
"The implementation of this function should be lock-free."
@@ -79,23 +78,36 @@ def : Function {
Param<"size_t", "propSize", "the number of bytes pointed to by pPropValue.", PARAM_IN>,
TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info. If propSize is not equal to or greater than the real "
"number of bytes needed to return the info then the OFFLOAD_ERRC_INVALID_SIZE error is returned and "
- "pPropValue is not used.", PARAM_OUT_OPTIONAL, TypeInfo<"propName" , "propSize">>,
- Param<"size_t*", "pPropSizeRet", "pointer to the actual size in bytes of the queried propName.", PARAM_OUT_OPTIONAL>
+ "pPropValue is not used.", PARAM_OUT, TypeInfo<"propName" , "propSize">>
];
let returns = [
Return<"OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION", [
- "If `propName` is not supported by the adapter."
+ "If `propName` is not supported by the device."
]>,
Return<"OFFLOAD_ERRC_INVALID_SIZE", [
- "`propSize == 0 && pPropValue != NULL`",
+ "`propSize == 0`",
"If `propSize` is less than the real number of bytes needed to return the info."
]>,
- Return<"OFFLOAD_ERRC_INVALID_NULL_POINTER", [
- "`propSize != 0 && pPropValue == NULL`",
- "`pPropValue == NULL && pPropSizeRet == NULL`"
+ Return<"OFFLOAD_ERRC_INVALID_DEVICE">
+ ];
+}
+
+def : Function {
+ let name = "offloadDeviceGetInfoSize";
+ let desc = "Returns the storage size of the given device query";
+ let details = [
+ "The application may call this function from simultaneous threads.",
+ "The implementation of this function should be lock-free."
+ ];
+ let params = [
+ Param<"offload_device_handle_t", "hDevice", "handle of the device instance", PARAM_IN>,
+ Param<"offload_device_info_t", "propName", "type of the info to retrieve", PARAM_IN>,
+ Param<"size_t*", "pPropSizeRet", "pointer to the number of bytes required to store the query", PARAM_OUT>
+ ];
+ let returns = [
+ Return<"OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION", [
+ "If `propName` is not supported by the device."
]>,
- Return<"OFFLOAD_ERRC_INVALID_DEVICE">,
- Return<"OFFLOAD_ERRC_OUT_OF_RESOURCES">,
- Return<"OFFLOAD_ERRC_OUT_OF_HOST_MEMORY">
+ Return<"OFFLOAD_ERRC_INVALID_DEVICE">
];
}
diff --git a/offload/new-api/API/Platform.td b/offload/new-api/API/Platform.td
index 0bfcbdce8d2d5c..564bea017cab9e 100644
--- a/offload/new-api/API/Platform.td
+++ b/offload/new-api/API/Platform.td
@@ -17,25 +17,33 @@ def : Function {
];
let params = [
Param<"uint32_t", "NumEntries",
- "The number of platforms to be added to phPlatforms. If phPlatforms is not NULL, then "
- "NumEntries should be greater than zero, otherwise OFFLOAD_ERRC_INVALID_SIZE "
- "will be returned.", PARAM_IN>,
+ "The number of platforms to be added to phPlatforms. NumEntries must be "
+ "greater than zero.",
+ PARAM_IN>,
RangedParam<"offload_platform_handle_t*", "phPlatforms",
"Array of handle of platforms. If NumEntries is less than the number of "
"platforms available, then offloadPlatformGet shall only retrieve that "
"number of platforms.",
- PARAM_OUT_OPTIONAL, Range<"0", "NumEntries">>,
- Param<"uint32_t*",
- "pNumPlatforms", "returns the total number of platforms available.",
- PARAM_OUT_OPTIONAL>
+ PARAM_OUT, Range<"0", "NumEntries">>
];
let returns = [
Return<"OFFLOAD_ERRC_INVALID_SIZE", [
- "`NumEntries == 0 && phPlatforms != NULL`"
+ "`NumEntries == 0`"
]>
];
}
+def : Function {
+ let name = "offloadPlatformGetCount";
+ let desc = "Retrieves the number of available platforms";
+ let params = [
+ Param<"uint32_t*",
+ "pNumPlatforms", "returns the total number of platforms available.",
+ PARAM_OUT>
+ ];
+ let returns = [];
+}
+
def : Enum {
let name = "offload_platform_info_t";
let desc = "Supported platform info";
@@ -60,8 +68,10 @@ def : Enum {
def : Function {
let name = "offloadPlatformGetInfo";
- let desc = "Retrieves various information about platform";
+ let desc = "Queries the given property of the platform";
let details = [
+ "`offloadPlatformGetInfoSize` can be used to query the storage size "
+ "required for the given query."
"The application may call this function from simultaneous threads.",
"The implementation of this function should be lock-free."
];
@@ -71,24 +81,37 @@ def : Function {
Param<"size_t", "propSize", "the number of bytes pointed to by pPlatformInfo.", PARAM_IN>,
TypeTaggedParam<"void*", "pPropValue", "array of bytes holding the info. "
"If Size is not equal to or greater to the real number of bytes needed to return the info "
- "then the OFFLOAD_ERRC_INVALID_SIZE error is returned and pPlatformInfo is not used.", PARAM_OUT_OPTIONAL,
- TypeInfo<"propName" , "propSize">>,
- Param<"size_t*", "pPropSizeRet", "pointer to the actual number of bytes being queried by pPlatformInfo.", PARAM_OUT_OPTIONAL>
+ "then the OFFLOAD_ERRC_INVALID_SIZE error is returned and pPlatformInfo is not used.", PARAM_OUT,
+ TypeInfo<"propName" , "propSize">>
];
let returns = [
Return<"OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION", [
"If `propName` is not supported by the platform."
]>,
Return<"OFFLOAD_ERRC_INVALID_SIZE", [
- "`propSize == 0 && pPropValue != NULL`",
+ "`propSize == 0`",
"If `propSize` is less than the real number of bytes needed to return the info."
]>,
- Return<"OFFLOAD_ERRC_INVALID_NULL_POINTER", [
- "`propSize != 0 && pPropValue == NULL`",
- "`pPropValue == NULL && pPropSizeRet == NULL`"
+ Return<"OFFLOAD_ERRC_INVALID_PLATFORM">
+ ];
+}
+
+def : Function {
+ let name = "offloadPlatformGetInfoSize";
+ let desc = "Returns the storage size of the given platform query";
+ let details = [
+ "The application may call this function from simultaneous threads.",
+ "The implementation of this function should be lock-free."
+ ];
+ let params = [
+ Param<"offload_platform_handle_t", "hPlatform", "handle of the platform", PARAM_IN>,
+ Param<"offload_platform_info_t", "propName", "type of the info to query", PARAM_IN>,
+ Param<"size_t*", "pPropSizeRet", "pointer to the number of bytes required to store the query", PARAM_OUT>
+ ];
+ let returns = [
+ Return<"OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION", [
+ "If `propName` is not supported by the platform."
]>,
- Return<"OFFLOAD_ERRC_INVALID_PLATFORM">,
- Return<"OFFLOAD_ERRC_OUT_OF_RESOURCES">,
- Return<"OFFLOAD_ERRC_OUT_OF_HOST_MEMORY">
+ Return<"OFFLOAD_ERRC_INVALID_PLATFORM">
];
}
diff --git a/offload/new-api/include/offload_api.h b/offload/new-api/include/offload_api.h
index 2e827655f16e7a..4816abc22fd0b4 100644
--- a/offload/new-api/include/offload_api.h
+++ b/offload/new-api/include/offload_api.h
@@ -122,12 +122,10 @@ typedef struct offload_error_struct_t {
typedef const offload_error_struct_t *offload_result_t;
///////////////////////////////////////////////////////////////////////////////
-#ifndef OFFLOAD_SUCCESS
-#if !defined(OFFLOAD_SUCCESS)
+#ifndef OFFLOAD_RESULT_SUCCESS
/// @brief Success condition
-#define OFFLOAD_SUCCESS NULL
-#endif // !defined(OFFLOAD_SUCCESS)
-#endif // OFFLOAD_SUCCESS
+#define OFFLOAD_RESULT_SUCCESS NULL
+#endif // OFFLOAD_RESULT_SUCCESS
///////////////////////////////////////////////////////////////////////////////
/// @brief Code location information that can optionally be associated with an
@@ -139,6 +137,38 @@ typedef struct offload_code_location_t {
uint32_t ColumnNumber; /// Source code column number
} offload_code_location_t;
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Perform initialization of the Offload library and plugins
+///
+/// @details
+/// - This must be the first API call made by a user of the Offload library
+/// - Each call will increment an internal reference count that is
+/// decremented by `offloadShutDown`
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadInit();
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Release the resources in use by Offload
+///
+/// @details
+/// - This decrements an internal reference count. When this reaches 0, all
+/// resources will be released
+/// - Subsequent API calls made after this are not valid
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadShutDown();
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves all available platforms
///
@@ -151,19 +181,33 @@ typedef struct offload_code_location_t {
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
/// - ::OFFLOAD_ERRC_INVALID_SIZE
-/// + `NumEntries == 0 && phPlatforms != NULL`
+/// + `NumEntries == 0`
/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
-/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == phPlatforms`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
- // [in] The number of platforms to be added to phPlatforms. If phPlatforms
- // is not NULL, then NumEntries should be greater than zero, otherwise
- // OFFLOAD_ERRC_INVALID_SIZE will be returned.
+ // [in] The number of platforms to be added to phPlatforms. NumEntries must
+ // be greater than zero.
uint32_t NumEntries,
- // [out][optional] Array of handle of platforms. If NumEntries is less than
- // the number of platforms available, then offloadPlatformGet shall only
- // retrieve that number of platforms.
- offload_platform_handle_t *phPlatforms,
- // [out][optional] returns the total number of platforms available.
+ // [out] Array of handle of platforms. If NumEntries is less than the number
+ // of platforms available, then offloadPlatformGet shall only retrieve that
+ // number of platforms.
+ offload_platform_handle_t *phPlatforms);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Retrieves the number of available platforms
+///
+/// @details
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == pNumPlatforms`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetCount(
+ // [out] returns the total number of platforms available.
uint32_t *pNumPlatforms);
///////////////////////////////////////////////////////////////////////////////
@@ -203,10 +247,12 @@ typedef enum offload_platform_backend_t {
} offload_platform_backend_t;
///////////////////////////////////////////////////////////////////////////////
-/// @brief Retrieves various information about platform
+/// @brief Queries the given property of the platform
///
/// @details
-/// - The application may call this function from simultaneous threads.
+/// - `offloadPlatformGetInfoSize` can be used to query the storage size
+/// required for the given query.The application may call this function from
+/// simultaneous threads.
/// - The implementation of this function should be lock-free.
///
/// @returns
@@ -216,18 +262,14 @@ typedef enum offload_platform_backend_t {
/// - ::OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION
/// + If `propName` is not supported by the platform.
/// - ::OFFLOAD_ERRC_INVALID_SIZE
-/// + `propSize == 0 && pPropValue != NULL`
+/// + `propSize == 0`
/// + If `propSize` is less than the real number of bytes needed to
/// return the info.
-/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
-/// + `propSize != 0 && pPropValue == NULL`
-/// + `pPropValue == NULL && pPropSizeRet == NULL`
/// - ::OFFLOAD_ERRC_INVALID_PLATFORM
-/// - ::OFFLOAD_ERRC_OUT_OF_RESOURCES
-/// - ::OFFLOAD_ERRC_OUT_OF_HOST_MEMORY
/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
-/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == pPropValue`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
// [in] handle of the platform
offload_platform_handle_t hPlatform,
@@ -235,13 +277,36 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
offload_platform_info_t propName,
// [in] the number of bytes pointed to by pPlatformInfo.
size_t propSize,
- // [out][optional] array of bytes holding the info. If Size is not equal to
- // or greater to the real number of bytes needed to return the info then the
+ // [out] array of bytes holding the info. If Size is not equal to or greater
+ // to the real number of bytes needed to return the info then the
// OFFLOAD_ERRC_INVALID_SIZE error is returned and pPlatformInfo is not
// used.
- void *pPropValue,
- // [out][optional] pointer to the actual number of bytes being queried by
- // pPlatformInfo.
+ void *pPropValue);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Returns the storage size of the given platform query
+///
+/// @details
+/// - The application may call this function from simultaneous threads.
+/// - The implementation of this function should be lock-free.
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION
+/// + If `propName` is not supported by the platform.
+/// - ::OFFLOAD_ERRC_INVALID_PLATFORM
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// + `NULL == hPlatform`
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == pPropSizeRet`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfoSize(
+ // [in] handle of the platform
+ offload_platform_handle_t hPlatform,
+ // [in] type of the info to query
+ offload_platform_info_t propName,
+ // [out] pointer to the number of bytes required to store the query
size_t *pPropSizeRet);
///////////////////////////////////////////////////////////////////////////////
@@ -280,17 +345,31 @@ typedef enum offload_device_info_t {
} offload_device_info_t;
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Retrieves the number of available devices within a platform
+///
+/// @details
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// + `NULL == hPlatform`
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == pNumDevices`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetCount(
+ // [in] handle of the platform instance
+ offload_platform_handle_t hPlatform,
+ // [out] pointer to the number of devices.
+ uint32_t *pNumDevices);
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Retrieves devices within a platform
///
/// @details
/// - Multiple calls to this function will return identical device handles,
/// in the same order.
-/// - The number and order of handles returned from this function can be
-/// affected by environment variables that filter devices exposed through
-/// API.
-/// - The returned devices are taken a reference of and must be released with
-/// a subsequent call to olDeviceRelease.
/// - The application may call this function from simultaneous threads, the
/// implementation must be thread-safe
///
@@ -299,32 +378,24 @@ typedef enum offload_device_info_t {
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
/// - ::OFFLOAD_ERRC_INVALID_SIZE
-/// + `NumEntries == 0 && phDevices != NULL`
-/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
-/// + `NumEntries > 0 && phDevices == NULL`
-/// - ::OFFLOAD_ERRC_INVALID_VALUE
+/// + `NumEntries == 0`
/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
/// + `NULL == hPlatform`
-/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == phDevices`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
// [in] handle of the platform instance
offload_platform_handle_t hPlatform,
- // [in] the type of the devices.
- offload_device_type_t DeviceType,
- // [in] the number of devices to be added to phDevices. If phDevices is not
- // NULL, then NumEntries should be greater than zero. Otherwise
- // OFFLOAD_ERRC_INVALID_SIZE will be returned.
+ // [in] the number of devices to be added to phDevices, which must be
+ // greater than zero
uint32_t NumEntries,
- // [out][optional] Array of device handles. If NumEntries is less than the
- // number of devices available, then platform shall only retrieve that
- // number of devices.
- offload_device_handle_t *phDevices,
- // [out][optional] pointer to the number of devices. pNumDevices will be
- // updated with the total number of devices available.
- uint32_t *pNumDevices);
+ // [out] Array of device handles. If NumEntries is less than the number of
+ // devices available, then this function shall only retrieve that number of
+ // devices.
+ offload_device_handle_t *phDevices);
///////////////////////////////////////////////////////////////////////////////
-/// @brief Retrieves various information about device
+/// @brief Queries the given property of the device
///
/// @details
/// - The application may call this function from simultaneous threads.
@@ -335,20 +406,16 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGet(
/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
/// - ::OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION
-/// + If `propName` is not supported by the adapter.
+/// + If `propName` is not supported by the device.
/// - ::OFFLOAD_ERRC_INVALID_SIZE
-/// + `propSize == 0 && pPropValue != NULL`
+/// + `propSize == 0`
/// + If `propSize` is less than the real number of bytes needed to
/// return the info.
-/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
-/// + `propSize != 0 && pPropValue == NULL`
-/// + `pPropValue == NULL && pPropSizeRet == NULL`
/// - ::OFFLOAD_ERRC_INVALID_DEVICE
-/// - ::OFFLOAD_ERRC_OUT_OF_RESOURCES
-/// - ::OFFLOAD_ERRC_OUT_OF_HOST_MEMORY
/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
/// + `NULL == hDevice`
-/// - ::OFFLOAD_ERROR_CODE_INVALID_NULL_POINTER
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == pPropValue`
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
// [in] handle of the device instance
offload_device_handle_t hDevice,
@@ -356,13 +423,35 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
offload_device_info_t propName,
// [in] the number of bytes pointed to by pPropValue.
size_t propSize,
- // [out][optional] array of bytes holding the info. If propSize is not equal
- // to or greater than the real number of bytes needed to return the info
- // then the OFFLOAD_ERRC_INVALID_SIZE error is returned and pPropValue is
- // not used.
- void *pPropValue,
- // [out][optional] pointer to the actual size in bytes of the queried
- // propName.
+ // [out] array of bytes holding the info. If propSize is not equal to or
+ // greater than the real number of bytes needed to return the info then the
+ // OFFLOAD_ERRC_INVALID_SIZE error is returned and pPropValue is not used.
+ void *pPropValue);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Returns the storage size of the given device query
+///
+/// @details
+/// - The application may call this function from simultaneous threads.
+/// - The implementation of this function should be lock-free.
+///
+/// @returns
+/// - ::OFFLOAD_RESULT_SUCCESS
+/// - ::OFFLOAD_RESULT_ERROR_UNINITIALIZED
+/// - ::OFFLOAD_RESULT_ERROR_DEVICE_LOST
+/// - ::OFFLOAD_ERRC_UNSUPPORTED_ENUMERATION
+/// + If `propName` is not supported by the device.
+/// - ::OFFLOAD_ERRC_INVALID_DEVICE
+/// - ::OFFLOAD_ERRC_INVALID_NULL_HANDLE
+/// + `NULL == hDevice`
+/// - ::OFFLOAD_ERRC_INVALID_NULL_POINTER
+/// + `NULL == pPropSizeRet`
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfoSize(
+ // [in] handle of the device instance
+ offload_device_handle_t hDevice,
+ // [in] type of the info to retrieve
+ offload_device_info_t propName,
+ // [out] pointer to the number of bytes required to store the query
size_t *pPropSizeRet);
///////////////////////////////////////////////////////////////////////////////
@@ -371,9 +460,15 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
typedef struct offload_platform_get_params_t {
uint32_t *pNumEntries;
offload_platform_handle_t **pphPlatforms;
- uint32_t **ppNumPlatforms;
} offload_platform_get_params_t;
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadPlatformGetCount
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_platform_get_count_params_t {
+ uint32_t **ppNumPlatforms;
+} offload_platform_get_count_params_t;
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadPlatformGetInfo
/// @details Each entry is a pointer to the parameter passed to the function;
@@ -382,18 +477,32 @@ typedef struct offload_platform_get_info_params_t {
offload_platform_info_t *ppropName;
size_t *ppropSize;
void **ppPropValue;
- size_t **ppPropSizeRet;
} offload_platform_get_info_params_t;
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadPlatformGetInfoSize
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_platform_get_info_size_params_t {
+ offload_platform_handle_t *phPlatform;
+ offload_platform_info_t *ppropName;
+ size_t **ppPropSizeRet;
+} offload_platform_get_info_size_params_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadDeviceGetCount
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_device_get_count_params_t {
+ offload_platform_handle_t *phPlatform;
+ uint32_t **ppNumDevices;
+} offload_device_get_count_params_t;
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for offloadDeviceGet
/// @details Each entry is a pointer to the parameter passed to the function;
typedef struct offload_device_get_params_t {
offload_platform_handle_t *phPlatform;
- offload_device_type_t *pDeviceType;
uint32_t *pNumEntries;
offload_device_handle_t **pphDevices;
- uint32_t **ppNumDevices;
} offload_device_get_params_t;
///////////////////////////////////////////////////////////////////////////////
@@ -404,9 +513,31 @@ typedef struct offload_device_get_info_params_t {
offload_device_info_t *ppropName;
size_t *ppropSize;
void **ppPropValue;
- size_t **ppPropSizeRet;
} offload_device_get_info_params_t;
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Function parameters for offloadDeviceGetInfoSize
+/// @details Each entry is a pointer to the parameter passed to the function;
+typedef struct offload_device_get_info_size_params_t {
+ offload_device_handle_t *phDevice;
+ offload_device_info_t *ppropName;
+ size_t **ppPropSizeRet;
+} offload_device_get_info_size_params_t;
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadInit that also sets source code location
+/// information
+/// @details See also ::offloadInit
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadInitWithCodeLoc(offload_code_location_t *pCodeLocation);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadShutDown that also sets source code location
+/// information
+/// @details See also ::offloadShutDown
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadShutDownWithCodeLoc(offload_code_location_t *pCodeLocation);
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Variant of offloadPlatformGet that also sets source code location
/// information
@@ -414,9 +545,16 @@ typedef struct offload_device_get_info_params_t {
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
offloadPlatformGetWithCodeLoc(uint32_t NumEntries,
offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms,
offload_code_location_t *pCodeLocation);
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadPlatformGetCount that also sets source code
+/// location information
+/// @details See also ::offloadPlatformGetCount
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadPlatformGetCountWithCodeLoc(uint32_t *pNumPlatforms,
+ offload_code_location_t *pCodeLocation);
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Variant of offloadPlatformGetInfo that also sets source code location
/// information
@@ -425,17 +563,34 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
offloadPlatformGetInfoWithCodeLoc(offload_platform_handle_t hPlatform,
offload_platform_info_t propName,
size_t propSize, void *pPropValue,
- size_t *pPropSizeRet,
offload_code_location_t *pCodeLocation);
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadPlatformGetInfoSize that also sets source code
+/// location information
+/// @details See also ::offloadPlatformGetInfoSize
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadPlatformGetInfoSizeWithCodeLoc(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName,
+ size_t *pPropSizeRet,
+ offload_code_location_t *pCodeLocation);
+
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadDeviceGetCount that also sets source code location
+/// information
+/// @details See also ::offloadDeviceGetCount
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadDeviceGetCountWithCodeLoc(offload_platform_handle_t hPlatform,
+ uint32_t *pNumDevices,
+ offload_code_location_t *pCodeLocation);
+
///////////////////////////////////////////////////////////////////////////////
/// @brief Variant of offloadDeviceGet that also sets source code location
/// information
/// @details See also ::offloadDeviceGet
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetWithCodeLoc(
- offload_platform_handle_t hPlatform, offload_device_type_t DeviceType,
- uint32_t NumEntries, offload_device_handle_t *phDevices,
- uint32_t *pNumDevices, offload_code_location_t *pCodeLocation);
+ offload_platform_handle_t hPlatform, uint32_t NumEntries,
+ offload_device_handle_t *phDevices, offload_code_location_t *pCodeLocation);
///////////////////////////////////////////////////////////////////////////////
/// @brief Variant of offloadDeviceGetInfo that also sets source code location
@@ -444,9 +599,19 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetWithCodeLoc(
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
offloadDeviceGetInfoWithCodeLoc(offload_device_handle_t hDevice,
offload_device_info_t propName, size_t propSize,
- void *pPropValue, size_t *pPropSizeRet,
+ void *pPropValue,
offload_code_location_t *pCodeLocation);
+///////////////////////////////////////////////////////////////////////////////
+/// @brief Variant of offloadDeviceGetInfoSize that also sets source code
+/// location information
+/// @details See also ::offloadDeviceGetInfoSize
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadDeviceGetInfoSizeWithCodeLoc(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t *pPropSizeRet,
+ offload_code_location_t *pCodeLocation);
+
#if defined(__cplusplus)
} // extern "C"
#endif
diff --git a/offload/new-api/include/offload_entry_points.inc b/offload/new-api/include/offload_entry_points.inc
index 5c29f7cbe1ce44..cecc4699fb6646 100644
--- a/offload/new-api/include/offload_entry_points.inc
+++ b/offload/new-api/include/offload_entry_points.inc
@@ -6,32 +6,96 @@
//
//===----------------------------------------------------------------------===//
+///////////////////////////////////////////////////////////////////////////////
+offload_impl_result_t offloadInit_val() {
+ if (true /*enableParameterValidation*/) {
+ }
+
+ return offloadInit_impl();
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadInit() {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadInit";
+ }
+
+ offload_result_t result = offloadInit_val();
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "()";
+ std::cout << "-> " << result << "\n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
+ }
+ }
+ return result;
+}
+offload_result_t
+offloadInitWithCodeLoc(offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadInit();
+
+ CodeLocation() = nullptr;
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_impl_result_t offloadShutDown_val() {
+ if (true /*enableParameterValidation*/) {
+ }
+
+ return offloadShutDown_impl();
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadShutDown() {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadShutDown";
+ }
+
+ offload_result_t result = offloadShutDown_val();
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "()";
+ std::cout << "-> " << result << "\n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
+ }
+ }
+ return result;
+}
+offload_result_t
+offloadShutDownWithCodeLoc(offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadShutDown();
+
+ CodeLocation() = nullptr;
+ return result;
+}
+
///////////////////////////////////////////////////////////////////////////////
offload_impl_result_t
offloadPlatformGet_val(uint32_t NumEntries,
- offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms) {
+ offload_platform_handle_t *phPlatforms) {
if (true /*enableParameterValidation*/) {
- if (NumEntries == 0 && phPlatforms != NULL) {
+ if (NumEntries == 0) {
return OFFLOAD_ERRC_INVALID_SIZE;
}
+
+ if (NULL == phPlatforms) {
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ }
}
- return offloadPlatformGet_impl(NumEntries, phPlatforms, pNumPlatforms);
+ return offloadPlatformGet_impl(NumEntries, phPlatforms);
}
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
-offloadPlatformGet(uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms) {
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGet(
+ uint32_t NumEntries, offload_platform_handle_t *phPlatforms) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadPlatformGet";
}
- offload_result_t result =
- offloadPlatformGet_val(NumEntries, phPlatforms, pNumPlatforms);
+ offload_result_t result = offloadPlatformGet_val(NumEntries, phPlatforms);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_platform_get_params_t Params = {&NumEntries, &phPlatforms,
- &pNumPlatforms};
+ offload_platform_get_params_t Params = {&NumEntries, &phPlatforms};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result && result->details) {
@@ -40,12 +104,50 @@ offloadPlatformGet(uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
}
return result;
}
-offload_result_t offloadPlatformGetWithCodeLoc(
- uint32_t NumEntries, offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms, offload_code_location_t *pCodeLocation) {
+offload_result_t
+offloadPlatformGetWithCodeLoc(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms,
+ offload_code_location_t *pCodeLocation) {
CodeLocation() = pCodeLocation;
- offload_result_t result =
- offloadPlatformGet(NumEntries, phPlatforms, pNumPlatforms);
+ offload_result_t result = offloadPlatformGet(NumEntries, phPlatforms);
+
+ CodeLocation() = nullptr;
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_impl_result_t offloadPlatformGetCount_val(uint32_t *pNumPlatforms) {
+ if (true /*enableParameterValidation*/) {
+ if (NULL == pNumPlatforms) {
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ }
+ }
+
+ return offloadPlatformGetCount_impl(pNumPlatforms);
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadPlatformGetCount(uint32_t *pNumPlatforms) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadPlatformGetCount";
+ }
+
+ offload_result_t result = offloadPlatformGetCount_val(pNumPlatforms);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_platform_get_count_params_t Params = {&pNumPlatforms};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
+ }
+ }
+ return result;
+}
+offload_result_t
+offloadPlatformGetCountWithCodeLoc(uint32_t *pNumPlatforms,
+ offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadPlatformGetCount(pNumPlatforms);
CodeLocation() = nullptr;
return result;
@@ -55,41 +157,85 @@ offload_result_t offloadPlatformGetWithCodeLoc(
offload_impl_result_t
offloadPlatformGetInfo_val(offload_platform_handle_t hPlatform,
offload_platform_info_t propName, size_t propSize,
- void *pPropValue, size_t *pPropSizeRet) {
+ void *pPropValue) {
if (true /*enableParameterValidation*/) {
- if (propSize == 0 && pPropValue != NULL) {
+ if (propSize == 0) {
return OFFLOAD_ERRC_INVALID_SIZE;
}
- if (propSize != 0 && pPropValue == NULL) {
- return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ if (NULL == hPlatform) {
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
- if (pPropValue == NULL && pPropSizeRet == NULL) {
+ if (NULL == pPropValue) {
return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
+ }
+
+ return offloadPlatformGetInfo_impl(hPlatform, propName, propSize, pPropValue);
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName,
+ size_t propSize, void *pPropValue) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadPlatformGetInfo";
+ }
+
+ offload_result_t result =
+ offloadPlatformGetInfo_val(hPlatform, propName, propSize, pPropValue);
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_platform_get_info_params_t Params = {&hPlatform, &propName,
+ &propSize, &pPropValue};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
+ }
+ }
+ return result;
+}
+offload_result_t offloadPlatformGetInfoWithCodeLoc(
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName,
+ size_t propSize, void *pPropValue, offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result =
+ offloadPlatformGetInfo(hPlatform, propName, propSize, pPropValue);
+
+ CodeLocation() = nullptr;
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_impl_result_t
+offloadPlatformGetInfoSize_val(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName,
+ size_t *pPropSizeRet) {
+ if (true /*enableParameterValidation*/) {
if (NULL == hPlatform) {
return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
+
+ if (NULL == pPropSizeRet) {
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ }
}
- return offloadPlatformGetInfo_impl(hPlatform, propName, propSize, pPropValue,
- pPropSizeRet);
+ return offloadPlatformGetInfoSize_impl(hPlatform, propName, pPropSizeRet);
}
-OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfoSize(
offload_platform_handle_t hPlatform, offload_platform_info_t propName,
- size_t propSize, void *pPropValue, size_t *pPropSizeRet) {
+ size_t *pPropSizeRet) {
if (std::getenv("OFFLOAD_TRACE")) {
- std::cout << "---> offloadPlatformGetInfo";
+ std::cout << "---> offloadPlatformGetInfoSize";
}
- offload_result_t result = offloadPlatformGetInfo_val(
- hPlatform, propName, propSize, pPropValue, pPropSizeRet);
+ offload_result_t result =
+ offloadPlatformGetInfoSize_val(hPlatform, propName, pPropSizeRet);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_platform_get_info_params_t Params = {
- &hPlatform, &propName, &propSize, &pPropValue, &pPropSizeRet};
+ offload_platform_get_info_size_params_t Params = {&hPlatform, &propName,
+ &pPropSizeRet};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result && result->details) {
@@ -98,55 +244,94 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadPlatformGetInfo(
}
return result;
}
-offload_result_t offloadPlatformGetInfoWithCodeLoc(
+offload_result_t offloadPlatformGetInfoSizeWithCodeLoc(
offload_platform_handle_t hPlatform, offload_platform_info_t propName,
- size_t propSize, void *pPropValue, size_t *pPropSizeRet,
- offload_code_location_t *pCodeLocation) {
+ size_t *pPropSizeRet, offload_code_location_t *pCodeLocation) {
CodeLocation() = pCodeLocation;
- offload_result_t result = offloadPlatformGetInfo(
- hPlatform, propName, propSize, pPropValue, pPropSizeRet);
+ offload_result_t result =
+ offloadPlatformGetInfoSize(hPlatform, propName, pPropSizeRet);
CodeLocation() = nullptr;
return result;
}
///////////////////////////////////////////////////////////////////////////////
-offload_impl_result_t offloadDeviceGet_val(offload_platform_handle_t hPlatform,
- offload_device_type_t DeviceType,
- uint32_t NumEntries,
- offload_device_handle_t *phDevices,
- uint32_t *pNumDevices) {
+offload_impl_result_t
+offloadDeviceGetCount_val(offload_platform_handle_t hPlatform,
+ uint32_t *pNumDevices) {
if (true /*enableParameterValidation*/) {
- if (NumEntries == 0 && phDevices != NULL) {
- return OFFLOAD_ERRC_INVALID_SIZE;
+ if (NULL == hPlatform) {
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
- if (NumEntries > 0 && phDevices == NULL) {
+ if (NULL == pNumDevices) {
return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
+ }
+
+ return offloadDeviceGetCount_impl(hPlatform, pNumDevices);
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetCount(
+ offload_platform_handle_t hPlatform, uint32_t *pNumDevices) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadDeviceGetCount";
+ }
+
+ offload_result_t result = offloadDeviceGetCount_val(hPlatform, pNumDevices);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_device_get_count_params_t Params = {&hPlatform, &pNumDevices};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
+ }
+ }
+ return result;
+}
+offload_result_t
+offloadDeviceGetCountWithCodeLoc(offload_platform_handle_t hPlatform,
+ uint32_t *pNumDevices,
+ offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result = offloadDeviceGetCount(hPlatform, pNumDevices);
+
+ CodeLocation() = nullptr;
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_impl_result_t offloadDeviceGet_val(offload_platform_handle_t hPlatform,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices) {
+ if (true /*enableParameterValidation*/) {
+ if (NumEntries == 0) {
+ return OFFLOAD_ERRC_INVALID_SIZE;
+ }
if (NULL == hPlatform) {
return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
+
+ if (NULL == phDevices) {
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ }
}
- return offloadDeviceGet_impl(hPlatform, DeviceType, NumEntries, phDevices,
- pNumDevices);
+ return offloadDeviceGet_impl(hPlatform, NumEntries, phDevices);
}
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
-offloadDeviceGet(offload_platform_handle_t hPlatform,
- offload_device_type_t DeviceType, uint32_t NumEntries,
- offload_device_handle_t *phDevices, uint32_t *pNumDevices) {
+offloadDeviceGet(offload_platform_handle_t hPlatform, uint32_t NumEntries,
+ offload_device_handle_t *phDevices) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadDeviceGet";
}
- offload_result_t result = offloadDeviceGet_val(
- hPlatform, DeviceType, NumEntries, phDevices, pNumDevices);
+ offload_result_t result =
+ offloadDeviceGet_val(hPlatform, NumEntries, phDevices);
if (std::getenv("OFFLOAD_TRACE")) {
- offload_device_get_params_t Params = {&hPlatform, &DeviceType, &NumEntries,
- &phDevices, &pNumDevices};
+ offload_device_get_params_t Params = {&hPlatform, &NumEntries, &phDevices};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result && result->details) {
@@ -155,13 +340,13 @@ offloadDeviceGet(offload_platform_handle_t hPlatform,
}
return result;
}
-offload_result_t offloadDeviceGetWithCodeLoc(
- offload_platform_handle_t hPlatform, offload_device_type_t DeviceType,
- uint32_t NumEntries, offload_device_handle_t *phDevices,
- uint32_t *pNumDevices, offload_code_location_t *pCodeLocation) {
+offload_result_t
+offloadDeviceGetWithCodeLoc(offload_platform_handle_t hPlatform,
+ uint32_t NumEntries,
+ offload_device_handle_t *phDevices,
+ offload_code_location_t *pCodeLocation) {
CodeLocation() = pCodeLocation;
- offload_result_t result = offloadDeviceGet(hPlatform, DeviceType, NumEntries,
- phDevices, pNumDevices);
+ offload_result_t result = offloadDeviceGet(hPlatform, NumEntries, phDevices);
CodeLocation() = nullptr;
return result;
@@ -171,42 +356,36 @@ offload_result_t offloadDeviceGetWithCodeLoc(
offload_impl_result_t offloadDeviceGetInfo_val(offload_device_handle_t hDevice,
offload_device_info_t propName,
size_t propSize,
- void *pPropValue,
- size_t *pPropSizeRet) {
+ void *pPropValue) {
if (true /*enableParameterValidation*/) {
- if (propSize == 0 && pPropValue != NULL) {
+ if (propSize == 0) {
return OFFLOAD_ERRC_INVALID_SIZE;
}
- if (propSize != 0 && pPropValue == NULL) {
- return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ if (NULL == hDevice) {
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
}
- if (pPropValue == NULL && pPropSizeRet == NULL) {
+ if (NULL == pPropValue) {
return OFFLOAD_ERRC_INVALID_NULL_POINTER;
}
-
- if (NULL == hDevice) {
- return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
- }
}
- return offloadDeviceGetInfo_impl(hDevice, propName, propSize, pPropValue,
- pPropSizeRet);
+ return offloadDeviceGetInfo_impl(hDevice, propName, propSize, pPropValue);
}
OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
offload_device_handle_t hDevice, offload_device_info_t propName,
- size_t propSize, void *pPropValue, size_t *pPropSizeRet) {
+ size_t propSize, void *pPropValue) {
if (std::getenv("OFFLOAD_TRACE")) {
std::cout << "---> offloadDeviceGetInfo";
}
- offload_result_t result = offloadDeviceGetInfo_val(
- hDevice, propName, propSize, pPropValue, pPropSizeRet);
+ offload_result_t result =
+ offloadDeviceGetInfo_val(hDevice, propName, propSize, pPropValue);
if (std::getenv("OFFLOAD_TRACE")) {
offload_device_get_info_params_t Params = {&hDevice, &propName, &propSize,
- &pPropValue, &pPropSizeRet};
+ &pPropValue};
std::cout << "(" << &Params << ")";
std::cout << "-> " << result << "\n";
if (result && result->details) {
@@ -215,14 +394,61 @@ OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL offloadDeviceGetInfo(
}
return result;
}
-offload_result_t
-offloadDeviceGetInfoWithCodeLoc(offload_device_handle_t hDevice,
- offload_device_info_t propName, size_t propSize,
- void *pPropValue, size_t *pPropSizeRet,
- offload_code_location_t *pCodeLocation) {
+offload_result_t offloadDeviceGetInfoWithCodeLoc(
+ offload_device_handle_t hDevice, offload_device_info_t propName,
+ size_t propSize, void *pPropValue, offload_code_location_t *pCodeLocation) {
+ CodeLocation() = pCodeLocation;
+ offload_result_t result =
+ offloadDeviceGetInfo(hDevice, propName, propSize, pPropValue);
+
+ CodeLocation() = nullptr;
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+offload_impl_result_t
+offloadDeviceGetInfoSize_val(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t *pPropSizeRet) {
+ if (true /*enableParameterValidation*/) {
+ if (NULL == hDevice) {
+ return OFFLOAD_ERRC_INVALID_NULL_HANDLE;
+ }
+
+ if (NULL == pPropSizeRet) {
+ return OFFLOAD_ERRC_INVALID_NULL_POINTER;
+ }
+ }
+
+ return offloadDeviceGetInfoSize_impl(hDevice, propName, pPropSizeRet);
+}
+OFFLOAD_APIEXPORT offload_result_t OFFLOAD_APICALL
+offloadDeviceGetInfoSize(offload_device_handle_t hDevice,
+ offload_device_info_t propName, size_t *pPropSizeRet) {
+ if (std::getenv("OFFLOAD_TRACE")) {
+ std::cout << "---> offloadDeviceGetInfoSize";
+ }
+
+ offload_result_t result =
+ offloadDeviceGetInfoSize_val(hDevice, propName, pPropSizeRet);
+
+ if (std::getenv("OFFLOAD_TRACE")) {
+ offload_device_get_info_size_params_t Params = {&hDevice, &propName,
+ &pPropSizeRet};
+ std::cout << "(" << &Params << ")";
+ std::cout << "-> " << result << "\n";
+ if (result && result->details) {
+ std::cout << " *Error Details* " << result->details << " \n";
+ }
+ }
+ return result;
+}
+offload_result_t offloadDeviceGetInfoSizeWithCodeLoc(
+ offload_device_handle_t hDevice, offload_device_info_t propName,
+ size_t *pPropSizeRet, offload_code_location_t *pCodeLocation) {
CodeLocation() = pCodeLocation;
- offload_result_t result = offloadDeviceGetInfo(hDevice, propName, propSize,
- pPropValue, pPropSizeRet);
+ offload_result_t result =
+ offloadDeviceGetInfoSize(hDevice, propName, pPropSizeRet);
CodeLocation() = nullptr;
return result;
diff --git a/offload/new-api/include/offload_exports b/offload/new-api/include/offload_exports
index cb256fd34770d5..3a2e9adf1340c6 100644
--- a/offload/new-api/include/offload_exports
+++ b/offload/new-api/include/offload_exports
@@ -1,9 +1,15 @@
VERS1.0 {
global:
+ offloadInit;
+ offloadShutDown;
offloadPlatformGet;
+ offloadPlatformGetCount;
offloadPlatformGetInfo;
+ offloadPlatformGetInfoSize;
+ offloadDeviceGetCount;
offloadDeviceGet;
offloadDeviceGetInfo;
+ offloadDeviceGetInfoSize;
local:
*;
};
diff --git a/offload/new-api/include/offload_funcs.inc b/offload/new-api/include/offload_funcs.inc
index 14e878b9ed5a27..2b062b9f69cb4e 100644
--- a/offload/new-api/include/offload_funcs.inc
+++ b/offload/new-api/include/offload_funcs.inc
@@ -10,9 +10,15 @@
#error Please define the macro OFFLOAD_FUNC(Function)
#endif
+OFFLOAD_FUNC(offloadInit)
+OFFLOAD_FUNC(offloadShutDown)
OFFLOAD_FUNC(offloadPlatformGet)
+OFFLOAD_FUNC(offloadPlatformGetCount)
OFFLOAD_FUNC(offloadPlatformGetInfo)
+OFFLOAD_FUNC(offloadPlatformGetInfoSize)
+OFFLOAD_FUNC(offloadDeviceGetCount)
OFFLOAD_FUNC(offloadDeviceGet)
OFFLOAD_FUNC(offloadDeviceGetInfo)
+OFFLOAD_FUNC(offloadDeviceGetInfoSize)
#undef OFFLOAD_FUNC
diff --git a/offload/new-api/include/offload_impl.hpp b/offload/new-api/include/offload_impl.hpp
index b0bdf6c2221ae4..b7be0f0782e935 100644
--- a/offload/new-api/include/offload_impl.hpp
+++ b/offload/new-api/include/offload_impl.hpp
@@ -61,7 +61,7 @@ using ErrSetT = std::unordered_set<ErrPtrT, ErrPtrHash, ErrPtrEqual>;
ErrSetT &Errors();
struct offload_impl_result_t {
- offload_impl_result_t(std::nullptr_t) : Result(OFFLOAD_SUCCESS) {}
+ offload_impl_result_t(std::nullptr_t) : Result(OFFLOAD_RESULT_SUCCESS) {}
offload_impl_result_t(offload_errc_t Code) {
if (Code == OFFLOAD_ERRC_SUCCESS) {
Result = nullptr;
diff --git a/offload/new-api/include/offload_impl_func_decls.inc b/offload/new-api/include/offload_impl_func_decls.inc
index 1160ee1c6e2549..4dfedd0cde26b2 100644
--- a/offload/new-api/include/offload_impl_func_decls.inc
+++ b/offload/new-api/include/offload_impl_func_decls.inc
@@ -5,24 +5,40 @@
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
+offload_impl_result_t offloadInit_impl();
+
+offload_impl_result_t offloadShutDown_impl();
+
offload_impl_result_t
offloadPlatformGet_impl(uint32_t NumEntries,
- offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms);
+ offload_platform_handle_t *phPlatforms);
+
+offload_impl_result_t offloadPlatformGetCount_impl(uint32_t *pNumPlatforms);
offload_impl_result_t
offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
offload_platform_info_t propName, size_t propSize,
- void *pPropValue, size_t *pPropSizeRet);
+ void *pPropValue);
+
+offload_impl_result_t
+offloadPlatformGetInfoSize_impl(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName,
+ size_t *pPropSizeRet);
+
+offload_impl_result_t
+offloadDeviceGetCount_impl(offload_platform_handle_t hPlatform,
+ uint32_t *pNumDevices);
offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
- offload_device_type_t DeviceType,
uint32_t NumEntries,
- offload_device_handle_t *phDevices,
- uint32_t *pNumDevices);
+ offload_device_handle_t *phDevices);
offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
offload_device_info_t propName,
size_t propSize,
- void *pPropValue,
- size_t *pPropSizeRet);
+ void *pPropValue);
+
+offload_impl_result_t
+offloadDeviceGetInfoSize_impl(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t *pPropSizeRet);
diff --git a/offload/new-api/include/offload_print.hpp b/offload/new-api/include/offload_print.hpp
index c3a6c0a702e5b7..35245fd9e736a4 100644
--- a/offload/new-api/include/offload_print.hpp
+++ b/offload/new-api/include/offload_print.hpp
@@ -283,7 +283,7 @@ inline void printTagged(std::ostream &os, const void *ptr,
inline std::ostream &operator<<(std::ostream &os,
const offload_error_struct_t *err) {
if (err == nullptr) {
- os << "OFFLOAD_SUCCESS";
+ os << "OFFLOAD_RESULT_SUCCESS";
} else {
os << err->code;
}
@@ -305,7 +305,12 @@ inline std::ostream &operator<<(
printPtr(os, (*params->pphPlatforms)[i]);
}
os << "}";
- os << ", ";
+ return os;
+}
+
+inline std::ostream &operator<<(
+ std::ostream &os,
+ [[maybe_unused]] const struct offload_platform_get_count_params_t *params) {
os << ".pNumPlatforms = ";
printPtr(os, *params->ppNumPlatforms);
return os;
@@ -325,21 +330,41 @@ inline std::ostream &operator<<(
os << ", ";
os << ".pPropValue = ";
printTagged(os, *params->ppPropValue, *params->ppropName, *params->ppropSize);
+ return os;
+}
+
+inline std::ostream &
+operator<<(std::ostream &os,
+ [[maybe_unused]] const struct offload_platform_get_info_size_params_t
+ *params) {
+ os << ".hPlatform = ";
+ printPtr(os, *params->phPlatform);
+ os << ", ";
+ os << ".propName = ";
+ os << *params->ppropName;
os << ", ";
os << ".pPropSizeRet = ";
printPtr(os, *params->ppPropSizeRet);
return os;
}
+inline std::ostream &operator<<(
+ std::ostream &os,
+ [[maybe_unused]] const struct offload_device_get_count_params_t *params) {
+ os << ".hPlatform = ";
+ printPtr(os, *params->phPlatform);
+ os << ", ";
+ os << ".pNumDevices = ";
+ printPtr(os, *params->ppNumDevices);
+ return os;
+}
+
inline std::ostream &
operator<<(std::ostream &os,
[[maybe_unused]] const struct offload_device_get_params_t *params) {
os << ".hPlatform = ";
printPtr(os, *params->phPlatform);
os << ", ";
- os << ".DeviceType = ";
- os << *params->pDeviceType;
- os << ", ";
os << ".NumEntries = ";
os << *params->pNumEntries;
os << ", ";
@@ -352,9 +377,6 @@ operator<<(std::ostream &os,
printPtr(os, (*params->pphDevices)[i]);
}
os << "}";
- os << ", ";
- os << ".pNumDevices = ";
- printPtr(os, *params->ppNumDevices);
return os;
}
@@ -372,6 +394,18 @@ inline std::ostream &operator<<(
os << ", ";
os << ".pPropValue = ";
printTagged(os, *params->ppPropValue, *params->ppropName, *params->ppropSize);
+ return os;
+}
+
+inline std::ostream &
+operator<<(std::ostream &os,
+ [[maybe_unused]] const struct offload_device_get_info_size_params_t
+ *params) {
+ os << ".hDevice = ";
+ printPtr(os, *params->phDevice);
+ os << ", ";
+ os << ".propName = ";
+ os << *params->ppropName;
os << ", ";
os << ".pPropSizeRet = ";
printPtr(os, *params->ppPropSizeRet);
@@ -400,5 +434,5 @@ inline offload_result_t printPtr(std::ostream &os, const T *ptr) {
os << ")";
}
- return OFFLOAD_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
}
diff --git a/offload/new-api/src/offload_impl.cpp b/offload/new-api/src/offload_impl.cpp
index 31329d2aea683d..af26c28768b946 100644
--- a/offload/new-api/src/offload_impl.cpp
+++ b/offload/new-api/src/offload_impl.cpp
@@ -67,40 +67,44 @@ void initPlugins() {
}
}
-offload_impl_result_t
-offloadPlatformGet_impl(uint32_t NumEntries,
- offload_platform_handle_t *phPlatforms,
- uint32_t *pNumPlatforms) {
+// TODO: We can properly reference count here and manage the resources in a more
+// clever way
+offload_impl_result_t offloadInit_impl() {
+ static std::once_flag InitFlag;
+ std::call_once(InitFlag, initPlugins);
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+offload_impl_result_t offloadShutDown_impl() { return OFFLOAD_RESULT_SUCCESS; }
+
+offload_impl_result_t offloadPlatformGetCount_impl(uint32_t *pNumPlatforms) {
// It is expected that offloadPlatformGet is the first function to be called.
// In future it may make sense to have a specific entry point for Offload
// initialization, or expose explicit initialization of plugins.
- static std::once_flag InitFlag;
- std::call_once(InitFlag, initPlugins);
+ *pNumPlatforms = Platforms().size();
+ return OFFLOAD_RESULT_SUCCESS;
+}
+offload_impl_result_t
+offloadPlatformGet_impl(uint32_t NumEntries,
+ offload_platform_handle_t *phPlatforms) {
if (NumEntries > Platforms().size()) {
return {OFFLOAD_ERRC_INVALID_SIZE,
std::string{formatv("{0} platform(s) available but {1} requested.",
Platforms().size(), NumEntries)}};
}
- if (phPlatforms) {
- for (uint32_t PlatformIndex = 0; PlatformIndex < NumEntries;
- PlatformIndex++) {
- phPlatforms[PlatformIndex] = &(Platforms())[PlatformIndex];
- }
- }
-
- if (pNumPlatforms) {
- *pNumPlatforms = Platforms().size();
+ for (uint32_t PlatformIndex = 0; PlatformIndex < NumEntries;
+ PlatformIndex++) {
+ phPlatforms[PlatformIndex] = &(Platforms())[PlatformIndex];
}
- return OFFLOAD_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
}
-offload_impl_result_t
-offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
- offload_platform_info_t propName, size_t propSize,
- void *pPropValue, size_t *pPropSizeRet) {
+offload_impl_result_t offloadPlatformGetInfoImplDetail(
+ offload_platform_handle_t hPlatform, offload_platform_info_t propName,
+ size_t propSize, void *pPropValue, size_t *pPropSizeRet) {
ReturnHelper ReturnValue(propSize, pPropValue, pPropSizeRet);
switch (propName) {
@@ -127,33 +131,51 @@ offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
return OFFLOAD_ERRC_INVALID_ENUMERATION;
}
- return OFFLOAD_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
}
-offload_impl_result_t offloadDeviceGet_impl(offload_platform_handle_t hPlatform,
- offload_device_type_t,
- uint32_t NumEntries,
- offload_device_handle_t *phDevices,
- uint32_t *pNumDevices) {
+offload_impl_result_t
+offloadPlatformGetInfo_impl(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName, size_t propSize,
+ void *pPropValue) {
+ return offloadPlatformGetInfoImplDetail(hPlatform, propName, propSize,
+ pPropValue, nullptr);
+}
- if (phDevices) {
- for (uint32_t DeviceIndex = 0; DeviceIndex < NumEntries; DeviceIndex++) {
- phDevices[DeviceIndex] = &(hPlatform->Devices[DeviceIndex]);
- }
+offload_impl_result_t
+offloadPlatformGetInfoSize_impl(offload_platform_handle_t hPlatform,
+ offload_platform_info_t propName,
+ size_t *pPropSizeRet) {
+ return offloadPlatformGetInfoImplDetail(hPlatform, propName, 0, nullptr,
+ pPropSizeRet);
+}
+
+offload_impl_result_t
+offloadDeviceGetCount_impl(offload_platform_handle_t hPlatform,
+ uint32_t *pNumDevices) {
+ *pNumDevices = static_cast<uint32_t>(hPlatform->Devices.size());
+
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+offload_impl_result_t
+offloadDeviceGet_impl(offload_platform_handle_t hPlatform, uint32_t NumEntries,
+ offload_device_handle_t *phDevices) {
+ if (NumEntries > hPlatform->Devices.size()) {
+ return OFFLOAD_ERRC_INVALID_SIZE;
}
- if (pNumDevices) {
- *pNumDevices = static_cast<uint32_t>(hPlatform->Devices.size());
+ for (uint32_t DeviceIndex = 0; DeviceIndex < NumEntries; DeviceIndex++) {
+ phDevices[DeviceIndex] = &(hPlatform->Devices[DeviceIndex]);
}
- return OFFLOAD_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
}
-offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
- offload_device_info_t propName,
- size_t propSize,
- void *pPropValue,
- size_t *pPropSizeRet) {
+offload_impl_result_t
+offloadDeviceGetInfoImplDetail(offload_device_handle_t hDevice,
+ offload_device_info_t propName, size_t propSize,
+ void *pPropValue, size_t *pPropSizeRet) {
ReturnHelper ReturnValue(propSize, pPropValue, pPropSizeRet);
@@ -194,5 +216,21 @@ offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
return OFFLOAD_ERRC_INVALID_ENUMERATION;
}
- return OFFLOAD_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
+}
+
+offload_impl_result_t offloadDeviceGetInfo_impl(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t propSize,
+ void *pPropValue) {
+ return offloadDeviceGetInfoImplDetail(hDevice, propName, propSize, pPropValue,
+ nullptr);
+}
+
+offload_impl_result_t
+offloadDeviceGetInfoSize_impl(offload_device_handle_t hDevice,
+ offload_device_info_t propName,
+ size_t *pPropSizeRet) {
+ return offloadDeviceGetInfoImplDetail(hDevice, propName, 0, nullptr,
+ pPropSizeRet);
}
diff --git a/offload/tools/offload-tblgen/APIGen.cpp b/offload/tools/offload-tblgen/APIGen.cpp
index cd9fe54085f1a8..c50146f351288e 100644
--- a/offload/tools/offload-tblgen/APIGen.cpp
+++ b/offload/tools/offload-tblgen/APIGen.cpp
@@ -153,6 +153,10 @@ static void ProcessStruct(const StructRec &Struct, raw_ostream &OS) {
}
static void ProcessFuncParamStruct(const FunctionRec &Func, raw_ostream &OS) {
+ if (Func.getParams().size() == 0) {
+ return;
+ }
+
auto FuncParamStructBegin = R"(
///////////////////////////////////////////////////////////////////////////////
/// @brief Function parameters for {0}
diff --git a/offload/tools/offload-tblgen/EntryPointGen.cpp b/offload/tools/offload-tblgen/EntryPointGen.cpp
index c5c823f68cb56e..16adcdce1e07e2 100644
--- a/offload/tools/offload-tblgen/EntryPointGen.cpp
+++ b/offload/tools/offload-tblgen/EntryPointGen.cpp
@@ -82,15 +82,19 @@ static void EmitEntryPointFunc(const FunctionRec &F, raw_ostream &OS) {
// Emit post-call prints
OS << TAB_1 "if (std::getenv(\"OFFLOAD_TRACE\")) {\n";
- OS << formatv(TAB_2 "{0} Params = {{ ", F.getParamStructName());
- for (const auto &Param : F.getParams()) {
- OS << "&" << Param.getName();
- if (Param != F.getParams().back()) {
- OS << ", ";
+ if (F.getParams().size() > 0) {
+ OS << formatv(TAB_2 "{0} Params = {{ ", F.getParamStructName());
+ for (const auto &Param : F.getParams()) {
+ OS << "&" << Param.getName();
+ if (Param != F.getParams().back()) {
+ OS << ", ";
+ }
}
+ OS << formatv("};\n");
+ OS << TAB_2 "std::cout << \"(\" << &Params << \")\";\n";
+ } else {
+ OS << TAB_2 "std::cout << \"()\";\n";
}
- OS << formatv("};\n");
- OS << TAB_2 "std::cout << \"(\" << &Params << \")\";\n";
OS << TAB_2 "std::cout << \"-> \" << result << \"\\n\";\n";
OS << TAB_2 "if (result && result->details) {\n";
OS << TAB_3 "std::cout << \" *Error Details* \" << result->details "
diff --git a/offload/tools/offload-tblgen/FuncsGen.cpp b/offload/tools/offload-tblgen/FuncsGen.cpp
index 033ed8db3240ad..e0152053c9e19c 100644
--- a/offload/tools/offload-tblgen/FuncsGen.cpp
+++ b/offload/tools/offload-tblgen/FuncsGen.cpp
@@ -54,22 +54,14 @@ void EmitOffloadImplFuncDecls(RecordKeeper &Records, raw_ostream &OS) {
OS << GenericHeader;
for (auto *R : Records.getAllDerivedDefinitions("Function")) {
FunctionRec F{R};
- // The error details function does not set error details itself, so don't
- // use the impl result type
- if (F.getName() == "offloadGetErrorDetails") {
- OS << formatv("{0}_result_t {1}_impl(", PrefixLower, F.getName());
- } else {
- OS << formatv("{0}_impl_result_t {1}_impl(", PrefixLower, F.getName());
- }
+ OS << formatv("{0}_impl_result_t {1}_impl(", PrefixLower, F.getName());
auto Params = F.getParams();
for (auto &Param : Params) {
OS << Param.getType() << " " << Param.getName();
if (Param != Params.back()) {
OS << ", ";
- } else {
- OS << ");";
}
}
- OS << "\n\n";
+ OS << ");\n\n";
}
}
diff --git a/offload/tools/offload-tblgen/PrintGen.cpp b/offload/tools/offload-tblgen/PrintGen.cpp
index 0e73c29c338a1b..ef9ecf0f3ac235 100644
--- a/offload/tools/offload-tblgen/PrintGen.cpp
+++ b/offload/tools/offload-tblgen/PrintGen.cpp
@@ -99,7 +99,7 @@ static void EmitResultPrint(raw_ostream &OS) {
inline std::ostream &operator<<(std::ostream &os,
const offload_error_struct_t *err) {
if (err == nullptr) {
- os << "OFFLOAD_SUCCESS";
+ os << "OFFLOAD_RESULT_SUCCESS";
} else {
os << err->code;
}
@@ -110,6 +110,10 @@ inline std::ostream &operator<<(std::ostream &os,
static void EmitFunctionParamStructPrint(const FunctionRec &Func,
raw_ostream &OS) {
+ if (Func.getParams().size() == 0) {
+ return;
+ }
+
OS << formatv(R"(
inline std::ostream &operator<<(std::ostream &os, [[maybe_unused]] const struct {0} *params) {{
)",
@@ -216,7 +220,7 @@ template <typename T> inline offload_result_t printPtr(std::ostream &os, const T
os << ")";
}
- return OFFLOAD_SUCCESS;
+ return OFFLOAD_RESULT_SUCCESS;
}
)""";
}
diff --git a/offload/unittests/OffloadAPI/CMakeLists.txt b/offload/unittests/OffloadAPI/CMakeLists.txt
index 0960fa5a135111..276eb2be281922 100644
--- a/offload/unittests/OffloadAPI/CMakeLists.txt
+++ b/offload/unittests/OffloadAPI/CMakeLists.txt
@@ -4,9 +4,13 @@ set(PLUGINS_TEST_INCLUDE ${LIBOMPTARGET_INCLUDE_DIR} ${CMAKE_CURRENT_SOURCE_DIR}
add_libompt_unittest("offload.unittests"
${CMAKE_CURRENT_SOURCE_DIR}/common/environment.cpp
${CMAKE_CURRENT_SOURCE_DIR}/platform/offloadPlatformGet.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/platform/offloadPlatformGetCount.cpp
${CMAKE_CURRENT_SOURCE_DIR}/platform/offloadPlatformGetInfo.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/platform/offloadPlatformGetInfoSize.cpp
${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGet.cpp
- ${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGetInfo.cpp)
+ ${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGetCount.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGetInfo.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/device/offloadDeviceGetInfoSize.cpp)
add_dependencies("offload.unittests" ${PLUGINS_TEST_COMMON})
target_link_libraries("offload.unittests" PRIVATE ${PLUGINS_TEST_COMMON})
target_include_directories("offload.unittests" PRIVATE ${PLUGINS_TEST_INCLUDE})
diff --git a/offload/unittests/OffloadAPI/common/environment.cpp b/offload/unittests/OffloadAPI/common/environment.cpp
index 01973ec117e300..ec71495d278552 100644
--- a/offload/unittests/OffloadAPI/common/environment.cpp
+++ b/offload/unittests/OffloadAPI/common/environment.cpp
@@ -13,6 +13,14 @@
using namespace llvm;
+// Wrapper so we don't have to constantly init and shutdown Offload in every
+// test, while having sensible lifetime for the platform environment
+struct OffloadInitWrapper {
+ OffloadInitWrapper() { offloadInit(); }
+ ~OffloadInitWrapper() { offloadShutDown(); }
+};
+static OffloadInitWrapper Wrapper{};
+
static cl::opt<std::string>
SelectedPlatform("platform", cl::desc("Only test the specified platform"),
cl::value_desc("platform"));
@@ -20,11 +28,10 @@ static cl::opt<std::string>
std::ostream &operator<<(std::ostream &Out,
const offload_platform_handle_t &Platform) {
size_t Size;
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_NAME, 0, nullptr,
- &Size);
+ offloadPlatformGetInfoSize(Platform, OFFLOAD_PLATFORM_INFO_NAME, &Size);
std::vector<char> Name(Size);
offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_NAME, Size,
- Name.data(), nullptr);
+ Name.data());
Out << Name.data();
return Out;
}
@@ -43,10 +50,10 @@ const std::vector<offload_platform_handle_t> &TestEnvironment::getPlatforms() {
if (Platforms.empty()) {
uint32_t PlatformCount = 0;
- offloadPlatformGet(0, nullptr, &PlatformCount);
+ offloadPlatformGetCount(&PlatformCount);
if (PlatformCount > 0) {
Platforms.resize(PlatformCount);
- offloadPlatformGet(PlatformCount, Platforms.data(), nullptr);
+ offloadPlatformGet(PlatformCount, Platforms.data());
}
}
@@ -77,8 +84,8 @@ offload_platform_handle_t TestEnvironment::getPlatform() {
Platform = Platforms[0];
for (auto CandidatePlatform : Platforms) {
uint32_t NumDevices = 0;
- if (offloadDeviceGet(CandidatePlatform, OFFLOAD_DEVICE_TYPE_ALL, 0,
- nullptr, &NumDevices) == OFFLOAD_SUCCESS) {
+ if (offloadDeviceGetCount(CandidatePlatform, &NumDevices) ==
+ OFFLOAD_RESULT_SUCCESS) {
if (NumDevices > 0) {
Platform = CandidatePlatform;
break;
diff --git a/offload/unittests/OffloadAPI/common/fixtures.hpp b/offload/unittests/OffloadAPI/common/fixtures.hpp
index 2366f8687a8e93..5dcf6232504bd9 100644
--- a/offload/unittests/OffloadAPI/common/fixtures.hpp
+++ b/offload/unittests/OffloadAPI/common/fixtures.hpp
@@ -15,7 +15,7 @@
#pragma once
#ifndef ASSERT_SUCCESS
-#define ASSERT_SUCCESS(ACTUAL) ASSERT_EQ(OFFLOAD_SUCCESS, ACTUAL)
+#define ASSERT_SUCCESS(ACTUAL) ASSERT_EQ(OFFLOAD_RESULT_SUCCESS, ACTUAL)
#endif
// TODO: rework this so the EXPECTED/ACTUAL results are readable
@@ -34,9 +34,13 @@
} \
(void)0
-struct offloadPlatformTest : ::testing::Test {
+struct offloadTest : ::testing::Test {
+ // No special behavior now, but just in case we need to override it in future
+};
+
+struct offloadPlatformTest : offloadTest {
void SetUp() override {
- RETURN_ON_FATAL_FAILURE(::testing::Test::SetUp());
+ RETURN_ON_FATAL_FAILURE(offloadTest::SetUp());
Platform = TestEnvironment::getPlatform();
ASSERT_NE(Platform, nullptr);
@@ -50,13 +54,11 @@ struct offloadDeviceTest : offloadPlatformTest {
RETURN_ON_FATAL_FAILURE(offloadPlatformTest::SetUp());
uint32_t NumDevices;
- ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 0,
- nullptr, &NumDevices));
+ ASSERT_SUCCESS(offloadDeviceGetCount(Platform, &NumDevices));
if (NumDevices == 0) {
GTEST_SKIP() << "No available devices on this platform.";
}
- ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 1,
- &Device, nullptr));
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, 1, &Device));
}
offload_device_handle_t Device;
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp
index 1442ebc98c7f6e..c14fd3997ee62d 100644
--- a/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGet.cpp
@@ -14,12 +14,10 @@ using offloadDeviceGetTest = offloadPlatformTest;
TEST_F(offloadDeviceGetTest, Success) {
uint32_t Count = 0;
- ASSERT_SUCCESS(
- offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 0, nullptr, &Count));
+ ASSERT_SUCCESS(offloadDeviceGetCount(Platform, &Count));
ASSERT_NE(Count, 0lu);
std::vector<offload_device_handle_t> Devices(Count);
- ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, Count,
- Devices.data(), nullptr));
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, Count, Devices.data()));
for (auto Device : Devices) {
ASSERT_NE(nullptr, Device);
}
@@ -27,14 +25,12 @@ TEST_F(offloadDeviceGetTest, Success) {
TEST_F(offloadDeviceGetTest, SuccessSubsetOfDevices) {
uint32_t Count;
- ASSERT_SUCCESS(
- offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, 0, nullptr, &Count));
+ ASSERT_SUCCESS(offloadDeviceGetCount(Platform, &Count));
if (Count < 2) {
GTEST_SKIP() << "Only one device is available on this platform.";
}
std::vector<offload_device_handle_t> Devices(Count - 1);
- ASSERT_SUCCESS(offloadDeviceGet(Platform, OFFLOAD_DEVICE_TYPE_ALL, Count - 1,
- Devices.data(), nullptr));
+ ASSERT_SUCCESS(offloadDeviceGet(Platform, Count - 1, Devices.data()));
for (auto Device : Devices) {
ASSERT_NE(nullptr, Device);
}
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGetCount.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGetCount.cpp
new file mode 100644
index 00000000000000..87ebbef1d8b7d8
--- /dev/null
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGetCount.cpp
@@ -0,0 +1,30 @@
+//===------- Offload API tests - offloadDeviceGetCount --------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include "../common/fixtures.hpp"
+#include <gtest/gtest.h>
+#include <offload_api.h>
+
+using offloadDeviceGetCountTest = offloadPlatformTest;
+
+TEST_F(offloadDeviceGetCountTest, Success) {
+ uint32_t Count = 0;
+ ASSERT_SUCCESS(offloadDeviceGetCount(Platform, &Count));
+ ASSERT_NE(Count, 0lu);
+}
+
+TEST_F(offloadDeviceGetCountTest, InvalidNullPlatform) {
+ uint32_t Count = 0;
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_HANDLE,
+ offloadDeviceGetCount(nullptr, &Count));
+}
+
+TEST_F(offloadDeviceGetCountTest, InvalidNullPointer) {
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadDeviceGetCount(Platform, nullptr));
+}
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
index e9f5615b35d3ef..099bc3abcf8c4b 100644
--- a/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfo.cpp
@@ -7,6 +7,7 @@
//===----------------------------------------------------------------------===//
#include "../common/fixtures.hpp"
+#include "offloadDeviceInfo.hpp"
#include <gtest/gtest.h>
#include <offload_api.h>
@@ -17,39 +18,22 @@ struct offloadDeviceGetInfoTest
void SetUp() override { RETURN_ON_FATAL_FAILURE(offloadDeviceTest::SetUp()); }
};
-// TODO: We could autogenerate the list of enum values
INSTANTIATE_TEST_SUITE_P(
- , offloadDeviceGetInfoTest,
- ::testing::Values(OFFLOAD_DEVICE_INFO_TYPE, OFFLOAD_DEVICE_INFO_PLATFORM,
- OFFLOAD_DEVICE_INFO_NAME, OFFLOAD_DEVICE_INFO_VENDOR,
- OFFLOAD_DEVICE_INFO_DRIVER_VERSION),
+ , offloadDeviceGetInfoTest, ::testing::ValuesIn(DeviceQueries),
[](const ::testing::TestParamInfo<offload_device_info_t> &info) {
std::stringstream ss;
ss << info.param;
return ss.str();
});
-// TODO: We could autogenerate this
-std::unordered_map<offload_device_info_t, size_t> DeviceInfoSizeMap = {
- {OFFLOAD_DEVICE_INFO_TYPE, sizeof(offload_device_type_t)},
- {OFFLOAD_DEVICE_INFO_PLATFORM, sizeof(offload_platform_handle_t)},
-};
-
TEST_P(offloadDeviceGetInfoTest, Success) {
offload_device_info_t InfoType = GetParam();
size_t Size = 0;
- ASSERT_SUCCESS(offloadDeviceGetInfo(Device, InfoType, 0, nullptr, &Size));
- auto ExpectedSize = DeviceInfoSizeMap.find(InfoType);
- if (ExpectedSize != DeviceInfoSizeMap.end()) {
- ASSERT_EQ(Size, ExpectedSize->second);
- } else {
- ASSERT_NE(Size, 0lu);
- }
+ ASSERT_SUCCESS(offloadDeviceGetInfoSize(Device, InfoType, &Size));
std::vector<char> InfoData(Size);
- ASSERT_SUCCESS(
- offloadDeviceGetInfo(Device, InfoType, Size, InfoData.data(), nullptr));
+ ASSERT_SUCCESS(offloadDeviceGetInfo(Device, InfoType, Size, InfoData.data()));
if (InfoType == OFFLOAD_DEVICE_INFO_PLATFORM) {
auto *ReturnedPlatform =
@@ -62,42 +46,35 @@ TEST_F(offloadDeviceGetInfoTest, InvalidNullHandleDevice) {
offload_device_type_t DeviceType;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_HANDLE,
offloadDeviceGetInfo(nullptr, OFFLOAD_DEVICE_INFO_TYPE,
- sizeof(offload_device_type_t), &DeviceType,
- nullptr));
+ sizeof(offload_device_type_t),
+ &DeviceType));
}
TEST_F(offloadDeviceGetInfoTest, InvalidEnumerationInfoType) {
offload_device_type_t DeviceType;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_ENUMERATION,
offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_FORCE_UINT32,
- sizeof(offload_device_type_t), &DeviceType,
- nullptr));
+ sizeof(offload_device_type_t),
+ &DeviceType));
}
TEST_F(offloadDeviceGetInfoTest, InvalidSizePropSize) {
offload_device_type_t DeviceType;
- ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0,
- &DeviceType, nullptr));
+ ASSERT_ERROR(
+ OFFLOAD_ERRC_INVALID_SIZE,
+ offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0, &DeviceType));
}
TEST_F(offloadDeviceGetInfoTest, InvalidSizePropSizeSmall) {
offload_device_type_t DeviceType;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
- sizeof(DeviceType) - 1, &DeviceType,
- nullptr));
+ sizeof(DeviceType) - 1, &DeviceType));
}
TEST_F(offloadDeviceGetInfoTest, InvalidNullPointerPropValue) {
offload_device_type_t DeviceType;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE,
- sizeof(DeviceType), nullptr, nullptr));
-}
-
-TEST_F(offloadDeviceGetInfoTest, InvalidNullPointerPropSizeRet) {
- ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
- offloadDeviceGetInfo(Device, OFFLOAD_DEVICE_INFO_TYPE, 0,
- nullptr, nullptr));
+ sizeof(DeviceType), nullptr));
}
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceGetInfoSize.cpp b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfoSize.cpp
new file mode 100644
index 00000000000000..8e08a59af902c2
--- /dev/null
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceGetInfoSize.cpp
@@ -0,0 +1,61 @@
+//===------- Offload API tests - offloadDeviceGetInfoSize -----------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include <offload_api.h>
+
+#include "../common/fixtures.hpp"
+#include "offloadDeviceInfo.hpp"
+
+struct offloadDeviceGetInfoSizeTest
+ : offloadDeviceTest,
+ ::testing::WithParamInterface<offload_device_info_t> {
+
+ void SetUp() override { RETURN_ON_FATAL_FAILURE(offloadDeviceTest::SetUp()); }
+};
+
+// TODO: We could autogenerate the list of enum values
+INSTANTIATE_TEST_SUITE_P(
+ , offloadDeviceGetInfoSizeTest, ::testing::ValuesIn(DeviceQueries),
+ [](const ::testing::TestParamInfo<offload_device_info_t> &info) {
+ std::stringstream ss;
+ ss << info.param;
+ return ss.str();
+ });
+
+TEST_P(offloadDeviceGetInfoSizeTest, Success) {
+ offload_device_info_t InfoType = GetParam();
+ size_t Size = 0;
+
+ ASSERT_SUCCESS(offloadDeviceGetInfoSize(Device, InfoType, &Size));
+ auto ExpectedSize = DeviceInfoSizeMap.find(InfoType);
+ if (ExpectedSize != DeviceInfoSizeMap.end()) {
+ ASSERT_EQ(Size, ExpectedSize->second);
+ } else {
+ ASSERT_NE(Size, 0lu);
+ }
+}
+
+TEST_F(offloadDeviceGetInfoSizeTest, InvalidNullHandle) {
+ size_t Size = 0;
+ ASSERT_ERROR(
+ OFFLOAD_ERRC_INVALID_NULL_HANDLE,
+ offloadDeviceGetInfoSize(nullptr, OFFLOAD_DEVICE_INFO_TYPE, &Size));
+}
+
+TEST_F(offloadDeviceGetInfoSizeTest, InvalidDeviceInfoEnumeration) {
+ size_t Size = 0;
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_ENUMERATION,
+ offloadDeviceGetInfoSize(
+ Device, OFFLOAD_DEVICE_INFO_FORCE_UINT32, &Size));
+}
+
+TEST_F(offloadDeviceGetInfoSizeTest, InvalidNullPointer) {
+ ASSERT_ERROR(
+ OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadDeviceGetInfoSize(Device, OFFLOAD_DEVICE_INFO_TYPE, nullptr));
+}
diff --git a/offload/unittests/OffloadAPI/device/offloadDeviceInfo.hpp b/offload/unittests/OffloadAPI/device/offloadDeviceInfo.hpp
new file mode 100644
index 00000000000000..064935ac10107f
--- /dev/null
+++ b/offload/unittests/OffloadAPI/device/offloadDeviceInfo.hpp
@@ -0,0 +1,22 @@
+//===------- Offload API tests - Helpers for device info query testing ----===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+#pragma once
+
+#include <unordered_map>
+#include <vector>
+
+// TODO: We could autogenerate these
+inline std::vector<offload_device_info_t> DeviceQueries = {
+ OFFLOAD_DEVICE_INFO_TYPE, OFFLOAD_DEVICE_INFO_PLATFORM,
+ OFFLOAD_DEVICE_INFO_NAME, OFFLOAD_DEVICE_INFO_VENDOR,
+ OFFLOAD_DEVICE_INFO_DRIVER_VERSION};
+
+inline std::unordered_map<offload_device_info_t, size_t> DeviceInfoSizeMap = {
+ {OFFLOAD_DEVICE_INFO_TYPE, sizeof(offload_device_type_t)},
+ {OFFLOAD_DEVICE_INFO_PLATFORM, sizeof(offload_platform_handle_t)},
+};
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
index 14b7b9bba6b4d4..37d4b8c62b5ee1 100644
--- a/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGet.cpp
@@ -6,24 +6,24 @@
//
//===----------------------------------------------------------------------===//
+#include "../common/fixtures.hpp"
#include <gtest/gtest.h>
#include <offload_api.h>
-#include "../common/fixtures.hpp"
-using offloadPlatformGetTest = ::testing::Test;
+using offloadPlatformGetTest = offloadTest;
TEST_F(offloadPlatformGetTest, Success) {
uint32_t PlatformCount;
- ASSERT_SUCCESS(offloadPlatformGet(0, nullptr, &PlatformCount));
+ ASSERT_SUCCESS(offloadPlatformGetCount(&PlatformCount));
std::vector<offload_platform_handle_t> Platforms(PlatformCount);
- ASSERT_SUCCESS(offloadPlatformGet(PlatformCount, Platforms.data(), nullptr));
+ ASSERT_SUCCESS(offloadPlatformGet(PlatformCount, Platforms.data()));
}
TEST_F(offloadPlatformGetTest, InvalidNumEntries) {
uint32_t PlatformCount;
- ASSERT_SUCCESS(offloadPlatformGet(0, nullptr, &PlatformCount));
+ ASSERT_SUCCESS(offloadPlatformGetCount(&PlatformCount));
std::vector<offload_platform_handle_t> Platforms(PlatformCount);
ASSERT_ERROR(
OFFLOAD_ERRC_INVALID_SIZE,
- offloadPlatformGet(PlatformCount + 1, Platforms.data(), nullptr));
+ offloadPlatformGet(PlatformCount + 1, Platforms.data()));
}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGetCount.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGetCount.cpp
new file mode 100644
index 00000000000000..94e2f4cad0e5dc
--- /dev/null
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGetCount.cpp
@@ -0,0 +1,23 @@
+//===------- Offload API tests - offloadPlatformGetCount ------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include "../common/fixtures.hpp"
+#include <gtest/gtest.h>
+#include <offload_api.h>
+
+using offloadPlatformGetCountTest = offloadTest;
+
+TEST_F(offloadPlatformGetCountTest, Success) {
+ uint32_t PlatformCount;
+ ASSERT_SUCCESS(offloadPlatformGetCount(&PlatformCount));
+}
+
+TEST_F(offloadPlatformGetCountTest, InvalidNullPointer) {
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadPlatformGetCount(nullptr));
+}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
index a5c86a7ba1933f..2522b3bf56f77c 100644
--- a/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfo.cpp
@@ -9,90 +9,70 @@
#include <offload_api.h>
#include "../common/fixtures.hpp"
+#include "offloadPlatformInfo.hpp"
struct offloadPlatformGetInfoTest
: offloadPlatformTest,
::testing::WithParamInterface<offload_platform_info_t> {};
-// TODO: We could autogenerate the list of enum values
INSTANTIATE_TEST_SUITE_P(
offloadPlatformGetInfo, offloadPlatformGetInfoTest,
- ::testing::Values(OFFLOAD_PLATFORM_INFO_NAME,
- OFFLOAD_PLATFORM_INFO_VENDOR_NAME,
- OFFLOAD_PLATFORM_INFO_VERSION,
- OFFLOAD_PLATFORM_INFO_BACKEND),
+ ::testing::ValuesIn(PlatformQueries),
[](const ::testing::TestParamInfo<offload_platform_info_t> &info) {
std::stringstream ss;
ss << info.param;
return ss.str();
});
-// TODO: We could autogenerate this
-std::unordered_map<offload_platform_info_t, size_t> PlatformInfoSizeMap = {
- {OFFLOAD_PLATFORM_INFO_BACKEND, sizeof(offload_platform_backend_t)},
-};
-
TEST_P(offloadPlatformGetInfoTest, Success) {
size_t Size = 0;
offload_platform_info_t InfoType = GetParam();
- ASSERT_SUCCESS(offloadPlatformGetInfo(Platform, InfoType, 0, nullptr, &Size));
- auto ExpectedSize = PlatformInfoSizeMap.find(InfoType);
- if (ExpectedSize != PlatformInfoSizeMap.end()) {
- ASSERT_EQ(Size, ExpectedSize->second);
- } else {
- ASSERT_NE(Size, 0lu);
- }
-
+ ASSERT_SUCCESS(offloadPlatformGetInfoSize(Platform, InfoType, &Size));
std::vector<char> InfoData(Size);
- ASSERT_SUCCESS(offloadPlatformGetInfo(Platform, InfoType, Size,
- InfoData.data(), nullptr));
+ ASSERT_SUCCESS(
+ offloadPlatformGetInfo(Platform, InfoType, Size, InfoData.data()));
// Info types with a dynamic size are all char[] so we can verify the returned
// string is the expected size.
+ auto ExpectedSize = PlatformInfoSizeMap.find(InfoType);
if (ExpectedSize == PlatformInfoSizeMap.end()) {
ASSERT_EQ(Size, strlen(InfoData.data()) + 1);
}
}
TEST_F(offloadPlatformGetInfoTest, InvalidNullHandle) {
- size_t Size = 0;
+ offload_platform_backend_t Backend;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_HANDLE,
- offloadPlatformGetInfo(nullptr, OFFLOAD_PLATFORM_INFO_BACKEND, 0,
- nullptr, &Size));
+ offloadPlatformGetInfo(nullptr, OFFLOAD_PLATFORM_INFO_BACKEND,
+ sizeof(Backend), &Backend));
}
TEST_F(offloadPlatformGetInfoTest, InvalidPlatformInfoEnumeration) {
- size_t Size = 0;
+ offload_platform_backend_t Backend;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_ENUMERATION,
offloadPlatformGetInfo(Platform,
- OFFLOAD_PLATFORM_INFO_FORCE_UINT32, 0,
- nullptr, &Size));
+ OFFLOAD_PLATFORM_INFO_FORCE_UINT32,
+ sizeof(Backend), &Backend));
}
TEST_F(offloadPlatformGetInfoTest, InvalidSizeZero) {
offload_platform_backend_t Backend;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
- 0, &Backend, nullptr));
+ 0, &Backend));
}
TEST_F(offloadPlatformGetInfoTest, InvalidSizeSmall) {
offload_platform_backend_t Backend;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_SIZE,
offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
- sizeof(Backend) - 1, &Backend, nullptr));
+ sizeof(Backend) - 1, &Backend));
}
TEST_F(offloadPlatformGetInfoTest, InvalidNullPointerPropValue) {
offload_platform_backend_t Backend;
ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
- sizeof(Backend), nullptr, nullptr));
-}
-
-TEST_F(offloadPlatformGetInfoTest, InvalidNullPointerPropSizeRet) {
- ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
- offloadPlatformGetInfo(Platform, OFFLOAD_PLATFORM_INFO_BACKEND,
- 0, nullptr, nullptr));
+ sizeof(Backend), nullptr));
}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfoSize.cpp b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfoSize.cpp
new file mode 100644
index 00000000000000..e90340666d1312
--- /dev/null
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformGetInfoSize.cpp
@@ -0,0 +1,58 @@
+//===------- Offload API tests - offloadPlatformGetInfoSize ---------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+#include <offload_api.h>
+
+#include "../common/fixtures.hpp"
+#include "offloadPlatformInfo.hpp"
+
+struct offloadPlatformGetInfoSizeTest
+ : offloadPlatformTest,
+ ::testing::WithParamInterface<offload_platform_info_t> {};
+
+INSTANTIATE_TEST_SUITE_P(
+ offloadPlatformGetInfoSize, offloadPlatformGetInfoSizeTest,
+ ::testing::ValuesIn(PlatformQueries),
+ [](const ::testing::TestParamInfo<offload_platform_info_t> &info) {
+ std::stringstream ss;
+ ss << info.param;
+ return ss.str();
+ });
+
+TEST_P(offloadPlatformGetInfoSizeTest, Success) {
+ size_t Size = 0;
+ offload_platform_info_t InfoType = GetParam();
+
+ ASSERT_SUCCESS(offloadPlatformGetInfoSize(Platform, InfoType, &Size));
+ auto ExpectedSize = PlatformInfoSizeMap.find(InfoType);
+ if (ExpectedSize != PlatformInfoSizeMap.end()) {
+ ASSERT_EQ(Size, ExpectedSize->second);
+ } else {
+ ASSERT_NE(Size, 0lu);
+ }
+}
+
+TEST_F(offloadPlatformGetInfoSizeTest, InvalidNullHandle) {
+ size_t Size = 0;
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_HANDLE,
+ offloadPlatformGetInfoSize(
+ nullptr, OFFLOAD_PLATFORM_INFO_BACKEND, &Size));
+}
+
+TEST_F(offloadPlatformGetInfoSizeTest, InvalidPlatformInfoEnumeration) {
+ size_t Size = 0;
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_ENUMERATION,
+ offloadPlatformGetInfoSize(
+ Platform, OFFLOAD_PLATFORM_INFO_FORCE_UINT32, &Size));
+}
+
+TEST_F(offloadPlatformGetInfoSizeTest, InvalidNullPointer) {
+ ASSERT_ERROR(OFFLOAD_ERRC_INVALID_NULL_POINTER,
+ offloadPlatformGetInfoSize(
+ Platform, OFFLOAD_PLATFORM_INFO_BACKEND, nullptr));
+}
diff --git a/offload/unittests/OffloadAPI/platform/offloadPlatformInfo.hpp b/offload/unittests/OffloadAPI/platform/offloadPlatformInfo.hpp
new file mode 100644
index 00000000000000..c4a0fc21b33d05
--- /dev/null
+++ b/offload/unittests/OffloadAPI/platform/offloadPlatformInfo.hpp
@@ -0,0 +1,21 @@
+//===------- Offload API tests - Helpers for platform info query testing --===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+#pragma once
+
+#include <vector>
+
+// TODO: We could autogenerate these
+
+inline std::vector<offload_platform_info_t> PlatformQueries = {
+ OFFLOAD_PLATFORM_INFO_NAME, OFFLOAD_PLATFORM_INFO_VENDOR_NAME,
+ OFFLOAD_PLATFORM_INFO_VERSION, OFFLOAD_PLATFORM_INFO_BACKEND};
+
+inline std::unordered_map<offload_platform_info_t, size_t> PlatformInfoSizeMap =
+ {
+ {OFFLOAD_PLATFORM_INFO_BACKEND, sizeof(offload_platform_backend_t)},
+};
More information about the llvm-commits
mailing list