diff options
author | Glenn Elliott <gelliott@cs.unc.edu> | 2013-03-05 18:32:01 -0500 |
---|---|---|
committer | Glenn Elliott <gelliott@cs.unc.edu> | 2013-03-12 12:47:20 -0400 |
commit | 0a54a84457bb8a33113c7dd2a2b63b2a837cc92e (patch) | |
tree | 78bd56eec72c7d47749155315d3506c424e0dc24 | |
parent | 93ffe5be8fd1954bbfe5a04e55b81ac1d83d2de7 (diff) |
Cluster-aware rtspin and rt_launch.
This patch adds cluster scheduling options to
rtspin and rt_launch. The convenience routines
in litmus.h were also updated to facilitate
clustered scheduling. For partitioned scheduling,
just set cluster_size = 1 (default size for rtspin
and rt_launch).
-rw-r--r-- | bin/rt_launch.c | 22 | ||||
-rw-r--r-- | bin/rtspin.c | 24 | ||||
-rw-r--r-- | include/litmus.h | 36 | ||||
-rw-r--r-- | src/litmus.c | 17 | ||||
-rw-r--r-- | src/task.c | 14 |
5 files changed, 68 insertions, 45 deletions
diff --git a/bin/rt_launch.c b/bin/rt_launch.c index 97d9df9..ae68601 100644 --- a/bin/rt_launch.c +++ b/bin/rt_launch.c | |||
@@ -29,10 +29,11 @@ int launch(void *task_info_p) { | |||
29 | } | 29 | } |
30 | 30 | ||
31 | void usage(char *error) { | 31 | void usage(char *error) { |
32 | fprintf(stderr, "%s\nUsage: rt_launch [-w][-v][-p cpu][-q prio][-c hrt | srt | be] wcet period program [arg1 arg2 ...]\n" | 32 | fprintf(stderr, "%s\nUsage: rt_launch [-w][-v][-p partition/cluster [-z cluster size]][-q prio][-c hrt | srt | be] wcet period program [arg1 arg2 ...]\n" |
33 | "\t-w\tSynchronous release\n" | 33 | "\t-w\tSynchronous release\n" |
34 | "\t-v\tVerbose\n" | 34 | "\t-v\tVerbose\n" |
35 | "\t-p\tcpu (or initial cpu)\n" | 35 | "\t-p\tpartition or cluster\n" |
36 | "\t-z\tsize of cluster (default = 1 for partitioned)\n" | ||
36 | "\t-c\tClass\n" | 37 | "\t-c\tClass\n" |
37 | "\twcet, period in ms\n" | 38 | "\twcet, period in ms\n" |
38 | "\tprogram to be launched\n", | 39 | "\tprogram to be launched\n", |
@@ -41,7 +42,7 @@ void usage(char *error) { | |||
41 | } | 42 | } |
42 | 43 | ||
43 | 44 | ||
44 | #define OPTSTR "p:c:vwq:" | 45 | #define OPTSTR "p:z:c:vwq:" |
45 | 46 | ||
46 | int main(int argc, char** argv) | 47 | int main(int argc, char** argv) |
47 | { | 48 | { |
@@ -49,7 +50,8 @@ int main(int argc, char** argv) | |||
49 | lt_t wcet; | 50 | lt_t wcet; |
50 | lt_t period; | 51 | lt_t period; |
51 | int migrate = 0; | 52 | int migrate = 0; |
52 | int cpu = 0; | 53 | int cluster = 0; |
54 | int cluster_size = 1; | ||
53 | int opt; | 55 | int opt; |
54 | int verbose = 0; | 56 | int verbose = 0; |
55 | int wait = 0; | 57 | int wait = 0; |
@@ -66,9 +68,12 @@ int main(int argc, char** argv) | |||
66 | verbose = 1; | 68 | verbose = 1; |
67 | break; | 69 | break; |
68 | case 'p': | 70 | case 'p': |
69 | cpu = atoi(optarg); | 71 | cluster = atoi(optarg); |
70 | migrate = 1; | 72 | migrate = 1; |
71 | break; | 73 | break; |
74 | case 'z': | ||
75 | cluster_size = atoi(optarg); | ||
76 | break; | ||
72 | case 'q': | 77 | case 'q': |
73 | priority = atoi(optarg); | 78 | priority = atoi(optarg); |
74 | if (!litmus_is_valid_fixed_prio(priority)) | 79 | if (!litmus_is_valid_fixed_prio(priority)) |
@@ -109,11 +114,12 @@ int main(int argc, char** argv) | |||
109 | info.argv = argv + optind + 2; | 114 | info.argv = argv + optind + 2; |
110 | info.wait = wait; | 115 | info.wait = wait; |
111 | if (migrate) { | 116 | if (migrate) { |
112 | ret = be_migrate_to_cpu(cpu); | 117 | ret = be_migrate_to_cluster(cluster, cluster_size); |
113 | if (ret < 0) | 118 | if (ret < 0) |
114 | bail_out("could not migrate to target partition"); | 119 | bail_out("could not migrate to target partition or cluster"); |
115 | } | 120 | } |
116 | ret = __create_rt_task(launch, &info, cpu, wcet, period, priority, class); | 121 | ret = __create_rt_task(launch, &info, cluster, cluster_size, wcet, period, |
122 | priority, class); | ||
117 | 123 | ||
118 | 124 | ||
119 | if (ret < 0) | 125 | if (ret < 0) |
diff --git a/bin/rtspin.c b/bin/rtspin.c index 5f12930..b05c17c 100644 --- a/bin/rtspin.c +++ b/bin/rtspin.c | |||
@@ -21,7 +21,8 @@ static void usage(char *error) { | |||
21 | " rt_spin [COMMON-OPTS] -f FILE [-o COLUMN] WCET PERIOD\n" | 21 | " rt_spin [COMMON-OPTS] -f FILE [-o COLUMN] WCET PERIOD\n" |
22 | " rt_spin -l\n" | 22 | " rt_spin -l\n" |
23 | "\n" | 23 | "\n" |
24 | "COMMON-OPTS = [-w] [-p PARTITION] [-c CLASS] [-s SCALE]\n" | 24 | "COMMON-OPTS = [-w] [-s SCALE]\n" |
25 | " [-p PARTITION/CLUSTER [-z CLUSTER SIZE]] [-c CLASS]\n" | ||
25 | " [-X LOCKING-PROTOCOL] [-L CRITICAL SECTION LENGTH] [-Q RESOURCE-ID]" | 26 | " [-X LOCKING-PROTOCOL] [-L CRITICAL SECTION LENGTH] [-Q RESOURCE-ID]" |
26 | "\n" | 27 | "\n" |
27 | "WCET and PERIOD are milliseconds, DURATION is seconds.\n" | 28 | "WCET and PERIOD are milliseconds, DURATION is seconds.\n" |
@@ -183,7 +184,7 @@ static int job(double exec_time, double program_end, int lock_od, double cs_leng | |||
183 | } | 184 | } |
184 | } | 185 | } |
185 | 186 | ||
186 | #define OPTSTR "p:c:wlveo:f:s:q:X:L:Q:" | 187 | #define OPTSTR "p:z:c:wlveo:f:s:q:X:L:Q:" |
187 | 188 | ||
188 | int main(int argc, char** argv) | 189 | int main(int argc, char** argv) |
189 | { | 190 | { |
@@ -193,7 +194,8 @@ int main(int argc, char** argv) | |||
193 | double wcet_ms, period_ms; | 194 | double wcet_ms, period_ms; |
194 | unsigned int priority = LITMUS_LOWEST_PRIORITY; | 195 | unsigned int priority = LITMUS_LOWEST_PRIORITY; |
195 | int migrate = 0; | 196 | int migrate = 0; |
196 | int cpu = 0; | 197 | int cluster = 0; |
198 | int cluster_size = 1; | ||
197 | int opt; | 199 | int opt; |
198 | int wait = 0; | 200 | int wait = 0; |
199 | int test_loop = 0; | 201 | int test_loop = 0; |
@@ -221,9 +223,12 @@ int main(int argc, char** argv) | |||
221 | wait = 1; | 223 | wait = 1; |
222 | break; | 224 | break; |
223 | case 'p': | 225 | case 'p': |
224 | cpu = atoi(optarg); | 226 | cluster = atoi(optarg); |
225 | migrate = 1; | 227 | migrate = 1; |
226 | break; | 228 | break; |
229 | case 'z': | ||
230 | cluster_size = atoi(optarg); | ||
231 | break; | ||
227 | case 'q': | 232 | case 'q': |
228 | priority = atoi(optarg); | 233 | priority = atoi(optarg); |
229 | if (!litmus_is_valid_fixed_prio(priority)) | 234 | if (!litmus_is_valid_fixed_prio(priority)) |
@@ -320,13 +325,8 @@ int main(int argc, char** argv) | |||
320 | else if (file && num_jobs > 1) | 325 | else if (file && num_jobs > 1) |
321 | duration += period_ms * 0.001 * (num_jobs - 1); | 326 | duration += period_ms * 0.001 * (num_jobs - 1); |
322 | 327 | ||
323 | if (migrate) { | 328 | ret = sporadic_task_ns(wcet, period, 0, cluster, cluster_size, |
324 | ret = be_migrate_to_cpu(cpu); | 329 | priority, class, |
325 | if (ret < 0) | ||
326 | bail_out("could not migrate to target partition"); | ||
327 | } | ||
328 | |||
329 | ret = sporadic_task_ns(wcet, period, 0, cpu, priority, class, | ||
330 | want_enforcement ? PRECISE_ENFORCEMENT | 330 | want_enforcement ? PRECISE_ENFORCEMENT |
331 | : NO_ENFORCEMENT, | 331 | : NO_ENFORCEMENT, |
332 | migrate); | 332 | migrate); |
@@ -341,7 +341,7 @@ int main(int argc, char** argv) | |||
341 | 341 | ||
342 | if (protocol >= 0) { | 342 | if (protocol >= 0) { |
343 | /* open reference to semaphore */ | 343 | /* open reference to semaphore */ |
344 | lock_od = litmus_open_lock(protocol, resource_id, lock_namespace, &cpu); | 344 | lock_od = litmus_open_lock(protocol, resource_id, lock_namespace, &cluster); |
345 | if (lock_od < 0) { | 345 | if (lock_od < 0) { |
346 | perror("litmus_open_lock"); | 346 | perror("litmus_open_lock"); |
347 | usage("Could not open lock."); | 347 | usage("Could not open lock."); |
diff --git a/include/litmus.h b/include/litmus.h index b2e81f2..335d01b 100644 --- a/include/litmus.h +++ b/include/litmus.h | |||
@@ -21,6 +21,13 @@ extern "C" { | |||
21 | int set_rt_task_param(pid_t pid, struct rt_task* param); | 21 | int set_rt_task_param(pid_t pid, struct rt_task* param); |
22 | int get_rt_task_param(pid_t pid, struct rt_task* param); | 22 | int get_rt_task_param(pid_t pid, struct rt_task* param); |
23 | 23 | ||
24 | /* Release-master-aware functions for getting the first | ||
25 | * CPU in a particular cluster or partition. Use these | ||
26 | * to set rt_task::cpu for cluster/partitioned scheduling. | ||
27 | */ | ||
28 | int partition_to_cpu(int partition); | ||
29 | int cluster_to_first_cpu(int cluster, int cluster_size); | ||
30 | |||
24 | /* setup helper */ | 31 | /* setup helper */ |
25 | 32 | ||
26 | /* Times are given in ms. The 'priority' parameter | 33 | /* Times are given in ms. The 'priority' parameter |
@@ -30,9 +37,9 @@ int get_rt_task_param(pid_t pid, struct rt_task* param); | |||
30 | */ | 37 | */ |
31 | int sporadic_task( | 38 | int sporadic_task( |
32 | lt_t e, lt_t p, lt_t phase, | 39 | lt_t e, lt_t p, lt_t phase, |
33 | int partition, unsigned int priority, | 40 | int cluster, int cluster_size, unsigned int priority, |
34 | task_class_t cls, | 41 | task_class_t cls, |
35 | budget_policy_t budget_policy, int set_cpu_set); | 42 | budget_policy_t budget_policy, int be_migrate); |
36 | 43 | ||
37 | /* Times are given in ns. The 'priority' parameter | 44 | /* Times are given in ns. The 'priority' parameter |
38 | * is only relevant under fixed-priority scheduling (and | 45 | * is only relevant under fixed-priority scheduling (and |
@@ -41,16 +48,19 @@ int sporadic_task( | |||
41 | */ | 48 | */ |
42 | int sporadic_task_ns( | 49 | int sporadic_task_ns( |
43 | lt_t e, lt_t p, lt_t phase, | 50 | lt_t e, lt_t p, lt_t phase, |
44 | int cpu, unsigned int priority, | 51 | int cluster, int cluster_size, unsigned int priority, |
45 | task_class_t cls, | 52 | task_class_t cls, |
46 | budget_policy_t budget_policy, int set_cpu_set); | 53 | budget_policy_t budget_policy, int be_migrate); |
47 | 54 | ||
48 | /* Convenience macros. Budget enforcement off by default in these macros. */ | 55 | /* Convenience macros. Budget enforcement off by default in these macros. */ |
49 | #define sporadic_global(e, p) \ | 56 | #define sporadic_global(e, p) \ |
50 | sporadic_task(e, p, 0, 0, LITMUS_LOWEST_PRIORITY, \ | 57 | sporadic_task(e, p, 0, 0, 0, LITMUS_LOWEST_PRIORITY, \ |
51 | RT_CLASS_SOFT, NO_ENFORCEMENT, 0) | 58 | RT_CLASS_SOFT, NO_ENFORCEMENT, 0) |
52 | #define sporadic_partitioned(e, p, cpu) \ | 59 | #define sporadic_partitioned(e, p, partition) \ |
53 | sporadic_task(e, p, 0, cpu, LITMUS_LOWEST_PRIORITY, \ | 60 | sporadic_task(e, p, 0, partition, 1, LITMUS_LOWEST_PRIORITY, \ |
61 | RT_CLASS_SOFT, NO_ENFORCEMENT, 1) | ||
62 | #define sporadic_clustered(e, p, cluster, cluster_size) \ | ||
63 | sporadic_task(e, p, 0, cluster, cluster_size, LITMUS_LOWEST_PRIORITY, \ | ||
54 | RT_CLASS_SOFT, NO_ENFORCEMENT, 1) | 64 | RT_CLASS_SOFT, NO_ENFORCEMENT, 1) |
55 | 65 | ||
56 | /* file descriptor attached shared objects support */ | 66 | /* file descriptor attached shared objects support */ |
@@ -98,10 +108,14 @@ void exit_litmus(void); | |||
98 | /* A real-time program. */ | 108 | /* A real-time program. */ |
99 | typedef int (*rt_fn_t)(void*); | 109 | typedef int (*rt_fn_t)(void*); |
100 | 110 | ||
101 | /* These two functions configure the RT task to use enforced exe budgets */ | 111 | /* These two functions configure the RT task to use enforced exe budgets. |
102 | int create_rt_task(rt_fn_t rt_prog, void *arg, int cpu, lt_t wcet, lt_t period, unsigned int prio); | 112 | * Partitioned scheduling: cluster = desired partition, cluster_size = 1 |
103 | int __create_rt_task(rt_fn_t rt_prog, void *arg, int cpu, lt_t wcet, | 113 | * Global scheduling: cluster = 0, cluster_size = 0 |
104 | lt_t period, unsigned int priority, task_class_t cls); | 114 | */ |
115 | int create_rt_task(rt_fn_t rt_prog, void *arg, int cluster, int cluster_size, | ||
116 | lt_t wcet, lt_t period, unsigned int prio); | ||
117 | int __create_rt_task(rt_fn_t rt_prog, void *arg, int cluster, int cluster_size, | ||
118 | lt_t wcet, lt_t period, unsigned int prio, task_class_t cls); | ||
105 | 119 | ||
106 | /* per-task modes */ | 120 | /* per-task modes */ |
107 | enum rt_task_mode_t { | 121 | enum rt_task_mode_t { |
diff --git a/src/litmus.c b/src/litmus.c index b774c63..1110bd4 100644 --- a/src/litmus.c +++ b/src/litmus.c | |||
@@ -92,18 +92,19 @@ task_class_t str2class(const char* str) | |||
92 | #define NS_PER_MS 1000000 | 92 | #define NS_PER_MS 1000000 |
93 | 93 | ||
94 | int sporadic_task(lt_t e, lt_t p, lt_t phase, | 94 | int sporadic_task(lt_t e, lt_t p, lt_t phase, |
95 | int cpu, unsigned int priority, | 95 | int cluster, int cluster_size, unsigned int priority, |
96 | task_class_t cls, | 96 | task_class_t cls, |
97 | budget_policy_t budget_policy, int set_cpu_set) | 97 | budget_policy_t budget_policy, int set_cpu_set) |
98 | { | 98 | { |
99 | return sporadic_task_ns(e * NS_PER_MS, p * NS_PER_MS, phase * NS_PER_MS, | 99 | return sporadic_task_ns(e * NS_PER_MS, p * NS_PER_MS, phase * NS_PER_MS, |
100 | cpu, priority, cls, budget_policy, set_cpu_set); | 100 | cluster, cluster_size, priority, cls, |
101 | budget_policy, set_cpu_set); | ||
101 | } | 102 | } |
102 | 103 | ||
103 | int sporadic_task_ns(lt_t e, lt_t p, lt_t phase, | 104 | int sporadic_task_ns(lt_t e, lt_t p, lt_t phase, |
104 | int cpu, unsigned int priority, | 105 | int cluster, int cluster_size, unsigned int priority, |
105 | task_class_t cls, | 106 | task_class_t cls, |
106 | budget_policy_t budget_policy, int set_cpu_set) | 107 | budget_policy_t budget_policy, int migrate) |
107 | { | 108 | { |
108 | struct rt_task param; | 109 | struct rt_task param; |
109 | int ret; | 110 | int ret; |
@@ -116,15 +117,15 @@ int sporadic_task_ns(lt_t e, lt_t p, lt_t phase, | |||
116 | param.exec_cost = e; | 117 | param.exec_cost = e; |
117 | param.period = p; | 118 | param.period = p; |
118 | param.relative_deadline = p; /* implicit deadline */ | 119 | param.relative_deadline = p; /* implicit deadline */ |
119 | param.cpu = cpu; | 120 | param.cpu = cluster_to_first_cpu(cluster, cluster_size); |
120 | param.cls = cls; | 121 | param.cls = cls; |
121 | param.phase = phase; | 122 | param.phase = phase; |
122 | param.budget_policy = budget_policy; | 123 | param.budget_policy = budget_policy; |
123 | param.priority = priority; | 124 | param.priority = priority; |
124 | 125 | ||
125 | if (set_cpu_set) { | 126 | if (migrate) { |
126 | ret = be_migrate_to_cpu(cpu); | 127 | ret = be_migrate_to_cluster(cluster, cluster_size); |
127 | check("migrate to cpu"); | 128 | check("migrate to cluster"); |
128 | } | 129 | } |
129 | return set_rt_task_param(gettid(), ¶m); | 130 | return set_rt_task_param(gettid(), ¶m); |
130 | } | 131 | } |
@@ -40,11 +40,11 @@ int __launch_rt_task(rt_fn_t rt_prog, void *rt_arg, rt_setup_fn_t setup, | |||
40 | return rt_task; | 40 | return rt_task; |
41 | } | 41 | } |
42 | 42 | ||
43 | int __create_rt_task(rt_fn_t rt_prog, void *arg, int cpu, lt_t wcet, lt_t period, | 43 | int __create_rt_task(rt_fn_t rt_prog, void *arg, int cluster, int cluster_size, |
44 | unsigned int priority, task_class_t class) | 44 | lt_t wcet, lt_t period, unsigned int priority, task_class_t class) |
45 | { | 45 | { |
46 | struct rt_task params; | 46 | struct rt_task params; |
47 | params.cpu = cpu; | 47 | params.cpu = cluster_to_first_cpu(cluster, cluster_size); |
48 | params.period = period; | 48 | params.period = period; |
49 | params.exec_cost = wcet; | 49 | params.exec_cost = wcet; |
50 | params.cls = class; | 50 | params.cls = class; |
@@ -57,9 +57,11 @@ int __create_rt_task(rt_fn_t rt_prog, void *arg, int cpu, lt_t wcet, lt_t period | |||
57 | (rt_setup_fn_t) set_rt_task_param, ¶ms); | 57 | (rt_setup_fn_t) set_rt_task_param, ¶ms); |
58 | } | 58 | } |
59 | 59 | ||
60 | int create_rt_task(rt_fn_t rt_prog, void *arg, int cpu, lt_t wcet, lt_t period, | 60 | int create_rt_task(rt_fn_t rt_prog, void *arg, int cluster, int cluster_size, |
61 | unsigned int priority) { | 61 | lt_t wcet, lt_t period, unsigned int prio) |
62 | return __create_rt_task(rt_prog, arg, cpu, wcet, period, priority, RT_CLASS_HARD); | 62 | { |
63 | return __create_rt_task(rt_prog, arg, cluster, cluster_size, wcet, period, | ||
64 | prio, RT_CLASS_HARD); | ||
63 | } | 65 | } |
64 | 66 | ||
65 | 67 | ||