RCCE_pwr_wq_framework.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. #include "RCCE.h"
  2. #include "RCCE_pwr_wq.h"
  3. #include <stdio.h>
  4. int RCCE_WI_valid(void *);
  5. int RCCE_qsort(char *, size_t, size_t, int (*)(const void*, const void*));
  6. /* comparison function used in routine to sort core IDs */
  7. int id_compare(const void *e1, const void *e2);
  8. int RCCE_setup_work_queue_teams(QUEUE_PARMS *wq_pars){
  9. int NP, ID, ue, size, mem, master, team_lead, team_size, local_rank;
  10. int test, isleader;
  11. int *team_member, *master_list;
  12. NP = wq_pars->NP = RCCE_num_ues();
  13. ID = wq_pars->ID = RCCE_ue();
  14. team_member = wq_pars->team_member;
  15. master_list = wq_pars->master_list;
  16. /* determine the number of UEs in the local power domain and form teams */
  17. wq_pars->team_size = team_size = RCCE_power_domain_size();
  18. wq_pars->team_lead = team_lead = RCCE_power_domain_master();
  19. if (team_lead == ID) {
  20. /* the team lead is the first team member */
  21. team_member[0] = team_lead;
  22. size = 1;
  23. /* the team leads collects IDs from its team members ... */
  24. while (size<team_size) for (ue=0; ue<NP; ue++) if (ue != team_lead) {
  25. RCCE_recv_test((char *)(&(team_member[size])), sizeof(int), ue, &test);
  26. if (test) team_member[size++] = ue;
  27. }
  28. /* ... and sends the list to all other team members, after sorting it */
  29. RCCE_qsort((char *)team_member, team_size, sizeof(int), id_compare);
  30. for (ue=1; ue<team_size; ue++)
  31. RCCE_send((char *)team_member, team_size*sizeof(int), team_member[ue]);
  32. }
  33. else {
  34. /* team members check in with the team lead ... */
  35. RCCE_send((char *)(&ID), sizeof(int), team_lead);
  36. /* ... and receive the complete list of team members */
  37. RCCE_recv((char *)team_member, team_size*sizeof(int), team_lead);
  38. }
  39. /* we assign the UE with the highest rank the role of master. We know that
  40. this UE is either in a power domain by itself, or there is another UE
  41. in the same power domain who is the power domain master, because the
  42. power domain master is always the UE in that domain with the lowest rank */
  43. master = wq_pars->master = NP-1;
  44. /* the team containing the overall master must remove it from its member list */
  45. if (team_member[team_size-1] == master) wq_pars->team_size = --team_size;
  46. /* the overall master is not in any team */
  47. if (ID==master) team_size = wq_pars->team_size = 0;
  48. /* each UE determines its rank within the team */
  49. local_rank = wq_pars->local_rank = 0;
  50. for (ue=0; ue<team_size; ue++) if (ID==team_member[ue])
  51. local_rank = wq_pars->local_rank = ue;
  52. /* this code determines number of power domain leads, plus list of UEs */
  53. if (ID == master) {
  54. wq_pars->master_number = 0;
  55. for (int ue=0; ue<RCCE_num_ues()-1; ue++) {
  56. /* ask each core whether it is a team lead or not */
  57. RCCE_recv((char *)(&isleader), sizeof(int), ue);
  58. if (isleader) {
  59. master_list[wq_pars->master_number] = ue;
  60. (wq_pars->master_number)++;
  61. }
  62. }
  63. }
  64. else {
  65. /* all cores let the master know their team lead status */
  66. isleader = (ID == team_lead);
  67. RCCE_send((char *)(&isleader), sizeof(int), master);
  68. }
  69. /* all UEs report their team size and memberships */
  70. // for (ue=0; ue<NP; ue++) {
  71. // RCCE_barrier(&RCCE_COMM_WORLD);
  72. // if (ID==ue) {
  73. // printf("UE %d (%d) is in a team with %d members: ", ID,
  74. // local_rank, team_size);
  75. // for (mem=0; mem<team_size; mem++) printf("%d ", team_member[mem]);
  76. // printf("\n");
  77. // }
  78. // }
  79. return (RCCE_SUCCESS);
  80. }
  81. int RCCE_queue_master_loop(void *work_item, QUEUE_PARMS *wq_pars){
  82. int ue, ignore, test, count;
  83. int size = RCCE_WI_size(work_item);
  84. void *address = RCCE_WI_address(work_item);
  85. count = 0;
  86. if (RCCE_WI_valid(work_item)) {
  87. /* service work requests from any UE; first come, first served */
  88. for (ue=0; ue<wq_pars->master_number; ue++) {
  89. RCCE_recv_test((char *)(&ignore), sizeof(int), wq_pars->master_list[ue], &test);
  90. if (test) {
  91. // printf("Master sends work to UE %d\n", wq_pars->master_list[ue]);
  92. RCCE_send((char *)address, size, wq_pars->master_list[ue]);
  93. count++;
  94. /* generate the next work item */
  95. RCCE_new_work_item(work_item, wq_pars);
  96. }
  97. }
  98. }
  99. else {
  100. /* this loop ends all teams, so must insist each team checks in */
  101. for (ue=0; ue<wq_pars->master_number; ue++) {
  102. RCCE_recv((char *)(&ignore), sizeof(int), wq_pars->master_list[ue]);
  103. // printf("Master sends end of work message to UE %d\n", ue);
  104. RCCE_send((char *)address, size, wq_pars->master_list[ue]);
  105. }
  106. }
  107. return(count);
  108. }
  109. int RCCE_queue_member_loop(void *work_item, QUEUE_PARMS *wq_pars) {
  110. int gimme_work, mem;
  111. int size = RCCE_WI_size(work_item);
  112. void *address = RCCE_WI_address(work_item);
  113. /* ask for work if I am a team lead */
  114. if (wq_pars->ID == wq_pars->team_lead) {
  115. RCCE_send((char *)(&gimme_work), sizeof(int), wq_pars->master);
  116. RCCE_recv((char *)address, size, wq_pars->master);
  117. /* team leads parcel out the work to the workers */
  118. for (mem=1; mem<(wq_pars->team_size); mem++) {
  119. printf("Team lead %d sends work to UE %d\n", RCCE_ue(), wq_pars->team_member[mem]);
  120. fflush(0);
  121. RCCE_send((char *)address, size, wq_pars->team_member[mem]);
  122. }
  123. }
  124. else {
  125. RCCE_recv((char *)address, size, wq_pars->team_lead);
  126. }
  127. if (RCCE_WI_valid(work_item)) {
  128. RCCE_execute_work_item(work_item, wq_pars);
  129. // printf("UE %d executed work item\n", wq_pars->ID);
  130. }
  131. else {
  132. // printf("UE %d received stop queue task\n", RCCE_ue());
  133. return(1);
  134. }
  135. return(RCCE_SUCCESS);
  136. }