Skip to content

Commit 1cf5de8

Browse files
authored
Merge pull request #2970 from notoria/clpz
Make `(mod)/2` stronger in CLP(Z)
2 parents 686f20d + 4b480a7 commit 1cf5de8

File tree

1 file changed

+39
-4
lines changed

1 file changed

+39
-4
lines changed

src/lib/clpz.pl

Lines changed: 39 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -5088,19 +5088,45 @@
50885088
; nonvar(Z), nonvar(X) ->
50895089
( Z > 0 ->
50905090
( X < 0 -> true
5091-
; X >= Z
5091+
; X >= Z,
5092+
% due to X = Z+Y*_ and Y > Z
5093+
( X-Z > 0 ->
5094+
X-Z > Z
5095+
; true
5096+
)
50925097
)
50935098
; Z < 0 ->
50945099
( X > 0 -> true
5095-
; X =< Z
5100+
; X =< Z,
5101+
% due to X = Z+Y*_ and Y < Z
5102+
( X-Z < 0 ->
5103+
X-Z < Z
5104+
; true
5105+
)
50965106
)
50975107
; Z =:= 0 % Multiple solutions so do nothing special.
50985108
),
50995109
( { fd_get(Y, _, _, n(YU), _),
51005110
YU < X, X =< 0 } -> kill(MState), Z =:= X
51015111
; { fd_get(Y, _, n(YL), _, _),
51025112
YL > X, X >= 0 } -> kill(MState), Z =:= X
5103-
; ( Z > 0 ->
5113+
; ( Z > 0, X < 0 ->
5114+
{ fd_get(Y, YD, YPs),
5115+
YMin is Z+1,
5116+
YMax is Z-X,
5117+
domain_remove_smaller_than(YD, YMin, YD1),
5118+
domain_remove_greater_than(YD1, YMax, YD2) },
5119+
fd_put(Y, YD2, YPs)
5120+
% queue_goal((Y #> Z, Y #=< Z-X))
5121+
; Z < 0, X > 0 ->
5122+
{ fd_get(Y, YD, YPs),
5123+
YMax is Z-1,
5124+
YMin is Z-X,
5125+
domain_remove_greater_than(YD, YMax, YD1),
5126+
domain_remove_smaller_than(YD1, YMin, YD2) },
5127+
fd_put(Y, YD2, YPs)
5128+
% queue_goal((Y #< Z, Y #>= Z-X))
5129+
; Z > 0 ->
51045130
{ fd_get(Y, YD, YPs),
51055131
YMin is Z + 1,
51065132
domain_remove_smaller_than(YD, YMin, YD1) },
@@ -5112,7 +5138,16 @@
51125138
domain_remove_greater_than(YD, YMax, YD1) },
51135139
fd_put(Y, YD1, YPs)
51145140
% queue_goal(Y #< Z)
5115-
; true
5141+
; Z =:= 0,
5142+
( X =:= 0 ->
5143+
kill(MState) % trivial
5144+
; % only 4 solutions {-abs(X),-1,1,abs(X)}
5145+
{ YL is -abs(X), YU is abs(X),
5146+
fd_get(Y, YD0, YPs),
5147+
domain_remove_smaller_than(YD0, YL, YD1),
5148+
domain_remove_greater_than(YD1, YU, YD) },
5149+
fd_put(Y, YD, YPs)
5150+
)
51165151
)
51175152
)
51185153
; run_propagator(pmodz(X,Y,Z), MState),

0 commit comments

Comments
 (0)