/* StarPU --- Runtime system for heterogeneous multicore architectures. * * 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 #include #ifdef STARPU_HAVE_VALGRIND_H #include #endif #ifdef STARPU_QUICK_CHECK #define NTASKS 64 #else #define NTASKS 1000 #endif int tasks_executed = 0; int ctx1_tasks_executed = 0; int ctx2_tasks_executed = 0; int cpu_tasks_executed = 0; int gpu_tasks_executed = 0; static void sched_ctx_cpu_func(void *descr[], void *arg) { (void)descr; (void)arg; (void)STARPU_ATOMIC_ADD(&tasks_executed,1); (void)STARPU_ATOMIC_ADD(&ctx1_tasks_executed,1); (void)STARPU_ATOMIC_ADD(&cpu_tasks_executed,1); } static void sched_ctx2_cpu_func(void *descr[], void *arg) { (void)descr; (void)arg; (void)STARPU_ATOMIC_ADD(&tasks_executed,1); (void)STARPU_ATOMIC_ADD(&ctx2_tasks_executed,1); (void)STARPU_ATOMIC_ADD(&cpu_tasks_executed,1); } static void sched_ctx2_cuda_func(void *descr[], void *arg) { (void)descr; (void)arg; (void)STARPU_ATOMIC_ADD(&tasks_executed,1); (void)STARPU_ATOMIC_ADD(&ctx2_tasks_executed,1); (void)STARPU_ATOMIC_ADD(&gpu_tasks_executed,1); } static struct starpu_codelet sched_ctx_codelet1 = { .cpu_funcs = {sched_ctx_cpu_func}, .model = NULL, .nbuffers = 0, .name = "sched_ctx" }; static struct starpu_codelet sched_ctx_codelet2 = { .cpu_funcs = {sched_ctx2_cpu_func}, .cuda_funcs = {sched_ctx2_cuda_func}, .model = NULL, .nbuffers = 0, .name = "sched_ctx" }; int main(void) { int ntasks = NTASKS; int ret; unsigned ncuda = 0; int nprocs1 = 0; int nprocs2 = 0; int procs1[STARPU_NMAXWORKERS], procs2[STARPU_NMAXWORKERS]; char *sched = getenv("STARPU_SCHED"); ret = starpu_init(NULL); if (ret == -ENODEV) return 77; STARPU_CHECK_RETURN_VALUE(ret, "starpu_init"); #ifdef STARPU_HAVE_VALGRIND_H if (RUNNING_ON_VALGRIND) ntasks = 8; #endif #ifdef STARPU_USE_CPU nprocs1 = starpu_cpu_worker_get_count(); starpu_worker_get_ids_by_type(STARPU_CPU_WORKER, procs1, nprocs1); #endif // if there is no cpu, skip if (nprocs1 == 0) goto enodev; #ifdef STARPU_USE_CUDA ncuda = nprocs2 = starpu_cuda_worker_get_count(); starpu_worker_get_ids_by_type(STARPU_CUDA_WORKER, procs2, nprocs2); #endif if (nprocs2 == 0) { nprocs2 = 1; procs2[0] = procs1[0]; } /*create contexts however you want*/ unsigned sched_ctx1 = starpu_sched_ctx_create(procs1, nprocs1, "ctx1", STARPU_SCHED_CTX_POLICY_NAME, sched?sched:"eager", 0); unsigned sched_ctx2 = starpu_sched_ctx_create(procs2, nprocs2, "ctx2", STARPU_SCHED_CTX_POLICY_NAME, sched?sched:"eager", 0); /*indicate what to do with the resources when context 2 finishes (it depends on your application)*/ starpu_sched_ctx_set_inheritor(sched_ctx2, sched_ctx1); starpu_sched_ctx_display_workers(sched_ctx2, stderr); int i; for (i = 0; i < ntasks/2; i++) { struct starpu_task *task = starpu_task_create(); task->cl = &sched_ctx_codelet1; task->cl_arg = NULL; /*submit tasks to context*/ ret = starpu_task_submit_to_ctx(task,sched_ctx1); STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit"); } /* tell starpu when you finished submitting tasks to this context in order to allow moving resources from this context to the inheritor one when its corresponding tasks finished executing */ starpu_sched_ctx_finished_submit(sched_ctx1); for (i = 0; i < ntasks/2; i++) { struct starpu_task *task = starpu_task_create(); task->cl = &sched_ctx_codelet2; task->cl_arg = NULL; ret = starpu_task_submit_to_ctx(task,sched_ctx2); STARPU_CHECK_RETURN_VALUE(ret, "starpu_task_submit"); } starpu_sched_ctx_finished_submit(sched_ctx2); /* wait for all tasks at the end*/ starpu_task_wait_for_all(); starpu_sched_ctx_add_workers(procs1, nprocs1, sched_ctx2); starpu_sched_ctx_delete(sched_ctx1); starpu_sched_ctx_delete(sched_ctx2); printf("tasks executed %d out of %d\n", tasks_executed, ntasks); printf("tasks executed on ctx1: %d\n", ctx1_tasks_executed); printf("tasks executed on ctx2: %d\n", ctx2_tasks_executed); printf("tasks executed on CPU: %d\n", cpu_tasks_executed); printf("tasks executed on GPU: %d\n", gpu_tasks_executed); enodev: starpu_shutdown(); return nprocs1 == 0 ? 77 : 0; }