summaryrefslogtreecommitdiffstats
path: root/SD-VBS/common/c
diff options
context:
space:
mode:
Diffstat (limited to 'SD-VBS/common/c')
-rw-r--r--SD-VBS/common/c/extra.h239
1 files changed, 134 insertions, 105 deletions
diff --git a/SD-VBS/common/c/extra.h b/SD-VBS/common/c/extra.h
index 8c67b33..9c72064 100644
--- a/SD-VBS/common/c/extra.h
+++ b/SD-VBS/common/c/extra.h
@@ -29,8 +29,8 @@ extern int sched_getcpu();
29#include <sys/syscall.h> 29#include <sys/syscall.h>
30#endif 30#endif
31 31
32// This is a proxy for "case study mode" now
32#define LITMUS 1 33#define LITMUS 1
33#define MC2 0
34#define MMDC_PROF 0 34#define MMDC_PROF 0
35 35
36#if LITMUS 36#if LITMUS
@@ -41,18 +41,6 @@ extern int sched_getcpu();
41#include "/media/speedy/litmus/tools/mmdc/mmdc.h" 41#include "/media/speedy/litmus/tools/mmdc/mmdc.h"
42#endif 42#endif
43 43
44#if LITMUS
45#define SET_UP LOAD_PARAMS SETUP_LITMUS
46#else
47#define SET_UP LOAD_PARAMS
48#endif
49
50#if MMDC_PROF
51#define LOAD_PARAMS LOAD_PARAMS_ITRL SETUP_MMDC
52#else
53#define LOAD_PARAMS LOAD_PARAMS_ITRL
54#endif
55
56// Store state globally so that the job can be outside main() 44// Store state globally so that the job can be outside main()
57// Arrays use float as a comprimise between overflow and size 45// Arrays use float as a comprimise between overflow and size
58// Paired arrays use long longs as precision is more important for those times 46// Paired arrays use long longs as precision is more important for those times
@@ -82,50 +70,72 @@ char *_rt_other_core;
82char *_rt_barrier; 70char *_rt_barrier;
83sem_t *_rt_first_sem, *_rt_second_sem; 71sem_t *_rt_first_sem, *_rt_second_sem;
84int _rt_lock_id; 72int _rt_lock_id;
73#define _ID_SZ 128
74char _rt_sem1_name[_ID_SZ] = "/_libextra_first_sem-";
75char _rt_sem2_name[_ID_SZ] = "/_libextra_second_sem-";
76char _rt_shm_name[_ID_SZ] = "/_libextra_barrier-";
77#endif /* PAIRED */
78
79#if LITMUS
80long unsigned int _rt_period;
85#endif 81#endif
86 82
87static void _rt_load_params_itrl(int argc, char **argv) { 83static void _rt_load_params_itrl(int argc, char **argv) {
88#ifdef PAIRED 84#ifdef PAIRED
89 if (argc != 8) { 85 if (argc != (8 + LITMUS*2) && argc != (9 + LITMUS*2)) {
90 fprintf(stderr, "Usage: %s <name> <loops> <my core> <other core> <other name> <runID> <lockID>", argv[0]); 86 fprintf(stderr, "Usage: %s <name> <loops> <my core> <other core> <other name> <runID> <save results?>", argv[0]);
91 fprintf(stderr, " <name> string for logging. Name of this task.\n");
92 fprintf(stderr, " <loops> integer number of iterations. -1 for infinite.\n");
93 fprintf(stderr, " <my core> UNUSED. Core is now auto-detected.\n");
94 fprintf(stderr, " <other core> integer for logging. Core of paired task.\n");
95 fprintf(stderr, " <other name> string for logging. Name of paired task.\n");
96 fprintf(stderr, " <runID> string to append with .txt to yield output file name.\n");
97 fprintf(stderr, " <lockID> 1 to indicate this is pair member 1, otherwise pair member 2.\n");
98 exit(1);
99 }
100#else 87#else
101 if (argc != 6) { 88 if (argc != (6 + LITMUS*2)) {
102 fprintf(stderr, "Usage: %s <name> <loops> <my core> <runID> <save results?>\n", argv[0]); 89 fprintf(stderr, "Usage: %s <name> <loops> <my core> <runID> <save results?>\n", argv[0]);
90#endif /* PAIRED */
103 fprintf(stderr, " <name> string for logging. Name of this task.\n"); 91 fprintf(stderr, " <name> string for logging. Name of this task.\n");
104 fprintf(stderr, " <loops> integer number of iterations. -1 for infinite.\n"); 92 fprintf(stderr, " <loops> integer number of iterations. -1 for infinite.\n");
105 fprintf(stderr, " <my core> UNUSED. Core is now auto-detected.\n"); 93 fprintf(stderr, " <my core> integer core number. Only used for LITMUS-RT.\n");
94#ifdef PAIRED
95 fprintf(stderr, " <other core> integer for logging. Core of paired task.\n");
96 fprintf(stderr, " <other name> string for logging. Name of paired task.\n");
97#endif /* PAIRED */
106 fprintf(stderr, " <runID> string to append with .txt to yield output file name.\n"); 98 fprintf(stderr, " <runID> string to append with .txt to yield output file name.\n");
107 fprintf(stderr, " <save results?> 1 to save results, 0 to discard.\n"); 99 fprintf(stderr, " <save results?> 1 to save results, 0 to discard.\n");
100#ifdef PAIRED
101 fprintf(stderr, " <pairID> (optional).\n");
102#endif
103#if LITMUS
104 fprintf(stderr, " <task period> in ms\n");
105 fprintf(stderr, " <task criticality level> 0 for Level-A, 1 for Level-B, 2 for Level-C\n");
106#endif /* LITMUS */
108 exit(1); 107 exit(1);
109 } 108 }
110#endif
111 _rt_our_prog_name = argv[1]; 109 _rt_our_prog_name = argv[1];
112 _rt_max_jobs = atol(argv[2]); 110 _rt_max_jobs = atol(argv[2]);
111#if !LITMUS
113 _rt_core = sched_getcpu(); 112 _rt_core = sched_getcpu();
113#else
114 _rt_core = atoi(argv[3]);
115#endif
114#ifdef PAIRED 116#ifdef PAIRED
115 _rt_other_core = argv[4]; 117 _rt_other_core = argv[4];
116 _rt_other_prog_name = argv[5]; 118 _rt_other_prog_name = argv[5];
117 _rt_run_id = argv[6]; 119 _rt_run_id = argv[6];
118 _rt_lock_id = atoi(argv[7]); 120 _rt_will_output = atoi(argv[7]);
119 // The paired version doesn't support disabling output (legacy compatibility) 121 char *pairId;
120 _rt_will_output = 1; 122 int end;
123 if (argc > 8) {
124 pairId = argv[8];
125 end = 8;
126 } else {
127 pairId = "none";
128 end = 9;
129 }
121#else 130#else
122 _rt_other_core = "none"; 131 _rt_other_core = "none";
123 _rt_other_prog_name = "none"; 132 _rt_other_prog_name = "none";
124 _rt_run_id = argv[4]; 133 _rt_run_id = argv[4];
125 _rt_will_output = atoi(argv[5]); 134 _rt_will_output = atoi(argv[5]);
135 int end = 6;
126#endif /* PAIRED */ 136#endif /* PAIRED */
127 if (_rt_max_jobs < 0 && _rt_will_output != 0) { 137 if (_rt_max_jobs < 0 && _rt_will_output != 0) {
128 fprintf(stderr, "Infinite loops only supported when _rt_will_output is disabled!\n"); 138 fprintf(stderr, "Infinite loops only supported when output is disabled!\n");
129 exit(1); 139 exit(1);
130 } 140 }
131 if (strlen(_rt_run_id) + 5 > _RT_FILENAME_LEN) { 141 if (strlen(_rt_run_id) + 5 > _RT_FILENAME_LEN) {
@@ -133,32 +143,54 @@ static void _rt_load_params_itrl(int argc, char **argv) {
133 exit(1); 143 exit(1);
134 } 144 }
135#ifdef PAIRED 145#ifdef PAIRED
146 // __rt_sem2_name happens to be the longest
147 if (strlen(pairId) + strlen(_rt_sem2_name) > _ID_SZ) {
148 fprintf(stderr, "PairID is too long! Maximum length is %ld characters.\n", _ID_SZ - strlen(_rt_sem2_name));
149 exit(1);
150 }
136 _rt_start_time = calloc(_rt_max_jobs * _rt_will_output, sizeof(long long)); 151 _rt_start_time = calloc(_rt_max_jobs * _rt_will_output, sizeof(long long));
137 _rt_end_time = calloc(_rt_max_jobs * _rt_will_output, sizeof(long long)); 152 _rt_end_time = calloc(_rt_max_jobs * _rt_will_output, sizeof(long long));
138 if (!_rt_end_time || !_rt_start_time) { 153 if (!_rt_end_time || !_rt_start_time) {
139 perror("Unable to allocate buffers for execution times"); 154 perror("Unable to allocate buffers for execution times");
140 exit(1); 155 exit(1);
141 } 156 }
142 _rt_first_sem = sem_open("/_libextra_first_sem", O_CREAT, 644, 0); 157 // Use PairID to create unique semaphore and shared memory paths
143 _rt_second_sem = sem_open("/_libextra_second_sem", O_CREAT, 644, 0); 158 strcat(_rt_sem1_name, pairId);
159 strcat(_rt_sem2_name, pairId);
160 strcat(_rt_shm_name, pairId);
161 _rt_first_sem = sem_open(_rt_sem1_name, O_CREAT, 644, 0);
162 _rt_second_sem = sem_open(_rt_sem2_name, O_CREAT, 644, 0);
144 if (_rt_first_sem == SEM_FAILED || _rt_second_sem == SEM_FAILED) { 163 if (_rt_first_sem == SEM_FAILED || _rt_second_sem == SEM_FAILED) {
145 perror("Error while creating semaphores"); 164 perror("Error while creating semaphores");
146 exit(1); 165 exit(1);
147 } 166 }
148 int barrier_file = shm_open("/_libextra_barrier", O_CREAT | O_RDWR, 644); 167 // Create shared memory for barrier synchronization and infer lock ID
168 int barrier_file = shm_open(_rt_shm_name, O_CREAT | O_RDWR | O_EXCL, 644);
169 if (barrier_file == -1) {
170 // File already existed - we're the 2nd program and thus lock ID 2
171 _rt_lock_id = 2;
172 barrier_file = shm_open(_rt_shm_name, O_CREAT | O_RDWR, 644);
173 } else {
174 _rt_lock_id = 1;
175 }
149 if (barrier_file == -1) { 176 if (barrier_file == -1) {
150 perror("Error while creating shared memory for barrier synchronization"); 177 perror("Error while creating shared memory for barrier synchronization");
151 exit(1); 178 exit(1);
152 } 179 }
153 if (ftruncate(barrier_file, 1) == -1) { 180 if (ftruncate(barrier_file, 2) == -1) {
154 perror("Error while setting size of shared memory for barrier synchronization"); 181 perror("Error while setting size of shared memory for barrier synchronization");
155 exit(1); 182 exit(1);
156 } 183 }
157 _rt_barrier = mmap(NULL, 1, PROT_WRITE, MAP_SHARED, barrier_file, 0); 184 _rt_barrier = mmap(NULL, 2, PROT_WRITE, MAP_SHARED, barrier_file, 0);
158 if (_rt_barrier == MAP_FAILED) { 185 if (_rt_barrier == MAP_FAILED) {
159 perror("Error while mapping shared memory for barrier synchronization"); 186 perror("Error while mapping shared memory for barrier synchronization");
160 exit(1); 187 exit(1);
161 } 188 }
189 // If we're the 2nd user of this barrier, mark it as in-use
190 if (_rt_lock_id == 2 && !__sync_bool_compare_and_swap(_rt_barrier+1, 0, 1)) {
191 fprintf(stderr, "Pair ID already in use!\n");
192 exit(1);
193 }
162 *_rt_barrier = 0; 194 *_rt_barrier = 0;
163#else 195#else
164 _rt_exec_time = calloc(_rt_max_jobs * _rt_will_output, sizeof(float)); 196 _rt_exec_time = calloc(_rt_max_jobs * _rt_will_output, sizeof(float));
@@ -169,8 +201,47 @@ static void _rt_load_params_itrl(int argc, char **argv) {
169#endif /* PAIRED */ 201#endif /* PAIRED */
170 _rt_jobs_complete = 0; 202 _rt_jobs_complete = 0;
171 mlockall(MCL_CURRENT || MCL_FUTURE); 203 mlockall(MCL_CURRENT || MCL_FUTURE);
204#if LITMUS
205 _rt_period = strtoul(argv[end], NULL, 10);
206 unsigned int crit = atoi(argv[end+1]);
207 unsigned int wait = 1;
208 if (be_migrate_to_domain(_rt_core) < 0) {
209 perror("Unable to migrate to specified CPU");
210 exit(1);
211 }
212 struct rt_task rt_param;
213 init_rt_task_param(&rt_param);
214 // Fake exec cost - this value ignored by the MC^2 scheduler
215 rt_param.exec_cost = _rt_period;
216 rt_param.period = ms2ns(_rt_period);
217 rt_param.relative_deadline = 0;
218 rt_param.phase = 0;
219 rt_param.priority = LITMUS_LOWEST_PRIORITY;
220 rt_param.cls = crit;
221 rt_param.release_policy = TASK_PERIODIC;
222 rt_param.budget_policy = NO_ENFORCEMENT;
223 rt_param.cpu = _rt_core;
224 if (set_rt_task_param(gettid(), &rt_param) < 0) {
225 perror("Unable to set real-time parameters");
226 exit(1);
227 }
228 if (init_litmus() != 0) {
229 perror("init_litmus failed");
230 exit(1);
231 }
232 if (task_mode(LITMUS_RT_TASK) != 0) {
233 perror("Unable to become real-time task");
234 exit(1);
235 }
236 if (wait && wait_for_ts_release() != 0) {
237 perror("Unable to wait for taskset release");
238 exit(1);
239 }
240#endif /* LITMUS */
241#if MMDC_PROF
242 SETUP_MMDC
243#endif
172} 244}
173#define LOAD_PARAMS_ITRL _rt_load_params_itrl(argc, argv);
174 245
175#define SETUP_MMDC \ 246#define SETUP_MMDC \
176 _rt_mmdc_read = calloc(_rt_max_jobs * _rt_will_output, sizeof(float));\ 247 _rt_mmdc_read = calloc(_rt_max_jobs * _rt_will_output, sizeof(float));\
@@ -194,54 +265,6 @@ static void _rt_load_params_itrl(int argc, char **argv) {
194 mmdc->madpcr1 = axi_arm1;\ 265 mmdc->madpcr1 = axi_arm1;\
195 msync(&(mmdc->madpcr1),4,MS_SYNC); 266 msync(&(mmdc->madpcr1),4,MS_SYNC);
196 267
197#define SETUP_LITMUS \
198 unsigned int wait = 0; \
199 if (be_migrate_to_domain(_rt_core) < 0) { \
200 perror("Unable to migrate to specified CPU"); \
201 exit(1); \
202 } \
203 struct rt_task rt_param; \
204 init_rt_task_param(&rt_param); \
205 /* Supposedly the next two parameters are irrelevant when reservations are enabled, but I'm leaving them anyway... */ \
206 rt_param.exec_cost = ms2ns(999); \
207 rt_param.period = ms2ns(1000); \
208 rt_param.priority = LITMUS_HIGHEST_PRIORITY; \
209 rt_param.cls = RT_CLASS_HARD; \
210 rt_param.release_policy = TASK_PERIODIC; \
211 rt_param.budget_policy = NO_ENFORCEMENT; \
212 rt_param.cpu = _rt_core; \
213 if (set_rt_task_param(gettid(), &rt_param) < 0) { \
214 perror("Unable to set real-time parameters"); \
215 exit(1); \
216 } \
217 if (init_litmus() != 0) { \
218 perror("init_litmus failed"); \
219 exit(1); \
220 } \
221 MC2_SETUP \
222 if (task_mode(LITMUS_RT_TASK) != 0) { \
223 perror("Unable to become real-time task"); \
224 exit(1); \
225 } \
226 if (wait && wait_for_ts_release() != 0) { \
227 perror("Unable to wait for taskset release"); \
228 exit(1); \
229 }
230
231#if MC2
232#define MC2_SETUP \
233
234 set_page_color(rt_param.cpu);
235#else
236#define MC2_SETUP
237#endif
238
239#define CLEANUP_LITMUS \
240 if (task_mode(BACKGROUND_TASK) != 0) { \
241 perror("Unable to become a real-time task"); \
242 exit(1); \
243 } \
244
245#if __arm__ 268#if __arm__
246// On ARM, manually flush the cache 269// On ARM, manually flush the cache
247#define FLUSH_CACHES \ 270#define FLUSH_CACHES \
@@ -305,28 +328,28 @@ static void _rt_load_params_itrl(int argc, char **argv) {
305 328
306// Buffer timing result from a single job 329// Buffer timing result from a single job
307static void _rt_save_job_result() { 330static void _rt_save_job_result() {
331 if (!_rt_will_output)
332 return;
308 if (_rt_jobs_complete >= _rt_max_jobs) { 333 if (_rt_jobs_complete >= _rt_max_jobs) {
309 fprintf(stderr, "Max jobs setting too small! Trying to record job #%ld when we only have space for %ld jobs. Exiting...\n", _rt_jobs_complete, _rt_max_jobs); 334 fprintf(stderr, "Max jobs setting too small! Trying to record job #%ld when we only have space for %ld jobs. Exiting...\n", _rt_jobs_complete, _rt_max_jobs);
310 exit(1); 335 exit(1);
311 } 336 }
312 if (_rt_jobs_complete > -1 && _rt_will_output) {
313#ifdef PAIRED 337#ifdef PAIRED
314 _rt_start_time[_rt_jobs_complete] = _rt_start.tv_sec; 338 _rt_start_time[_rt_jobs_complete] = _rt_start.tv_sec;
315 _rt_start_time[_rt_jobs_complete] *= _BILLION; 339 _rt_start_time[_rt_jobs_complete] *= _BILLION;
316 _rt_start_time[_rt_jobs_complete] += _rt_start.tv_nsec; 340 _rt_start_time[_rt_jobs_complete] += _rt_start.tv_nsec;
317 _rt_end_time[_rt_jobs_complete] = _rt_end.tv_sec; 341 _rt_end_time[_rt_jobs_complete] = _rt_end.tv_sec;
318 _rt_end_time[_rt_jobs_complete] *= _BILLION; 342 _rt_end_time[_rt_jobs_complete] *= _BILLION;
319 _rt_end_time[_rt_jobs_complete] += _rt_end.tv_nsec; 343 _rt_end_time[_rt_jobs_complete] += _rt_end.tv_nsec;
320#else 344#else
321 _rt_exec_time[_rt_jobs_complete] = _rt_end.tv_sec - _rt_start.tv_sec; 345 _rt_exec_time[_rt_jobs_complete] = _rt_end.tv_sec - _rt_start.tv_sec;
322 _rt_exec_time[_rt_jobs_complete] *= _BILLION; 346 _rt_exec_time[_rt_jobs_complete] *= _BILLION;
323 _rt_exec_time[_rt_jobs_complete] += _rt_end.tv_nsec - _rt_start.tv_nsec; 347 _rt_exec_time[_rt_jobs_complete] += _rt_end.tv_nsec - _rt_start.tv_nsec;
324#endif /* PAIRED */ 348#endif /* PAIRED */
325#if MMDC_PROF 349#if MMDC_PROF
326 _rt_mmdc_read[_rt_jobs_complete] = mmdc_res.read_bytes; 350 _rt_mmdc_read[_rt_jobs_complete] = mmdc_res.read_bytes;
327 _rt_mmdc_write[_rt_jobs_complete] = mmdc_res.write_bytes; 351 _rt_mmdc_write[_rt_jobs_complete] = mmdc_res.write_bytes;
328#endif 352#endif /* MMDC_PROF */
329 }
330} 353}
331 354
332// Save all buffered timing results to disk 355// Save all buffered timing results to disk
@@ -365,13 +388,16 @@ static void _rt_write_to_file() {
365 fclose(fp); 388 fclose(fp);
366out: 389out:
367#if LITMUS 390#if LITMUS
368 CLEANUP_LITMUS 391 if (task_mode(BACKGROUND_TASK) != 0) {
392 perror("Unable to become a real-time task");
393 exit(1);
394 }
369#endif /* LITMUS */ 395#endif /* LITMUS */
370#ifdef PAIRED 396#ifdef PAIRED
371 munmap(_rt_barrier, 1); 397 munmap(_rt_barrier, 2);
372 shm_unlink("/_libextra_barrier"); 398 sem_unlink(_rt_sem1_name);
373 sem_unlink("/_libextra_first_sem"); 399 sem_unlink(_rt_sem2_name);
374 sem_unlink("/_libextra_second_sem"); 400 shm_unlink(_rt_shm_name);
375 free(_rt_start_time); 401 free(_rt_start_time);
376 free(_rt_end_time); 402 free(_rt_end_time);
377#else 403#else
@@ -396,7 +422,9 @@ static void _rt_start_loop() {
396 FIRST_UNLOCK 422 FIRST_UNLOCK
397 FIRST_LOCK 423 FIRST_LOCK
398#endif /* PAIRED */ 424#endif /* PAIRED */
425#if !LITMUS
399 FLUSH_CACHES 426 FLUSH_CACHES
427#endif
400#ifdef PAIRED 428#ifdef PAIRED
401 BARRIER_SYNC 429 BARRIER_SYNC
402#endif /* PAIRED */ 430#endif /* PAIRED */
@@ -444,7 +472,7 @@ static void _rt_stop_loop() {
444 * result. We use this to call our void function from inside a comparison. 472 * result. We use this to call our void function from inside a comparison.
445 */ 473 */
446#define for_each_job \ 474#define for_each_job \
447 for (; _rt_jobs_complete < _rt_max_jobs && (_rt_start_loop(),1); \ 475 for (; (_rt_max_jobs == -1 || _rt_jobs_complete < _rt_max_jobs) && (_rt_start_loop(),1); \
448 _rt_stop_loop()) 476 _rt_stop_loop())
449 477
450/****** Legacy API ****** 478/****** Legacy API ******
@@ -466,6 +494,7 @@ static void _rt_stop_loop() {
466 * able to read them. 494 * able to read them.
467 */ 495 */
468static int jobsComplete = 0; 496static int jobsComplete = 0;
497#define SET_UP _rt_load_params_itrl(argc, argv);
469#define START_LOOP _rt_start_loop(); 498#define START_LOOP _rt_start_loop();
470#define STOP_LOOP _rt_stop_loop(); 499#define STOP_LOOP _rt_stop_loop();
471#define WRITE_TO_FILE _rt_write_to_file(); 500#define WRITE_TO_FILE _rt_write_to_file();