modeld: PC Thneed prereqs (#25615)
* pc thneed prereqs * ugh, out of date * that can stay private * memcpy here is fine in SNPE variant * release files * thneed docs don't work anymore. they didn't look too useful Co-authored-by: Comma Device <device@comma.ai>pull/25622/head
parent
452d5e42ec
commit
b6e355a933
15 changed files with 312 additions and 240 deletions
@ -0,0 +1,236 @@ |
||||
#include "selfdrive/modeld/thneed/thneed.h" |
||||
|
||||
#include <cassert> |
||||
#include <cstring> |
||||
#include <map> |
||||
|
||||
#include "common/clutil.h" |
||||
#include "common/timing.h" |
||||
|
||||
map<pair<cl_kernel, int>, string> g_args; |
||||
map<pair<cl_kernel, int>, int> g_args_size; |
||||
map<cl_program, string> g_program_source; |
||||
|
||||
void Thneed::clinit() { |
||||
device_id = cl_get_device_id(CL_DEVICE_TYPE_DEFAULT); |
||||
if (context == NULL) context = CL_CHECK_ERR(clCreateContext(NULL, 1, &device_id, NULL, NULL, &err)); |
||||
//cl_command_queue_properties props[3] = {CL_QUEUE_PROPERTIES, CL_QUEUE_PROFILING_ENABLE, 0};
|
||||
cl_command_queue_properties props[3] = {CL_QUEUE_PROPERTIES, 0, 0}; |
||||
command_queue = CL_CHECK_ERR(clCreateCommandQueueWithProperties(context, device_id, props, &err)); |
||||
printf("Thneed::clinit done\n"); |
||||
} |
||||
|
||||
cl_int Thneed::clexec() { |
||||
if (debug >= 1) printf("Thneed::clexec: running %lu queued kernels\n", kq.size()); |
||||
for (auto &k : kq) { |
||||
if (record) ckq.push_back(k); |
||||
cl_int ret = k->exec(); |
||||
assert(ret == CL_SUCCESS); |
||||
} |
||||
return clFinish(command_queue); |
||||
} |
||||
|
||||
void Thneed::copy_inputs(float **finputs) { |
||||
//cl_int ret;
|
||||
for (int idx = 0; idx < inputs.size(); ++idx) { |
||||
if (debug >= 1) printf("copying %lu -- %p -> %p (cl %p)\n", input_sizes[idx], finputs[idx], inputs[idx], input_clmem[idx]); |
||||
|
||||
// TODO: fix thneed caching
|
||||
if (finputs[idx] != NULL) memcpy(inputs[idx], finputs[idx], input_sizes[idx]); |
||||
//if (finputs[idx] != NULL) CL_CHECK(clEnqueueWriteBuffer(command_queue, input_clmem[idx], CL_TRUE, 0, input_sizes[idx], finputs[idx], 0, NULL, NULL));
|
||||
|
||||
// HACK
|
||||
//if (input_sizes[idx] == 16) memset((char*)inputs[idx] + 8, 0, 8);
|
||||
} |
||||
} |
||||
|
||||
void Thneed::copy_output(float *foutput) { |
||||
if (output != NULL) { |
||||
size_t sz; |
||||
clGetMemObjectInfo(output, CL_MEM_SIZE, sizeof(sz), &sz, NULL); |
||||
if (debug >= 1) printf("copying %lu for output %p -> %p\n", sz, output, foutput); |
||||
CL_CHECK(clEnqueueReadBuffer(command_queue, output, CL_TRUE, 0, sz, foutput, 0, NULL, NULL)); |
||||
} else { |
||||
printf("CAUTION: model output is NULL, does it have no outputs?\n"); |
||||
} |
||||
} |
||||
|
||||
// *********** CLQueuedKernel ***********
|
||||
|
||||
CLQueuedKernel::CLQueuedKernel(Thneed *lthneed, |
||||
cl_kernel _kernel, |
||||
cl_uint _work_dim, |
||||
const size_t *_global_work_size, |
||||
const size_t *_local_work_size) { |
||||
thneed = lthneed; |
||||
kernel = _kernel; |
||||
work_dim = _work_dim; |
||||
assert(work_dim <= 3); |
||||
for (int i = 0; i < work_dim; i++) { |
||||
global_work_size[i] = _global_work_size[i]; |
||||
local_work_size[i] = _local_work_size[i]; |
||||
} |
||||
|
||||
char _name[0x100]; |
||||
clGetKernelInfo(kernel, CL_KERNEL_FUNCTION_NAME, sizeof(_name), _name, NULL); |
||||
name = string(_name); |
||||
clGetKernelInfo(kernel, CL_KERNEL_NUM_ARGS, sizeof(num_args), &num_args, NULL); |
||||
|
||||
// get args
|
||||
for (int i = 0; i < num_args; i++) { |
||||
char arg_name[0x100] = {0}; |
||||
clGetKernelArgInfo(kernel, i, CL_KERNEL_ARG_NAME, sizeof(arg_name), arg_name, NULL); |
||||
arg_names.push_back(string(arg_name)); |
||||
clGetKernelArgInfo(kernel, i, CL_KERNEL_ARG_TYPE_NAME, sizeof(arg_name), arg_name, NULL); |
||||
arg_types.push_back(string(arg_name)); |
||||
|
||||
args.push_back(g_args[make_pair(kernel, i)]); |
||||
args_size.push_back(g_args_size[make_pair(kernel, i)]); |
||||
} |
||||
|
||||
// get program
|
||||
clGetKernelInfo(kernel, CL_KERNEL_PROGRAM, sizeof(program), &program, NULL); |
||||
} |
||||
|
||||
int CLQueuedKernel::get_arg_num(const char *search_arg_name) { |
||||
for (int i = 0; i < num_args; i++) { |
||||
if (arg_names[i] == search_arg_name) return i; |
||||
} |
||||
printf("failed to find %s in %s\n", search_arg_name, name.c_str()); |
||||
assert(false); |
||||
} |
||||
|
||||
cl_int CLQueuedKernel::exec() { |
||||
if (kernel == NULL) { |
||||
kernel = clCreateKernel(program, name.c_str(), NULL); |
||||
arg_names.clear(); |
||||
arg_types.clear(); |
||||
|
||||
for (int j = 0; j < num_args; j++) { |
||||
char arg_name[0x100] = {0}; |
||||
clGetKernelArgInfo(kernel, j, CL_KERNEL_ARG_NAME, sizeof(arg_name), arg_name, NULL); |
||||
arg_names.push_back(string(arg_name)); |
||||
clGetKernelArgInfo(kernel, j, CL_KERNEL_ARG_TYPE_NAME, sizeof(arg_name), arg_name, NULL); |
||||
arg_types.push_back(string(arg_name)); |
||||
|
||||
cl_int ret; |
||||
if (args[j].size() != 0) { |
||||
assert(args[j].size() == args_size[j]); |
||||
ret = thneed_clSetKernelArg(kernel, j, args[j].size(), args[j].data()); |
||||
} else { |
||||
ret = thneed_clSetKernelArg(kernel, j, args_size[j], NULL); |
||||
} |
||||
assert(ret == CL_SUCCESS); |
||||
} |
||||
} |
||||
|
||||
if (thneed->debug >= 1) { |
||||
debug_print(thneed->debug >= 2); |
||||
} |
||||
|
||||
return clEnqueueNDRangeKernel(thneed->command_queue, |
||||
kernel, work_dim, NULL, global_work_size, local_work_size, 0, NULL, NULL); |
||||
} |
||||
|
||||
uint64_t CLQueuedKernel::benchmark() { |
||||
uint64_t ret = 0; |
||||
int old_record = thneed->record; |
||||
thneed->record = 0; |
||||
clFinish(thneed->command_queue); |
||||
// TODO: benchmarking at a lower level will make this more accurate
|
||||
for (int i = 0; i < 10; i++) { |
||||
uint64_t sb = nanos_since_boot(); |
||||
exec(); |
||||
clFinish(thneed->command_queue); |
||||
uint64_t et = nanos_since_boot() - sb; |
||||
if (ret == 0 || et < ret) ret = et; |
||||
} |
||||
thneed->record = old_record; |
||||
return ret; |
||||
} |
||||
|
||||
void CLQueuedKernel::debug_print(bool verbose) { |
||||
printf("%p %56s -- ", kernel, name.c_str()); |
||||
for (int i = 0; i < work_dim; i++) { |
||||
printf("%4zu ", global_work_size[i]); |
||||
} |
||||
printf(" -- "); |
||||
for (int i = 0; i < work_dim; i++) { |
||||
printf("%4zu ", local_work_size[i]); |
||||
} |
||||
printf("\n"); |
||||
|
||||
if (verbose) { |
||||
for (int i = 0; i < num_args; i++) { |
||||
string arg = args[i]; |
||||
printf(" %s %s", arg_types[i].c_str(), arg_names[i].c_str()); |
||||
void *arg_value = (void*)arg.data(); |
||||
int arg_size = arg.size(); |
||||
if (arg_size == 0) { |
||||
printf(" (size) %d", args_size[i]); |
||||
} else if (arg_size == 1) { |
||||
printf(" = %d", *((char*)arg_value)); |
||||
} else if (arg_size == 2) { |
||||
printf(" = %d", *((short*)arg_value)); |
||||
} else if (arg_size == 4) { |
||||
if (arg_types[i] == "float") { |
||||
printf(" = %f", *((float*)arg_value)); |
||||
} else { |
||||
printf(" = %d", *((int*)arg_value)); |
||||
} |
||||
} else if (arg_size == 8) { |
||||
cl_mem val = (cl_mem)(*((uintptr_t*)arg_value)); |
||||
printf(" = %p", val); |
||||
if (val != NULL) { |
||||
cl_mem_object_type obj_type; |
||||
clGetMemObjectInfo(val, CL_MEM_TYPE, sizeof(obj_type), &obj_type, NULL); |
||||
if (arg_types[i] == "image2d_t" || arg_types[i] == "image1d_t" || obj_type == CL_MEM_OBJECT_IMAGE2D) { |
||||
cl_image_format format; |
||||
size_t width, height, depth, array_size, row_pitch, slice_pitch; |
||||
cl_mem buf; |
||||
clGetImageInfo(val, CL_IMAGE_FORMAT, sizeof(format), &format, NULL); |
||||
assert(format.image_channel_order == CL_RGBA); |
||||
assert(format.image_channel_data_type == CL_HALF_FLOAT || format.image_channel_data_type == CL_FLOAT); |
||||
clGetImageInfo(val, CL_IMAGE_WIDTH, sizeof(width), &width, NULL); |
||||
clGetImageInfo(val, CL_IMAGE_HEIGHT, sizeof(height), &height, NULL); |
||||
clGetImageInfo(val, CL_IMAGE_ROW_PITCH, sizeof(row_pitch), &row_pitch, NULL); |
||||
clGetImageInfo(val, CL_IMAGE_DEPTH, sizeof(depth), &depth, NULL); |
||||
clGetImageInfo(val, CL_IMAGE_ARRAY_SIZE, sizeof(array_size), &array_size, NULL); |
||||
clGetImageInfo(val, CL_IMAGE_SLICE_PITCH, sizeof(slice_pitch), &slice_pitch, NULL); |
||||
assert(depth == 0); |
||||
assert(array_size == 0); |
||||
assert(slice_pitch == 0); |
||||
|
||||
clGetImageInfo(val, CL_IMAGE_BUFFER, sizeof(buf), &buf, NULL); |
||||
size_t sz; |
||||
clGetMemObjectInfo(buf, CL_MEM_SIZE, sizeof(sz), &sz, NULL); |
||||
printf(" image %zu x %zu rp %zu @ %p buffer %zu", width, height, row_pitch, buf, sz); |
||||
} else { |
||||
size_t sz; |
||||
clGetMemObjectInfo(val, CL_MEM_SIZE, sizeof(sz), &sz, NULL); |
||||
printf(" buffer %zu", sz); |
||||
} |
||||
} |
||||
} |
||||
printf("\n"); |
||||
} |
||||
} |
||||
} |
||||
|
||||
cl_int thneed_clSetKernelArg(cl_kernel kernel, cl_uint arg_index, size_t arg_size, const void *arg_value) { |
||||
g_args_size[make_pair(kernel, arg_index)] = arg_size; |
||||
if (arg_value != NULL) { |
||||
g_args[make_pair(kernel, arg_index)] = string((char*)arg_value, arg_size); |
||||
} else { |
||||
g_args[make_pair(kernel, arg_index)] = string(""); |
||||
} |
||||
cl_int ret = clSetKernelArg(kernel, arg_index, arg_size, arg_value); |
||||
return ret; |
||||
} |
||||
|
||||
cl_program thneed_clCreateProgramWithSource(cl_context context, cl_uint count, const char **strings, const size_t *lengths, cl_int *errcode_ret) { |
||||
assert(count == 1); |
||||
cl_program ret = clCreateProgramWithSource(context, count, strings, lengths, errcode_ret); |
||||
g_program_source[ret] = strings[0]; |
||||
return ret; |
||||
} |
Loading…
Reference in new issue