diff options
author | Aiden Gall <aiden@aidengall.xyz> | 2024-02-04 14:43:37 +0000 |
---|---|---|
committer | Aiden Gall <aiden@aidengall.xyz> | 2024-02-04 14:46:37 +0000 |
commit | c44d2805ba480fc7c2faf8ae07f2c1c9febf7e34 (patch) | |
tree | 0171574e51da53dbe2664360949cc78259b2da0b /src/spirt.c | |
parent | 5ff9d6184771970c14e682ecf55cbd9d27b2fada (diff) |
render with raylib
Diffstat (limited to 'src/spirt.c')
-rw-r--r-- | src/spirt.c | 313 |
1 files changed, 105 insertions, 208 deletions
diff --git a/src/spirt.c b/src/spirt.c index d1e8399..fd6457b 100644 --- a/src/spirt.c +++ b/src/spirt.c @@ -13,27 +13,21 @@ GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. */ -#include "farbfeld.h" #include "util.h" +#include "util_cl.h" #include <CL/cl.h> #include <CL/cl_platform.h> -#include <errno.h> #include <limits.h> -#include <stdint.h> +#include <raylib.h> #include <stdio.h> #include <stdlib.h> -#include <string.h> - -#ifndef SPIRV_START -# define SPIRV_START _binary_src_cl_spirt_spv_start -#endif -#ifndef SPIRV_END -# define SPIRV_END _binary_src_cl_spirt_spv_end -#endif + +#define SPIRV_START _binary_src_cl_spirt_spv_start +#define SPIRV_END _binary_src_cl_spirt_spv_end #define SPIRV_SIZE ((size_t)SPIRV_END - (size_t)SPIRV_START) -struct ClRuntime { +struct kernel_context { cl_platform_id platform; cl_device_id device; cl_context context; @@ -42,119 +36,72 @@ struct ClRuntime { cl_kernel k_hello; }; -static cl_platform_id *get_platforms(cl_uint *num_platforms); -static cl_device_id *get_devices(cl_platform_id platform, cl_uint *num_devices); -static cl_program compile_spirv_program(cl_context context, cl_device_id device, - const void *spirv_start, - size_t spirv_size); +struct opts { + long width; + long height; + long platidx; + long devidx; +}; + +static unsigned char *kernel_hello(struct kernel_context *runtime, + size_t image_width, size_t image_height); -static struct ClRuntime init(size_t platidx, size_t devidx); -static uint16_t *run_k_hello(struct ClRuntime *runtime, size_t *canvas_size_ret, - size_t image_width, size_t image_height); -static void clean(struct ClRuntime *runtime); +static struct kernel_context kernel_context_init(size_t platidx, size_t devidx); +static void kernel_context_clean(struct kernel_context *runtime); -static long argtonum(const char *arg, long minval, long maxval); +static struct opts get_args(int argc, char *argv[]); extern const char *SPIRV_START[]; extern const char *SPIRV_END[]; -static cl_platform_id * -get_platforms(cl_uint *const num_platforms) +static unsigned char * +kernel_hello(struct kernel_context *const runtime, const size_t image_width, + const size_t image_height) { cl_int err; - cl_platform_id *platforms; - cl_uint len; - - err = clGetPlatformIDs(0, NULL, &len); - if (err != CL_SUCCESS) - die("clGetPlatformIDs: %s\n", cl_strerror(err)); - warn("number of platforms = %d\n", len); - if (len < 1) - die("clGetPlatformIDs: No OpenCL platforms\n"); - - platforms = calloc(len, sizeof(*platforms)); - if (!platforms) - die("calloc: Out of memory\n"); - err = clGetPlatformIDs(len, platforms, NULL); - if (err != CL_SUCCESS) - die("clGetPlatformIDs: %s\n", cl_strerror(err)); + size_t global_item_size[2]; - *num_platforms = len; - return platforms; -} + unsigned char *h_canvas; + cl_mem d_canvas; + size_t canvas_size; -static cl_device_id * -get_devices(const cl_platform_id platform, cl_uint *const num_devices) -{ - cl_int err; + canvas_size = (sizeof("RGBA") - 1) * image_width * image_height; - cl_device_id *devices; - cl_uint len; + h_canvas = calloc(canvas_size, sizeof(*h_canvas)); + if (!h_canvas) + die("calloc: Out of memory\n"); - err = clGetDeviceIDs(platform, CL_DEVICE_TYPE_ALL, 0, NULL, &len); + d_canvas = clCreateBuffer(runtime->context, CL_MEM_WRITE_ONLY, + sizeof(*h_canvas) * canvas_size, NULL, &err); if (err != CL_SUCCESS) - die("clGetDeviceIDs: %s\n", cl_strerror(err)); - warn("number of devices in platform = %d\n", len); - if (len < 1) - die("clGetDeviceIDs: No OpenCL devices in platform\n"); + die("clCreateBuffer: %s\n", cl_strerror(err)); - devices = calloc(len, sizeof(*devices)); - if (!devices) - die("calloc: Out of memory\n"); - err = clGetDeviceIDs(platform, CL_DEVICE_TYPE_ALL, len, devices, NULL); + err = clSetKernelArg(runtime->k_hello, 0, sizeof(d_canvas), &d_canvas); if (err != CL_SUCCESS) - die("clGetDeviceIDs: %s\n", cl_strerror(err)); - - *num_devices = len; - return devices; -} + die("clSetKernelArg: %s\n", cl_strerror(err)); -static cl_program -compile_spirv_program(const cl_context context, const cl_device_id device, - const void *const spirv_start, const size_t spirv_size) -{ - cl_int err, build_err; - cl_program program; + global_item_size[0] = image_width; + global_item_size[1] = image_height; + err = clEnqueueNDRangeKernel(runtime->command_queue, runtime->k_hello, + 2, NULL, global_item_size, NULL, 0, NULL, + NULL); + if (err != CL_SUCCESS) + die("clEnqueueNDRangeKernel: %s\n", cl_strerror(err)); - program = clCreateProgramWithIL(context, spirv_start, spirv_size, &err); + err = clEnqueueReadBuffer(runtime->command_queue, d_canvas, CL_TRUE, 0, + sizeof(*h_canvas) * canvas_size, h_canvas, 0, + NULL, NULL); if (err != CL_SUCCESS) - die("clCreateProgramWithIL: %s\n", cl_strerror(err)); - - build_err = clBuildProgram(program, 1, &device, NULL, NULL, NULL); - if (build_err != CL_SUCCESS) { - cl_char *log; - size_t buffer_size; - - err = clGetProgramBuildInfo(program, device, - CL_PROGRAM_BUILD_LOG, 0, NULL, - &buffer_size); - if (err != CL_SUCCESS) - die("clGetProgramBuildInfo: %s\n", cl_strerror(err)); - if (buffer_size < 1) - die("clGetProgramBuildInfo: " - "Build log buffer is empty\n"); - - log = calloc(buffer_size, sizeof(*log)); - if (!log) - die("calloc: Out of memory\n"); - err = clGetProgramBuildInfo( - program, device, CL_PROGRAM_BUILD_LOG, - sizeof(*log) * buffer_size, log, NULL); - if (err != CL_SUCCESS) - die("clGetProgramBuildInfo: %s\n", cl_strerror(err)); - - efwrite(log, sizeof(*log), buffer_size, stderr); - - free(log); - die("\nclBuildProgram: %s\n", cl_strerror(build_err)); - } + die("clEnqueueReadBuffer: %s\n", cl_strerror(err)); + + clReleaseMemObject(d_canvas); - return program; + return h_canvas; } -static struct ClRuntime -init(const size_t platidx, const size_t devidx) +static struct kernel_context +kernel_context_init(const size_t platidx, const size_t devidx) { cl_int err; @@ -163,7 +110,7 @@ init(const size_t platidx, const size_t devidx) cl_uint num_platforms; cl_uint num_devices; - struct ClRuntime runtime; + struct kernel_context runtime; platforms = get_platforms(&num_platforms); if (platidx >= num_platforms) @@ -198,55 +145,8 @@ init(const size_t platidx, const size_t devidx) return runtime; } -static uint16_t * -run_k_hello(struct ClRuntime *const runtime, size_t *const canvas_size_ret, - const size_t image_width, const size_t image_height) -{ - cl_int err; - - size_t global_item_size[2]; - - uint16_t *h_canvas; - cl_mem d_canvas; - size_t canvas_size; - - canvas_size = (sizeof("RGBA") - 1) * image_width * image_height; - - h_canvas = calloc(canvas_size, sizeof(*h_canvas)); - if (!h_canvas) - die("calloc: Out of memory\n"); - - d_canvas = clCreateBuffer(runtime->context, CL_MEM_WRITE_ONLY, - sizeof(*h_canvas) * canvas_size, NULL, &err); - if (err != CL_SUCCESS) - die("clCreateBuffer: %s\n", cl_strerror(err)); - - err = clSetKernelArg(runtime->k_hello, 0, sizeof(d_canvas), &d_canvas); - if (err != CL_SUCCESS) - die("clSetKernelArg: %s\n", cl_strerror(err)); - - global_item_size[0] = image_width; - global_item_size[1] = image_height; - err = clEnqueueNDRangeKernel(runtime->command_queue, runtime->k_hello, - 2, NULL, global_item_size, NULL, 0, NULL, - NULL); - if (err != CL_SUCCESS) - die("clEnqueueNDRangeKernel: %s\n", cl_strerror(err)); - - err = clEnqueueReadBuffer(runtime->command_queue, d_canvas, CL_TRUE, 0, - sizeof(*h_canvas) * canvas_size, h_canvas, 0, - NULL, NULL); - if (err != CL_SUCCESS) - die("clEnqueueReadBuffer: %s\n", cl_strerror(err)); - - clReleaseMemObject(d_canvas); - - *canvas_size_ret = canvas_size; - return h_canvas; -} - static void -clean(struct ClRuntime *const runtime) +kernel_context_clean(struct kernel_context *const runtime) { clFlush(runtime->command_queue); clFinish(runtime->command_queue); @@ -257,45 +157,15 @@ clean(struct ClRuntime *const runtime) clReleaseContext(runtime->context); } -static long -argtonum(const char *const arg, const long minval, const long maxval) +static struct opts +get_args(int argc, char *argv[]) { - long ret; - char *endptr; + struct opts opts; - if (minval > maxval) - die("argtonum: Invalid range\n"); - if (!arg) - die("argtonum: Missing argument\n"); - - errno = 0; - ret = strtol(arg, &endptr, 10); - if (errno) - die("strtol: %s\n", strerror(errno)); - if (arg == endptr || *endptr) - die("strtol: Invalid integer\n"); - - if (ret < minval || ret > maxval) - die("argtonum: Argument out of range\n"); - - return ret; -} - -int -main(int argc, char *argv[]) -{ - long opt_platidx, opt_devidx, opt_width, opt_height; - FILE *fd; - - struct ClRuntime runtime; - - uint16_t *h_canvas; - size_t canvas_size; - - opt_platidx = 0; - opt_devidx = 0; - opt_width = 256; - opt_height = 256; + opts.platidx = 0; + opts.devidx = 0; + opts.width = 256; + opts.height = 256; if (argv[0]) argv++, argc--; @@ -311,16 +181,16 @@ main(int argc, char *argv[]) switch (flag) { case 'p': - opt_platidx = argtonum(argv[0], 0, LONG_MAX); + opts.platidx = argtonum(argv[0], 0, LONG_MAX); break; case 'd': - opt_devidx = argtonum(argv[0], 0, LONG_MAX); + opts.devidx = argtonum(argv[0], 0, LONG_MAX); break; case 'w': - opt_width = argtonum(argv[0], 1, LONG_MAX); + opts.width = argtonum(argv[0], 1, LONG_MAX); break; case 'h': - opt_height = argtonum(argv[0], 1, LONG_MAX); + opts.height = argtonum(argv[0], 1, LONG_MAX); break; case '-': goto end_flags; @@ -330,28 +200,55 @@ main(int argc, char *argv[]) } end_flags: - if (argc > 1) + if (argc > 0) die("arg_parse: Unexpected argument\n"); - fd = stdout; - if (argv[0]) { - errno = 0; - fd = fopen(argv[0], "w"); - if (errno) - die("fopen: %s\n", strerror(errno)); - if (!fd) - die("fopen: Returned NULL\n"); - } + return opts; +} - runtime = init(opt_platidx, opt_devidx); +int +main(int argc, char *argv[]) +{ + struct opts opts; + + struct kernel_context runtime; + unsigned char *h_canvas; + + Image img; + Texture2D texture; + + opts = get_args(argc, argv); + + runtime = kernel_context_init(opts.platidx, opts.devidx); - h_canvas = run_k_hello(&runtime, &canvas_size, opt_width, opt_height); + h_canvas = kernel_hello(&runtime, opts.width, opts.height); + + img.data = h_canvas; + img.width = opts.width; + img.height = opts.height; + img.format = PIXELFORMAT_UNCOMPRESSED_R8G8B8A8; + img.mipmaps = 1; + + clFlush(runtime.command_queue); + + InitWindow(opts.width, opts.height, "spirt"); + + SetTargetFPS(60); + + texture = LoadTextureFromImage(img); + + while (!WindowShouldClose()) { + BeginDrawing(); + ClearBackground(BLACK); + DrawTexture(texture, 0, 0, WHITE); + EndDrawing(); + } - farbfeld_write(fd, h_canvas, canvas_size, opt_width, opt_height); - fclose(fd); + UnloadTexture(texture); + CloseWindow(); free(h_canvas); - clean(&runtime); + kernel_context_clean(&runtime); return 0; } |