diff options
| author | Bjoern Brandenburg <bbb@mpi-sws.org> | 2012-12-09 11:39:41 -0500 |
|---|---|---|
| committer | Bjoern Brandenburg <bbb@mpi-sws.org> | 2013-02-12 06:55:16 -0500 |
| commit | 308ff37a60a61e158bfe108bfcea0bf3b71e942d (patch) | |
| tree | bb8abf906bdd5823aeefa3c89668d38c6f1bc3d8 /native/src/blocking | |
| parent | adfef1cb9a8960690fb85cc0b99e8fe266e173eb (diff) | |
Add LP-based blocking analysis for FMLP+
Diffstat (limited to 'native/src/blocking')
| -rw-r--r-- | native/src/blocking/linprog/lp_fmlp.cpp | 286 |
1 files changed, 286 insertions, 0 deletions
diff --git a/native/src/blocking/linprog/lp_fmlp.cpp b/native/src/blocking/linprog/lp_fmlp.cpp new file mode 100644 index 0000000..75e2186 --- /dev/null +++ b/native/src/blocking/linprog/lp_fmlp.cpp | |||
| @@ -0,0 +1,286 @@ | |||
| 1 | #include <iostream> | ||
| 2 | #include <set> | ||
| 3 | #include <algorithm> | ||
| 4 | |||
| 5 | #include "lp_common.h" | ||
| 6 | #include "blocking.h" | ||
| 7 | #include "stl-hashmap.h" | ||
| 8 | |||
| 9 | #include "cpu_time.h" | ||
| 10 | |||
| 11 | typedef hashmap<unsigned int, unsigned int> BlockingLimits; | ||
| 12 | |||
| 13 | // Constraint 14 | ||
| 14 | static void add_fifo_cluster_constraints( | ||
| 15 | VarMapper& vars, | ||
| 16 | const ResourceSharingInfo& info, | ||
| 17 | const TaskInfo& ti, | ||
| 18 | LinearProgram& lp) | ||
| 19 | { | ||
| 20 | foreach_remote_task(info.get_tasks(), ti, tx) | ||
| 21 | { | ||
| 22 | unsigned int t = tx->get_id(); | ||
| 23 | |||
| 24 | // compute direct blocking opportunities, not counting Tx | ||
| 25 | BlockingLimits per_resource_remote; | ||
| 26 | |||
| 27 | foreach(ti.get_requests(), req) | ||
| 28 | per_resource_remote[req->get_resource_id()] = 0; | ||
| 29 | |||
| 30 | foreach_local_task_except(info.get_tasks(), *tx, ty) | ||
| 31 | { | ||
| 32 | foreach(ty->get_requests(), req) | ||
| 33 | { | ||
| 34 | unsigned int u = req->get_resource_id(); | ||
| 35 | if (per_resource_remote.find(u) != | ||
| 36 | per_resource_remote.end()) | ||
| 37 | per_resource_remote[u] += req->get_max_num_requests(ti.get_response()); | ||
| 38 | } | ||
| 39 | } | ||
| 40 | |||
| 41 | unsigned int total_limit = 0; | ||
| 42 | foreach(ti.get_requests(), req) | ||
| 43 | { | ||
| 44 | unsigned int u = req->get_resource_id(); | ||
| 45 | total_limit += std::min(req->get_num_requests(), | ||
| 46 | per_resource_remote[u]); | ||
| 47 | } | ||
| 48 | |||
| 49 | LinearExpression *exp = new LinearExpression(); | ||
| 50 | |||
| 51 | foreach(tx->get_requests(), request) | ||
| 52 | { | ||
| 53 | unsigned int q = request->get_resource_id(); | ||
| 54 | |||
| 55 | foreach_request_instance(*request, ti, v) | ||
| 56 | { | ||
| 57 | unsigned int var_id; | ||
| 58 | var_id = vars.lookup(t, q, v, BLOCKING_INDIRECT); | ||
| 59 | exp->add_var(var_id); | ||
| 60 | } | ||
| 61 | } | ||
| 62 | |||
| 63 | lp.add_inequality(exp, total_limit); | ||
| 64 | } | ||
| 65 | } | ||
| 66 | |||
| 67 | // Constraint 13 | ||
| 68 | static void add_total_fifo_constraints( | ||
| 69 | VarMapper& vars, | ||
| 70 | const ResourceSharingInfo& info, | ||
| 71 | const TaskInfo& ti, | ||
| 72 | LinearProgram& lp, | ||
| 73 | BlockingLimits &per_cluster_counts) | ||
| 74 | { | ||
| 75 | |||
| 76 | unsigned int total_num_requests = 0; | ||
| 77 | |||
| 78 | foreach(ti.get_requests(), req) | ||
| 79 | total_num_requests += req->get_num_requests(); | ||
| 80 | |||
| 81 | foreach_task_except(info.get_tasks(), ti, tx) | ||
| 82 | { | ||
| 83 | unsigned int t = tx->get_id(); | ||
| 84 | |||
| 85 | LinearExpression *exp = new LinearExpression(); | ||
| 86 | |||
| 87 | //for all requests accessed by tx | ||
| 88 | foreach(tx->get_requests(), request) | ||
| 89 | { | ||
| 90 | unsigned int q = request->get_resource_id(); | ||
| 91 | |||
| 92 | foreach_request_instance(*request, ti, v) | ||
| 93 | { | ||
| 94 | unsigned int var_id; | ||
| 95 | var_id = vars.lookup(t, q, v, BLOCKING_DIRECT); | ||
| 96 | exp->add_var(var_id); | ||
| 97 | var_id = vars.lookup(t, q, v, BLOCKING_INDIRECT); | ||
| 98 | exp->add_var(var_id); | ||
| 99 | } | ||
| 100 | } | ||
| 101 | |||
| 102 | lp.add_inequality(exp, per_cluster_counts[tx->get_cluster()]); | ||
| 103 | } | ||
| 104 | } | ||
| 105 | |||
| 106 | // Constraint 12 | ||
| 107 | static void add_fifo_resource_constraints( | ||
| 108 | VarMapper& vars, | ||
| 109 | const ResourceSharingInfo& info, | ||
| 110 | const TaskInfo& ti, | ||
| 111 | LinearProgram& lp) | ||
| 112 | { | ||
| 113 | BlockingLimits per_resource_counts; | ||
| 114 | |||
| 115 | foreach(ti.get_requests(), req) | ||
| 116 | per_resource_counts[req->get_resource_id()] = req->get_num_requests(); | ||
| 117 | |||
| 118 | foreach_task_except(info.get_tasks(), ti, tx) | ||
| 119 | { | ||
| 120 | unsigned int t = tx->get_id(); | ||
| 121 | //for all requests accessed by tx | ||
| 122 | foreach(tx->get_requests(), request) | ||
| 123 | { | ||
| 124 | unsigned int q = request->get_resource_id(); | ||
| 125 | LinearExpression *exp = new LinearExpression(); | ||
| 126 | |||
| 127 | foreach_request_instance(*request, ti, v) | ||
| 128 | { | ||
| 129 | unsigned int var_id; | ||
| 130 | var_id = vars.lookup(t, q, v, BLOCKING_DIRECT); | ||
| 131 | exp->add_var(var_id); | ||
| 132 | } | ||
| 133 | |||
| 134 | lp.add_inequality(exp, per_resource_counts[q]); | ||
| 135 | } | ||
| 136 | } | ||
| 137 | } | ||
| 138 | |||
| 139 | static BlockingLimits count_blocking_opportunities( | ||
| 140 | const ResourceSharingInfo &info, | ||
| 141 | const TaskInfo& ti) | ||
| 142 | { | ||
| 143 | BlockingLimits per_cluster_counts; | ||
| 144 | |||
| 145 | foreach_task_except(info.get_tasks(), ti, tx) | ||
| 146 | { | ||
| 147 | unsigned int c = tx->get_cluster(); | ||
| 148 | |||
| 149 | if (per_cluster_counts.find(c) == per_cluster_counts.end()) | ||
| 150 | { | ||
| 151 | // compute direct blocking opportunities on Tx's cluster | ||
| 152 | BlockingLimits per_resource_remote; | ||
| 153 | |||
| 154 | foreach(ti.get_requests(), req) | ||
| 155 | per_resource_remote[req->get_resource_id()] = 0; | ||
| 156 | |||
| 157 | foreach_local_task(info.get_tasks(), *tx, ty) | ||
| 158 | { | ||
| 159 | foreach(ty->get_requests(), req) | ||
| 160 | { | ||
| 161 | unsigned int u = req->get_resource_id(); | ||
| 162 | if (per_resource_remote.find(u) != | ||
| 163 | per_resource_remote.end()) | ||
| 164 | per_resource_remote[u] += req->get_max_num_requests(ti.get_response()); | ||
| 165 | } | ||
| 166 | } | ||
| 167 | |||
| 168 | per_cluster_counts[c] = 0; | ||
| 169 | foreach(ti.get_requests(), req) | ||
| 170 | { | ||
| 171 | unsigned int u = req->get_resource_id(); | ||
| 172 | per_cluster_counts[c] += std::min(req->get_num_requests(), | ||
| 173 | per_resource_remote[u]); | ||
| 174 | } | ||
| 175 | } | ||
| 176 | } | ||
| 177 | |||
| 178 | return per_cluster_counts; | ||
| 179 | } | ||
| 180 | |||
| 181 | static void add_fmlp_constraints( | ||
| 182 | VarMapper& vars, | ||
| 183 | const ResourceSharingInfo& info, | ||
| 184 | const TaskInfo& ti, | ||
| 185 | LinearProgram& lp) | ||
| 186 | { | ||
| 187 | // Constraint 1 | ||
| 188 | add_mutex_constraints(vars, info, ti, lp); | ||
| 189 | // Constraint 9 | ||
| 190 | add_local_higher_priority_constraints_shm(vars, info, ti, lp); | ||
| 191 | // Constraint 10 | ||
| 192 | add_topology_constraints_shm(vars, info, ti, lp); | ||
| 193 | // Constraint 11 | ||
| 194 | add_local_lower_priority_constraints_shm(vars, info, ti, lp); | ||
| 195 | |||
| 196 | BlockingLimits per_cluster_counts; | ||
| 197 | per_cluster_counts = count_blocking_opportunities(info, ti); | ||
| 198 | |||
| 199 | // Constraint 12 | ||
| 200 | add_fifo_resource_constraints(vars, info, ti, lp); | ||
| 201 | // Constraint 13 | ||
| 202 | add_total_fifo_constraints(vars, info, ti, lp, per_cluster_counts); | ||
| 203 | // Constraint 14 | ||
| 204 | add_fifo_cluster_constraints(vars, info, ti, lp); | ||
| 205 | } | ||
| 206 | |||
| 207 | static void apply_fmlp_bounds_for_task( | ||
| 208 | unsigned int i, | ||
| 209 | BlockingBounds& bounds, | ||
| 210 | const ResourceSharingInfo& info) | ||
| 211 | { | ||
| 212 | LinearProgram lp; | ||
| 213 | VarMapper vars; | ||
| 214 | const TaskInfo& ti = info.get_tasks()[i]; | ||
| 215 | LinearExpression *local_obj = new LinearExpression(); | ||
| 216 | |||
| 217 | #if DEBUG_LP_OVERHEADS >= 2 | ||
| 218 | static DEFINE_CPU_CLOCK(model_gen_cost); | ||
| 219 | static DEFINE_CPU_CLOCK(solver_cost); | ||
| 220 | |||
| 221 | std::cout << "---- " << __FUNCTION__ << " ----" << std::endl; | ||
| 222 | |||
| 223 | model_gen_cost.start(); | ||
| 224 | #endif | ||
| 225 | |||
| 226 | set_blocking_objective_part_shm(vars, info, ti, lp, local_obj); | ||
| 227 | vars.seal(); | ||
| 228 | |||
| 229 | add_fmlp_constraints(vars, info, ti, lp); | ||
| 230 | |||
| 231 | #if DEBUG_LP_OVERHEADS >= 2 | ||
| 232 | model_gen_cost.stop(); | ||
| 233 | std::cout << model_gen_cost << std::endl; | ||
| 234 | solver_cost.start(); | ||
| 235 | #endif | ||
| 236 | |||
| 237 | Solution *sol = linprog_solve(lp, vars.get_num_vars()); | ||
| 238 | |||
| 239 | #if DEBUG_LP_OVERHEADS >= 2 | ||
| 240 | solver_cost.stop(); | ||
| 241 | std::cout << solver_cost << std::endl; | ||
| 242 | #endif | ||
| 243 | assert(sol != NULL); | ||
| 244 | |||
| 245 | Interference total, remote, local; | ||
| 246 | |||
| 247 | total.total_length = sol->evaluate(*lp.get_objective()); | ||
| 248 | local.total_length = sol->evaluate(*local_obj); | ||
| 249 | remote.total_length = total.total_length - local.total_length; | ||
| 250 | |||
| 251 | bounds[i] = total; | ||
| 252 | bounds.set_remote_blocking(i, remote); | ||
| 253 | bounds.set_local_blocking(i, local); | ||
| 254 | |||
| 255 | delete local_obj; | ||
| 256 | delete sol; | ||
| 257 | } | ||
| 258 | |||
| 259 | |||
| 260 | static BlockingBounds* _lp_fmlp_bounds(const ResourceSharingInfo& info) | ||
| 261 | { | ||
| 262 | BlockingBounds* results = new BlockingBounds(info); | ||
| 263 | |||
| 264 | for (unsigned int i = 0; i < info.get_tasks().size(); i++) | ||
| 265 | apply_fmlp_bounds_for_task(i, *results, info); | ||
| 266 | |||
| 267 | return results; | ||
| 268 | } | ||
| 269 | |||
| 270 | BlockingBounds* lp_part_fmlp_bounds(const ResourceSharingInfo& info) | ||
| 271 | { | ||
| 272 | #if DEBUG_LP_OVERHEADS >= 1 | ||
| 273 | static DEFINE_CPU_CLOCK(cpu_costs); | ||
| 274 | |||
| 275 | cpu_costs.start(); | ||
| 276 | #endif | ||
| 277 | |||
| 278 | BlockingBounds *results = _lp_fmlp_bounds(info); | ||
| 279 | |||
| 280 | #if DEBUG_LP_OVERHEADS >= 1 | ||
| 281 | cpu_costs.stop(); | ||
| 282 | std::cout << cpu_costs << std::endl; | ||
| 283 | #endif | ||
| 284 | |||
| 285 | return results; | ||
| 286 | } | ||
