heteroprio_test.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. /* StarPU --- Runtime system for heterogeneous multicore architectures.
  2. *
  3. * Copyright (C) 2015 INRIA
  4. *
  5. * StarPU is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU Lesser General Public License as published by
  7. * the Free Software Foundation; either version 2.1 of the License, or (at
  8. * your option) any later version.
  9. *
  10. * StarPU is distributed in the hope that it will be useful, but
  11. * WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  13. *
  14. * See the GNU Lesser General Public License in COPYING.LGPL for more details.
  15. */
  16. #include <starpu.h>
  17. #include <schedulers/starpu_heteroprio.h>
  18. #include <unistd.h>
  19. #define FPRINTF(ofile, fmt, ...) do { if (!getenv("STARPU_SSILENT")) {fprintf(ofile, fmt, ## __VA_ARGS__); }} while(0)
  20. void initSchedulerCallback()
  21. {
  22. // CPU uses 3 buckets
  23. #ifdef STARPU_USE_CPU
  24. if (starpu_cpu_worker_get_count())
  25. {
  26. starpu_heteroprio_set_nb_prios(0, STARPU_CPU_IDX, 3);
  27. // It uses direct mapping idx => idx
  28. unsigned idx;
  29. for(idx = 0; idx < 3; ++idx)
  30. {
  31. starpu_heteroprio_set_mapping(0, STARPU_CPU_IDX, idx, idx);
  32. starpu_heteroprio_set_faster_arch(0, STARPU_CPU_IDX, idx);
  33. }
  34. }
  35. #endif
  36. #ifdef STARPU_USE_OPENCL
  37. // OpenCL is enabled and uses 2 buckets
  38. starpu_heteroprio_set_nb_prios(0, STARPU_OPENCL_IDX, 2);
  39. // OpenCL will first look to priority 2
  40. int prio2 = starpu_cpu_worker_get_count() ? 2 : 1;
  41. starpu_heteroprio_set_mapping(0, STARPU_OPENCL_IDX, 0, prio2);
  42. // For this bucket OpenCL is the fastest
  43. starpu_heteroprio_set_faster_arch(0, STARPU_OPENCL_IDX, prio2);
  44. // And CPU is 4 times slower
  45. #ifdef STARPU_USE_CPU
  46. starpu_heteroprio_set_arch_slow_factor(0, STARPU_CPU_IDX, 2, 4.0f);
  47. #endif
  48. int prio1 = starpu_cpu_worker_get_count() ? 1 : 0;
  49. starpu_heteroprio_set_mapping(0, STARPU_OPENCL_IDX, 1, prio1);
  50. // We let the CPU as the fastest and tell that OpenCL is 1.7 times slower
  51. starpu_heteroprio_set_arch_slow_factor(0, STARPU_OPENCL_IDX, prio1, 1.7f);
  52. #endif
  53. }
  54. void callback_a_cpu(void *buffers[], void *cl_arg)
  55. {
  56. //usleep(100000);
  57. FPRINTF(stderr, "[COMMUTE_LOG] callback %s\n", __FUNCTION__); fflush(stderr);
  58. }
  59. void callback_b_cpu(void *buffers[], void *cl_arg)
  60. {
  61. // usleep(100000);
  62. FPRINTF(stderr, "[COMMUTE_LOG] callback %s\n", __FUNCTION__); fflush(stderr);
  63. }
  64. void callback_c_cpu(void *buffers[], void *cl_arg)
  65. {
  66. // usleep(100000);
  67. FPRINTF(stderr, "[COMMUTE_LOG] callback %s\n", __FUNCTION__); fflush(stderr);
  68. }
  69. #ifdef STARPU_USE_OPENCL
  70. void callback_a_opencl(void *buffers[], void *cl_arg)
  71. {
  72. usleep(100000);
  73. FPRINTF(stderr, "[COMMUTE_LOG] callback %s\n", __FUNCTION__); fflush(stderr);
  74. }
  75. void callback_b_opencl(void *buffers[], void *cl_arg)
  76. {
  77. usleep(100000);
  78. FPRINTF(stderr, "[COMMUTE_LOG] callback %s\n", __FUNCTION__); fflush(stderr);
  79. }
  80. void callback_c_opencl(void *buffers[], void *cl_arg)
  81. {
  82. usleep(100000);
  83. FPRINTF(stderr, "[COMMUTE_LOG] callback %s\n", __FUNCTION__); fflush(stderr);
  84. }
  85. #endif
  86. int main(int argc, char** argv)
  87. {
  88. unsigned ret;
  89. struct starpu_conf conf;
  90. int ncpus, nopencls;
  91. ret = starpu_conf_init(&conf);
  92. STARPU_CHECK_RETURN_VALUE(ret, "starpu_conf_init");
  93. assert(ret == 0);
  94. conf.sched_policy_name = "heteroprio";
  95. conf.sched_policy_init = &initSchedulerCallback;
  96. ret = starpu_init(&conf);
  97. if (ret == -ENODEV)
  98. return 77;
  99. ncpus = starpu_cpu_worker_get_count();
  100. nopencls = starpu_opencl_worker_get_count();
  101. FPRINTF(stderr, "Worker = %d\n", starpu_worker_get_count());
  102. FPRINTF(stderr, "Worker CPU = %d\n", ncpus);
  103. FPRINTF(stderr, "Worker OpenCL = %d\n", nopencls);
  104. if (ncpus + nopencls == 0)
  105. {
  106. FPRINTF(stderr, "Needs at least one CPU or OpenCL device\n");
  107. starpu_shutdown();
  108. return 77;
  109. }
  110. struct starpu_codelet codeleteA;
  111. {
  112. memset(&codeleteA, 0, sizeof(codeleteA));
  113. codeleteA.nbuffers = 2;
  114. codeleteA.modes[0] = STARPU_RW;
  115. codeleteA.modes[1] = STARPU_RW;
  116. codeleteA.name = "codeleteA";
  117. #ifdef STARPU_USE_CPU
  118. codeleteA.where = STARPU_CPU;
  119. codeleteA.cpu_funcs[0] = callback_a_cpu;
  120. #endif
  121. #ifdef STARPU_USE_OPENCL
  122. codeleteA.where |= STARPU_OPENCL;
  123. codeleteA.opencl_funcs[0] = callback_a_opencl;
  124. #endif
  125. }
  126. struct starpu_codelet codeleteB;
  127. {
  128. memset(&codeleteB, 0, sizeof(codeleteB));
  129. codeleteB.nbuffers = 2;
  130. codeleteB.modes[0] = STARPU_RW;
  131. codeleteB.modes[1] = STARPU_RW;
  132. codeleteB.name = "codeleteB";
  133. codeleteB.where = STARPU_CPU;
  134. codeleteB.cpu_funcs[0] = callback_b_cpu;
  135. #ifdef STARPU_USE_OPENCL
  136. codeleteB.where |= STARPU_OPENCL;
  137. codeleteB.opencl_funcs[0] = callback_b_opencl;
  138. #endif
  139. }
  140. struct starpu_codelet codeleteC;
  141. {
  142. memset(&codeleteC, 0, sizeof(codeleteC));
  143. codeleteC.nbuffers = 2;
  144. codeleteC.modes[0] = STARPU_RW;
  145. codeleteC.modes[1] = STARPU_RW;
  146. codeleteC.name = "codeleteC";
  147. codeleteC.where = STARPU_CPU;
  148. codeleteC.cpu_funcs[0] = callback_c_cpu;
  149. #ifdef STARPU_USE_OPENCL
  150. codeleteC.where |= STARPU_OPENCL;
  151. codeleteC.opencl_funcs[0] = callback_c_opencl;
  152. #endif
  153. }
  154. const int nbHandles = 10;
  155. FPRINTF(stderr, "Nb handles = %d\n", nbHandles);
  156. starpu_data_handle_t handles[nbHandles];
  157. memset(handles, 0, sizeof(handles[0])*nbHandles);
  158. int dataA[nbHandles];
  159. int idx;
  160. for(idx = 0; idx < nbHandles; ++idx){
  161. dataA[idx] = idx;
  162. }
  163. int idxHandle;
  164. for(idxHandle = 0; idxHandle < nbHandles; ++idxHandle){
  165. starpu_variable_data_register(&handles[idxHandle], 0, (uintptr_t)&dataA[idxHandle], sizeof(dataA[idxHandle]));
  166. }
  167. const int nbTasks = 4;
  168. FPRINTF(stderr, "Submit %d tasks \n", nbTasks);
  169. int prio2 = starpu_cpu_worker_get_count() ? 2 : 1;
  170. int idxTask;
  171. for(idxTask = 0; idxTask < nbTasks; ++idxTask)
  172. {
  173. starpu_insert_task(&codeleteA,
  174. STARPU_PRIORITY, 0,
  175. (STARPU_RW), handles[(idxTask*2)%nbHandles],
  176. (STARPU_RW), handles[(idxTask*3+1)%nbHandles],
  177. 0);
  178. starpu_insert_task(&codeleteB,
  179. STARPU_PRIORITY, 1,
  180. (STARPU_RW), handles[(idxTask*2 +1 )%nbHandles],
  181. (STARPU_RW), handles[(idxTask*2)%nbHandles],
  182. 0);
  183. starpu_insert_task(&codeleteC,
  184. STARPU_PRIORITY, prio2,
  185. (STARPU_RW), handles[(idxTask)%nbHandles],
  186. (STARPU_RW), handles[(idxTask*idxTask)%nbHandles],
  187. 0);
  188. }
  189. FPRINTF(stderr, "Wait task\n");
  190. starpu_task_wait_for_all();
  191. FPRINTF(stderr, "Release data\n");
  192. for(idxHandle = 0 ; idxHandle < nbHandles ; ++idxHandle)
  193. {
  194. starpu_data_unregister(handles[idxHandle]);
  195. }
  196. FPRINTF(stderr, "Shutdown\n");
  197. starpu_shutdown();
  198. return 0;
  199. }