|
@@ -0,0 +1,207 @@
|
|
|
+
|
|
|
+ *
|
|
|
+ * Copyright (C) 2010-2020 Université de Bordeaux, CNRS (LaBRI UMR 5800), Inria
|
|
|
+ *
|
|
|
+ * StarPU is free software; you can redistribute it and/or modify
|
|
|
+ * it under the terms of the GNU Lesser General Public License as published by
|
|
|
+ * the Free Software Foundation; either version 2.1 of the License, or (at
|
|
|
+ * your option) any later version.
|
|
|
+ *
|
|
|
+ * StarPU is distributed in the hope that it will be useful, but
|
|
|
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
+ *
|
|
|
+ * See the GNU Lesser General Public License in COPYING.LGPL for more details.
|
|
|
+ */
|
|
|
+
|
|
|
+#include <starpu.h>
|
|
|
+#include "../helper.h"
|
|
|
+
|
|
|
+
|
|
|
+ * Check that _release_to correctly interacts with tasks working on the same data
|
|
|
+ */
|
|
|
+
|
|
|
+#ifdef STARPU_QUICK_CHECK
|
|
|
+static unsigned ntasks = 10;
|
|
|
+#elif !defined(STARPU_LONG_CHECK)
|
|
|
+static unsigned ntasks = 1000;
|
|
|
+#else
|
|
|
+static unsigned ntasks = 10000;
|
|
|
+#endif
|
|
|
+
|
|
|
+#ifdef STARPU_USE_CUDA
|
|
|
+extern void increment_cuda(void *descr[], void *_args);
|
|
|
+#endif
|
|
|
+#ifdef STARPU_USE_OPENCL
|
|
|
+extern void increment_opencl(void *buffers[], void *args);
|
|
|
+#endif
|
|
|
+
|
|
|
+void increment_cpu(void *descr[], void *arg)
|
|
|
+{
|
|
|
+ (void)arg;
|
|
|
+ unsigned *tokenptr = (unsigned *)STARPU_VARIABLE_GET_PTR(descr[0]);
|
|
|
+ (*tokenptr)++;
|
|
|
+}
|
|
|
+
|
|
|
+static struct starpu_codelet increment_cl =
|
|
|
+{
|
|
|
+ .modes = { STARPU_RW },
|
|
|
+ .cpu_funcs = {increment_cpu},
|
|
|
+#ifdef STARPU_USE_CUDA
|
|
|
+ .cuda_funcs = {increment_cuda},
|
|
|
+ .cuda_flags = {STARPU_CUDA_ASYNC},
|
|
|
+#endif
|
|
|
+#ifdef STARPU_USE_OPENCL
|
|
|
+ .opencl_funcs = {increment_opencl},
|
|
|
+ .opencl_flags = {STARPU_OPENCL_ASYNC},
|
|
|
+#endif
|
|
|
+ .cpu_funcs_name = {"increment_cpu"},
|
|
|
+ .nbuffers = 1
|
|
|
+};
|
|
|
+
|
|
|
+void check_cpu(void *descr[], void *arg)
|
|
|
+{
|
|
|
+ unsigned *val = arg;
|
|
|
+ unsigned *tokenptr = (unsigned *)STARPU_VARIABLE_GET_PTR(descr[0]);
|
|
|
+ STARPU_ASSERT(*tokenptr == *val);
|
|
|
+}
|
|
|
+
|
|
|
+static struct starpu_codelet check_cl =
|
|
|
+{
|
|
|
+ .modes = { STARPU_R },
|
|
|
+ .cpu_funcs = {check_cpu},
|
|
|
+ .cpu_funcs_name = {"increment_cpu"},
|
|
|
+ .nbuffers = 1
|
|
|
+};
|
|
|
+
|
|
|
+unsigned token = 0;
|
|
|
+starpu_data_handle_t token_handle;
|
|
|
+
|
|
|
+static
|
|
|
+int increment_token(void)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+ struct starpu_task *task = starpu_task_create();
|
|
|
+ task->cl = &increment_cl;
|
|
|
+ task->handles[0] = token_handle;
|
|
|
+ ret = starpu_task_submit(task);
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+static
|
|
|
+int check_token(unsigned value)
|
|
|
+{
|
|
|
+ unsigned *value_p;
|
|
|
+ int ret;
|
|
|
+ struct starpu_task *task = starpu_task_create();
|
|
|
+ task->cl = &check_cl;
|
|
|
+ task->handles[0] = token_handle;
|
|
|
+ task->cl_arg = value_p = malloc(sizeof(*value_p));
|
|
|
+ task->cl_arg_size = sizeof(*value_p);
|
|
|
+ task->cl_arg_free = 1;
|
|
|
+ *value_p = value;
|
|
|
+ ret = starpu_task_submit(task);
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+static
|
|
|
+void callback(void *arg)
|
|
|
+{
|
|
|
+ (void)arg;
|
|
|
+ token++;
|
|
|
+ starpu_data_release_to(token_handle, STARPU_R);
|
|
|
+ starpu_sleep(0.001);
|
|
|
+ starpu_data_release(token_handle);
|
|
|
+}
|
|
|
+
|
|
|
+#ifdef STARPU_USE_OPENCL
|
|
|
+struct starpu_opencl_program opencl_program;
|
|
|
+#endif
|
|
|
+int main(int argc, char **argv)
|
|
|
+{
|
|
|
+ unsigned i;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ ret = starpu_initialize(NULL, &argc, &argv);
|
|
|
+ if (ret == -ENODEV) return STARPU_TEST_SKIPPED;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_init");
|
|
|
+
|
|
|
+#ifdef STARPU_USE_OPENCL
|
|
|
+ ret = starpu_opencl_load_opencl_from_file("tests/datawizard/acquire_release_opencl_kernel.cl",
|
|
|
+ &opencl_program, NULL);
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_opencl_load_opencl_from_file");
|
|
|
+#endif
|
|
|
+ starpu_variable_data_register(&token_handle, STARPU_MAIN_RAM, (uintptr_t)&token, sizeof(unsigned));
|
|
|
+
|
|
|
+ FPRINTF(stderr, "Token: %u\n", token);
|
|
|
+
|
|
|
+ for(i=0; i<ntasks; i++)
|
|
|
+ {
|
|
|
+
|
|
|
+ ret = starpu_data_acquire(token_handle, STARPU_RW);
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_data_acquire");
|
|
|
+
|
|
|
+ token ++;
|
|
|
+
|
|
|
+ ret = check_token(4*i+1);
|
|
|
+ if (ret == -ENODEV) goto enodev;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit");
|
|
|
+
|
|
|
+ ret = increment_token();
|
|
|
+ if (ret == -ENODEV) goto enodev;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit");
|
|
|
+
|
|
|
+ ret = check_token(4*i+2);
|
|
|
+ if (ret == -ENODEV) goto enodev;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit");
|
|
|
+
|
|
|
+ starpu_sleep(0.001);
|
|
|
+ starpu_data_release_to(token_handle, STARPU_R);
|
|
|
+
|
|
|
+ starpu_sleep(0.001);
|
|
|
+ starpu_data_release(token_handle);
|
|
|
+
|
|
|
+ ret = starpu_data_acquire_cb(token_handle, STARPU_RW, callback, NULL);
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_data_acquire_cb");
|
|
|
+
|
|
|
+ ret = check_token(4*i+3);
|
|
|
+ if (ret == -ENODEV) goto enodev;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit");
|
|
|
+
|
|
|
+ ret = increment_token();
|
|
|
+ if (ret == -ENODEV) goto enodev;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit");
|
|
|
+
|
|
|
+ ret = check_token(4*i+4);
|
|
|
+ if (ret == -ENODEV) goto enodev;
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit");
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ starpu_data_unregister(token_handle);
|
|
|
+
|
|
|
+#ifdef STARPU_USE_OPENCL
|
|
|
+ ret = starpu_opencl_unload_opencl(&opencl_program);
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_opencl_unload_opencl");
|
|
|
+#endif
|
|
|
+ starpu_shutdown();
|
|
|
+
|
|
|
+ FPRINTF(stderr, "Token: %u\n", token);
|
|
|
+ if (token == ntasks * 4)
|
|
|
+ ret = EXIT_SUCCESS;
|
|
|
+ else
|
|
|
+ ret = EXIT_FAILURE;
|
|
|
+ return ret;
|
|
|
+
|
|
|
+enodev:
|
|
|
+ starpu_data_unregister(token_handle);
|
|
|
+ fprintf(stderr, "WARNING: No one can execute this task\n");
|
|
|
+
|
|
|
+ * could perform the kernel, so this is not an error from StarPU */
|
|
|
+#ifdef STARPU_USE_OPENCL
|
|
|
+ ret = starpu_opencl_unload_opencl(&opencl_program);
|
|
|
+ STARPU_CHECK_RETURN_VALUE(ret, "starpu_opencl_unload_opencl");
|
|
|
+#endif
|
|
|
+ starpu_shutdown();
|
|
|
+ return STARPU_TEST_SKIPPED;
|
|
|
+}
|