-
Notifications
You must be signed in to change notification settings - Fork 46
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[NFI] Move XPU driver code to
driver.c
(#264)
To be consistent with other 3p backends. Signed-off-by: Tsang, Whitney <whitney.tsang@intel.com>
- Loading branch information
1 parent
7f911ad
commit 53e277f
Showing
2 changed files
with
315 additions
and
318 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,313 @@ | ||
//===- driver.c -----------------------------------------------------------===// | ||
// | ||
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | ||
// See https://llvm.org/LICENSE.txt for license information. | ||
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | ||
// | ||
//===----------------------------------------------------------------------===// | ||
|
||
#include <cstddef> | ||
#include <iostream> | ||
#include <level_zero/ze_api.h> | ||
#include <string> | ||
#include <sycl/sycl.hpp> | ||
#include <unordered_map> | ||
#include <variant> | ||
#include <vector> | ||
|
||
#define NPY_NO_DEPRECATED_API NPY_1_7_API_VERSION | ||
#include <Python.h> | ||
#include <numpy/arrayobject.h> | ||
|
||
typedef struct l0_resc_handles { | ||
ze_context_handle_t context; | ||
ze_device_handle_t device; | ||
ze_command_queue_handle_t queue; | ||
ze_command_list_handle_t cmd_list; | ||
} l0_resc_handles; | ||
|
||
std::unordered_map<sycl::queue, l0_resc_handles> sycl_queue_map; | ||
static ze_context_handle_t context = {nullptr}; | ||
static ze_driver_handle_t driverHandle = {nullptr}; | ||
static ze_event_pool_handle_t eventPoolHandle = {nullptr}; | ||
|
||
static std::vector<ze_device_handle_t> devices; | ||
|
||
static inline void gpuAssert(ze_result_t code, const char *file, int line) { | ||
if (code != ZE_RESULT_SUCCESS) { | ||
const char *prefix = "Triton Error [ZE]: "; | ||
std::string str = std::to_string(code); | ||
char err[1024] = {0}; | ||
strcat(err, prefix); | ||
strcat(err, str.c_str()); | ||
PyErr_SetString(PyExc_RuntimeError, err); | ||
} | ||
} | ||
|
||
#define ZE_CHECK(ans) \ | ||
{ \ | ||
gpuAssert((ans), __FILE__, __LINE__); \ | ||
if (PyErr_Occurred()) \ | ||
return NULL; \ | ||
} | ||
|
||
static PyObject *getDeviceProperties(PyObject *self, PyObject *args) { | ||
int device_id; | ||
if (!PyArg_ParseTuple(args, "i", &device_id)) | ||
return NULL; | ||
|
||
if (device_id > devices.size()) { | ||
std::cerr << "Device ID not found: " << device_id << std::endl; | ||
return NULL; | ||
} | ||
|
||
// Get device handle | ||
ze_device_handle_t phDevice = devices[device_id]; | ||
|
||
// create a struct to hold device properties | ||
ze_device_properties_t device_properties = {}; | ||
device_properties.stype = ZE_STRUCTURE_TYPE_DEVICE_PROPERTIES; | ||
zeDeviceGetProperties(phDevice, &device_properties); | ||
|
||
int multiprocessor_count = | ||
device_properties.numSlices * device_properties.numSubslicesPerSlice; | ||
int sm_clock_rate = device_properties.coreClockRate; | ||
|
||
ze_device_compute_properties_t compute_properties = {}; | ||
compute_properties.stype = ZE_STRUCTURE_TYPE_DEVICE_COMPUTE_PROPERTIES; | ||
zeDeviceGetComputeProperties(phDevice, &compute_properties); | ||
int max_shared_mem = compute_properties.maxSharedLocalMemory; | ||
|
||
uint32_t memoryCount = 0; | ||
zeDeviceGetMemoryProperties(phDevice, &memoryCount, nullptr); | ||
auto pMemoryProperties = new ze_device_memory_properties_t[memoryCount]; | ||
for (uint32_t mem = 0; mem < memoryCount; ++mem) { | ||
pMemoryProperties[mem].stype = ZE_STRUCTURE_TYPE_DEVICE_MEMORY_PROPERTIES; | ||
pMemoryProperties[mem].pNext = nullptr; | ||
} | ||
zeDeviceGetMemoryProperties(phDevice, &memoryCount, pMemoryProperties); | ||
|
||
int mem_clock_rate = pMemoryProperties[0].maxClockRate; | ||
int mem_bus_width = pMemoryProperties[0].maxBusWidth; | ||
|
||
delete[] pMemoryProperties; | ||
|
||
return Py_BuildValue("{s:i, s:i, s:i, s:i, s:i}", "max_shared_mem", | ||
max_shared_mem, "multiprocessor_count", | ||
multiprocessor_count, "sm_clock_rate", sm_clock_rate, | ||
"mem_clock_rate", mem_clock_rate, "mem_bus_width", | ||
mem_bus_width); | ||
} | ||
|
||
static PyObject *loadBinary(PyObject *self, PyObject *args) { | ||
const char *name; | ||
int shared; | ||
PyObject *py_bytes; | ||
int device_id; | ||
if (!PyArg_ParseTuple(args, "sSii", &name, &py_bytes, &shared, &device_id)) { | ||
std::cerr << "loadBinary arg parse failed" << std::endl; | ||
return NULL; | ||
} | ||
|
||
if (device_id > devices.size()) { | ||
std::cerr << "Device ID not found: " << device_id << std::endl; | ||
return NULL; | ||
} | ||
|
||
ze_device_handle_t device = devices[device_id]; | ||
|
||
int32_t n_regs = 0; | ||
int32_t n_spills = 0; | ||
|
||
ze_module_desc_t module_desc = {}; | ||
module_desc.format = ZE_MODULE_FORMAT_IL_SPIRV; | ||
module_desc.inputSize = PyBytes_Size(py_bytes); | ||
module_desc.pInputModule = (uint8_t *)PyBytes_AsString(py_bytes); | ||
ze_module_handle_t module; | ||
ZE_CHECK(zeModuleCreate(context, device, &module_desc, &module, nullptr)); | ||
|
||
ze_kernel_desc_t kernel_desc = {}; | ||
kernel_desc.pKernelName = name; | ||
ze_kernel_handle_t fun; | ||
ZE_CHECK(zeKernelCreate(module, &kernel_desc, &fun)); | ||
|
||
if (PyErr_Occurred()) { | ||
std::cerr << "loadBinary error occurred" << std::endl; | ||
return NULL; | ||
} | ||
|
||
return Py_BuildValue("(KKii)", (uint64_t)module, (uint64_t)fun, n_regs, | ||
n_spills); | ||
} | ||
|
||
bool update(sycl::queue sycl_queue) { | ||
// Get l0-context | ||
auto sycl_context = sycl_queue.get_context(); | ||
ze_context_handle_t hCtxt = | ||
get_native<sycl::backend::level_zero>(sycl_context); | ||
// Get l0-device | ||
std::vector<sycl::device> sycl_devices = sycl_context.get_devices(); | ||
ze_device_handle_t hDev = | ||
get_native<sycl::backend::level_zero>(sycl_devices[0]); | ||
// Get l0-queue | ||
bool immediate_cmd_list = false; | ||
std::variant<ze_command_queue_handle_t, ze_command_list_handle_t> queue_var = | ||
get_native<sycl::backend::level_zero>(sycl_queue); | ||
auto l0_queue = std::get_if<ze_command_queue_handle_t>(&queue_var); | ||
if (l0_queue == nullptr) { | ||
auto imm_cmd_list = std::get_if<ze_command_list_handle_t>(&queue_var); | ||
if (imm_cmd_list == nullptr) { | ||
return false; | ||
} | ||
immediate_cmd_list = true; | ||
sycl_queue_map[sycl_queue].cmd_list = *imm_cmd_list; | ||
} | ||
sycl_queue_map[sycl_queue].context = hCtxt; | ||
sycl_queue_map[sycl_queue].device = hDev; | ||
sycl_queue_map[sycl_queue].queue = immediate_cmd_list ? 0 : *l0_queue; | ||
|
||
// Update global data | ||
context = sycl_queue_map[sycl_queue].context; | ||
uint32_t deviceCount = std::min(sycl_devices.size(), devices.size()); | ||
for (uint32_t i = 0; i < deviceCount; ++i) { | ||
devices[i] = sycl::get_native<sycl::backend::level_zero>(sycl_devices[i]); | ||
} | ||
|
||
return true; | ||
} | ||
|
||
static PyObject *initContext(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
if (sycl_queue_map.find(*sycl_queue) == sycl_queue_map.end()) { | ||
update(*sycl_queue); | ||
} | ||
context = sycl_queue_map[*sycl_queue].context; | ||
return Py_BuildValue("(K)", (uint64_t)context); | ||
} | ||
|
||
static PyObject *initEventPool(PyObject *self, PyObject *args) { | ||
// Create event pool | ||
ze_event_pool_desc_t tsEventPoolDesc = { | ||
ZE_STRUCTURE_TYPE_EVENT_POOL_DESC, nullptr, | ||
ZE_EVENT_POOL_FLAG_HOST_VISIBLE, // all events in pool are visible to Host | ||
1 // count | ||
}; | ||
ZE_CHECK(zeEventPoolCreate(context, &tsEventPoolDesc, 0, nullptr, | ||
&eventPoolHandle)); | ||
|
||
return Py_BuildValue("(K)", (uint64_t)eventPoolHandle); | ||
} | ||
|
||
static PyObject *initDevices(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
|
||
auto sycl_context = sycl_queue->get_context(); | ||
|
||
// Get l0-device | ||
std::vector<sycl::device> sycl_devices = sycl_context.get_devices(); | ||
|
||
// Retrieve devices | ||
uint32_t deviceCount = sycl_devices.size(); | ||
for (uint32_t i = 0; i < deviceCount; ++i) { | ||
devices.push_back( | ||
sycl::get_native<sycl::backend::level_zero>(sycl_devices[i])); | ||
} | ||
|
||
return Py_BuildValue("(i)", deviceCount); | ||
} | ||
|
||
static PyObject *getL0ImmCommandList(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
|
||
if (sycl_queue_map.find(*sycl_queue) == sycl_queue_map.end()) { | ||
update(*sycl_queue); | ||
} | ||
return Py_BuildValue("(K)", (uint64_t)(sycl_queue_map[*sycl_queue].cmd_list)); | ||
} | ||
static PyObject *getL0Queue(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
if (sycl_queue_map.find(*sycl_queue) == sycl_queue_map.end()) { | ||
update(*sycl_queue); | ||
} | ||
return Py_BuildValue("(K)", (uint64_t)(sycl_queue_map[*sycl_queue].queue)); | ||
} | ||
static PyObject *getL0DevPtr(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
if (sycl_queue_map.find(*sycl_queue) == sycl_queue_map.end()) { | ||
update(*sycl_queue); | ||
} | ||
return Py_BuildValue("(K)", (uint64_t)(sycl_queue_map[*sycl_queue].device)); | ||
} | ||
static PyObject *getL0CtxtPtr(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
if (sycl_queue_map.find(*sycl_queue) == sycl_queue_map.end()) { | ||
update(*sycl_queue); | ||
} | ||
return Py_BuildValue("(K)", (uint64_t)(sycl_queue_map[*sycl_queue].context)); | ||
} | ||
static PyObject *isUsingICL(PyObject *self, PyObject *args) { | ||
void *queue; | ||
if (!PyArg_ParseTuple(args, "K", &queue)) | ||
return NULL; | ||
sycl::queue *sycl_queue = static_cast<sycl::queue *>(queue); | ||
if (sycl_queue_map.find(*sycl_queue) == sycl_queue_map.end()) { | ||
update(*sycl_queue); | ||
} | ||
uint32_t using_icl = sycl_queue_map[*sycl_queue].cmd_list != 0 ? 1 : 0; | ||
return Py_BuildValue("(i)", using_icl); | ||
} | ||
|
||
static PyMethodDef ModuleMethods[] = { | ||
{"load_binary", loadBinary, METH_VARARGS, | ||
"Load provided SPV into ZE driver"}, | ||
{"get_device_properties", getDeviceProperties, METH_VARARGS, | ||
"Get the properties for a given device"}, | ||
{"init_context", initContext, METH_VARARGS, | ||
"Initialize the ZE GPU context"}, | ||
{"init_devices", initDevices, METH_VARARGS, | ||
"Initialize the ZE GPU devices and return device count"}, | ||
{"init_event_pool", initEventPool, METH_VARARGS, | ||
"Initialize ZE event pool"}, | ||
{"get_l0_imm_cmd_list", getL0ImmCommandList, METH_VARARGS, | ||
"Get l0 command list in case of immediate command list"}, | ||
{"get_l0_queue", getL0Queue, METH_VARARGS, "Get l0 queue from sycl queue"}, | ||
{"get_l0_dev_ptr", getL0DevPtr, METH_VARARGS, | ||
"Extract l0 device pointer from sycl queue"}, | ||
{"get_l0_ctxt_ptr", getL0CtxtPtr, METH_VARARGS, | ||
"Extract l0 context pointer from sycl queue"}, | ||
{"is_using_icl", isUsingICL, METH_VARARGS, | ||
"Extract sycl queue info, if it is using ICL"}, | ||
{NULL, NULL, 0, NULL} // sentinel | ||
}; | ||
|
||
static struct PyModuleDef ModuleDef = {PyModuleDef_HEAD_INIT, "spirv_utils", | ||
NULL, // documentation | ||
-1, // size | ||
ModuleMethods}; | ||
|
||
PyMODINIT_FUNC PyInit_spirv_utils(void) { | ||
PyObject *m = PyModule_Create(&ModuleDef); | ||
if (m == NULL) { | ||
return NULL; | ||
} | ||
PyModule_AddFunctions(m, ModuleMethods); | ||
return m; | ||
} |
Oops, something went wrong.