1
0
This repository has been archived on 2025-03-06. You can view files and clone it, but cannot push or open issues or pull requests.
Jip J. Dekker f2a1c4e389 Squashed 'software/mza/' content from commit f970a59b17
git-subtree-dir: software/mza
git-subtree-split: f970a59b177c13ca3dd8aaef8cc6681d83b7e813
2021-07-11 16:34:30 +10:00

109 lines
4.3 KiB
MiniZinc

/** @group globals.scheduling
Requires that a set of tasks given by start times \a s, durations \a d, and
resource requirements \a r, never require more than a global resource bound
\a b at any one time.
Assumptions:
- forall \p i, \a d[\p i] >= 0 and \a r[\p i] >= 0
Linear version.
*/
%% A global cumulative with SCIP
predicate fzn_cumulative(array[int] of var int: s,
array[int] of int: d,
array[int] of int: r, int: b);
predicate fzn_cumulative(array[int] of var int: s,
array[int] of var int: d,
array[int] of var int: r, var int: b) =
assert(index_set(s) == index_set(d) /\ index_set(s) == index_set(r),
"cumulative: the 3 array arguments must have identical index sets",
if 0==length(s) then true else
assert(lb_array(d) >= 0 /\ lb_array(r) >= 0,
"cumulative: durations and resource usages must be non-negative",
if mzn_in_redundant_constraint() /\ fMZN__IgnoreRedundantCumulative then
true
else
let {
set of int: tasks =
{i | i in index_set(s) where ub(r[i]) > 0 /\ ub(d[i]) > 0 },
set of int: times = dom_array( [ s[i] | i in tasks ] )
} in
if 0==card(tasks) then /*true*/ 0==card(index_set(s)) \/ b>=0
else
if nMZN__UnarySizeMax_cumul>=card(times)*card(tasks) then %%% -- Mem overflow on rcmsp. PARAMETER? TODO
cumulative_time_decomp(s, d, r, b, times)
else
cumulative_task_decomp(s, d, r, b)
endif
endif
endif
) endif );
%% Can be called with a given set of times:
predicate cumulative_set_times(array[int] of var int: s,
array[int] of var int: d,
array[int] of var int: r,
var int: b,
set of int: TIMES01) =
assert(index_set(s) == index_set(d) /\ index_set(s) == index_set(r),
"cumulative: the 3 array arguments must have identical index sets",
assert(lb_array(d) >= 0 /\ lb_array(r) >= 0,
"cumulative: durations and resource usages must be non-negative",
let {
set of int: tasks =
{i | i in index_set(s) where ub(r[i]) > 0 /\ ub(d[i]) > 0 },
set of int: times = dom_array( [ s[i] | i in tasks ] )
intersect TIMES01
} in
if false then %%% 100>=card(times) then %% PARAMETER ?
cumulative_time_decomp(s, d, r, b, times)
else
cumulative_task_decomp(s, d, r, b)
endif ));
predicate cumulative_time_decomp(array[int] of var int: s,
array[int] of var int: d,
array[int] of var int: r,
var int: b,
set of int: TIMES01) =
let {
set of int: tasks =
{i | i in index_set(s) where ub(r[i]) > 0 /\ ub(d[i]) > 0 },
set of int: times =
{ i | i in min([ lb(s[i]) | i in tasks ]) ..
max([ ub(s[i]) + ub(d[i]) | i in tasks ]) where i in TIMES01 }
}
in
% my_trace(" cumul_time: " ++ show(card(tasks)) ++ " tasks, " ++ show(card(times)) ++ " times\n") /\
forall( t in times ) (
b >= sum( i in tasks ) (
if is_fixed(d[i]) then
bool2int( s[i] in t-fix(d[i])+1..t )
else
bool2int( s[i] <= t /\ t < s[i] + d[i] )
endif
* r[i]
)
);
predicate cumulative_task_decomp(array[int] of var int: s,
array[int] of var int: d,
array[int] of var int: r,
var int: b) =
let {
set of int: tasks =
{i | i in index_set(s) where ub(r[i]) > 0 /\ ub(d[i]) > 0 }
}
in
% my_trace(" cumul_tasks: " ++ show(card(tasks)) ++ " tasks\n") /\
forall( j in tasks) (
b-r[j] >= sum( i in tasks where i != j
/\ lb(s[i])<=ub(s[j]) /\ lb(s[j])<ub(s[i]+d[i]) %% -- seems slower on mspsp ???
) (
r[i] * %% r[i] * !
bool2int( s[i] <= s[j] /\ s[j] < s[i]+d[i] )) )
;