Files
mercury/tests/hard_coded/boyer.m
Zoltan Somogyi 2bd7c5ee3e Rename X's aux modules as X_helper_N in hard_coded.
tests/hard_coded/*.m:
    Rename modules as mentioned above.

    In a few cases, where the main module's name itself had a suffix,
    such as "_mod_a" or "_main", remove that suffix. This entails
    renaming the .exp file as well. (In some cases, this meant that
    the name of a helper module was "taken over" by the main module
    of the test case.)

    Update all references to the moved modules.

    General updates to programming style, such as

    - replacing DCG notation with state var notation
    - replacing (C->T;E) with (if C then T else E)
    - moving pred/func declarations to just before their code
    - replacing io.write/io.nl sequences with io.write_line
    - replacing io.print/io.nl sequences with io.print_line
    - fixing too-long lines
    - fixing grammar errors in comments

tests/hard_coded/Mmakefile:
tests/hard_coded/Mercury.options:
    Update all references to the moved modules.

    Enable the constant_prop_int test case. The fact that it wasn't enabled
    before is probably an accident. (When constant_prop_int.m was created,
    the test case was added to a list in the Mmakefile, but that list
    was later removed due to never being referenced.)

tests/hard_coded/constant_prop_int.{m,exp}:
    Delete the calls to shift operations with negative shift amounts,
    since we have added a compile-time error for these since the test
    was originally created.
2023-06-16 08:33:22 +02:00

366 lines
8.3 KiB
Mathematica

%---------------------------------------------------------------------------%
% vim: ts=4 sw=4 et ft=mercury
%---------------------------------------------------------------------------%
%
% This is a regression test: it triggered a performance bug in type inference.
%
% generated: 20 November 1989
% option(s):
%
% boyer
% Evan Tick (from Lisp version by R. P. Gabriel)
% November 1985
% prove arithmetic theorem
% in Mercury by Bart Demoen - started 17 Jan 1997
:- module boyer.
:- interface.
:- import_module io.
:- pred main(io::di, io::uo) is det.
:- implementation.
:- import_module int.
:- import_module list.
main(!IO) :-
b(3, !IO).
% :- pred b(int, io, io).
:- mode b(in, di, uo) is det.
b(X, !IO) :-
( if X > 0 then
boyer(!IO),
Y = X - 1,
b(Y, !IO)
else
true
).
:- pred boyer(io::di, io::uo) is det.
boyer(!IO) :-
wff(Wff),
io.write_string("rewriting...", !IO),
rewrite(Wff, NewWff),
io.write_string("proving...", !IO),
( if tautology(NewWff, [], []) then
io.write_string("done...\n", !IO)
else
io.write_string("not done ...", !IO)
).
:- type type_wff
---> implies(type_wff, type_wff)
; and(type_wff, type_wff)
; f(type_wff)
; my_plus(type_wff, type_wff)
; equal1(type_wff, type_wff)
; append(type_wff, type_wff)
; lessp(type_wff, type_wff)
; my_times(type_wff, type_wff)
; reverse(type_wff)
; difference(type_wff, type_wff)
; remainder(type_wff, type_wff)
; b_member(type_wff, type_wff)
; b_length(type_wff)
; if(type_wff, type_wff, type_wff)
; a1
; b1
; c1
; d1
; t1
; f1
; x1
; y1
; zero
; nil.
:- pred wff(type_wff).
:- mode wff(out) is det.
wff(implies(and(implies(X, Y),
and(implies(Y, Z),
and(implies(Z, U),
implies(U, W)))),
implies(X, W))) :-
X = f(my_plus(my_plus(a1, b1), my_plus(c1, zero))),
Y = f(my_times(my_times(a1, b1), my_plus(c1, d1))),
Z = f(reverse(append(append(a1, b1), nil))),
U = equal1(my_plus(a1, b1), difference(x1, y1)),
W = lessp(remainder(a1, b1), b_member(a1, b_length(b1))).
:- pred tautology(type_wff, list(type_wff), list(type_wff)).
:- mode tautology(in, in, in) is semidet.
tautology(Wff, Tlist, Flist) :-
( if truep(Wff, Tlist) then
true
else if falsep(Wff, Flist) then
fail
else if Wff = if(If, Then, Else) then
( if truep(If, Tlist) then
tautology(Then, Tlist, Flist)
else if falsep(If, Flist) then
tautology(Else, Tlist, Flist)
else
tautology(Then, [If | Tlist], Flist),
tautology(Else, Tlist, [If | Flist])
)
else
fail
).
% :- pred rewrite(type_wff, type_wff).
:- mode rewrite(in, out) is det.
rewrite(a1, a1).
rewrite(b1, b1).
rewrite(c1, c1).
rewrite(d1, d1).
rewrite(f1, f1).
rewrite(t1, t1).
rewrite(x1, x1).
rewrite(y1, y1).
rewrite(zero, zero).
rewrite(nil, nil).
rewrite(lessp(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = lessp(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(b_member(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = b_member(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(remainder(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = remainder(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(my_plus(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = my_plus(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(and(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = and(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(equal1(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = equal1(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(difference(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = difference(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(append(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = append(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(my_times(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = my_times(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(implies(X1, X2), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
Mid = implies(Y1, Y2),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(b_length(X1), New) :-
rewrite(X1, Y1),
Mid = b_length(Y1),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(f(X1), New) :-
rewrite(X1, Y1),
Mid = f(Y1),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(reverse(X1), New) :-
rewrite(X1, Y1),
Mid = reverse(Y1),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
rewrite(if(X1, X2, X3), New) :-
rewrite(X1, Y1),
rewrite(X2, Y2),
rewrite(X3, Y3),
Mid = if(Y1, Y2, Y3),
( if equal(Mid, Next) then
rewrite(Next, New)
else
New = Mid
).
% :- pred rewrite_args(list(type_wff), list(type_wff)).
:- mode rewrite_args(in, out) is det.
rewrite_args([], []).
rewrite_args([A | RA], [B | RB]) :-
rewrite(A, B),
rewrite_args(RA, RB).
:- pred truep(type_wff, list(type_wff)).
:- mode truep(in, in) is semidet.
truep(Wff, List) :-
( if Wff = t1 then
true
else
member_chk(Wff, List)
).
% :- pred falsep(type_wff, list(type_wff)).
:- mode falsep(in, in) is semidet.
falsep(Wff, List) :-
( if Wff = f1 then
true
else
member_chk(Wff, List)
).
% :- pred member_chk(type_wff, list(type_wff)).
:- mode member_chk(in, in) is semidet.
member_chk(X, [Y | T]) :-
( if X = Y then
true
else
member_chk(X, T)
).
% :- pred equal(type_wff, type_wff).
:- mode equal(in, out) is semidet.
equal(and(P, Q), if(P, if(Q, t1, f1), f1)).
equal(append(append(X, Y), Z), append(X, append(Y, Z))).
equal(difference(A, B), C) :-
difference1(A, B, C).
equal(equal1(A, B), C) :-
eq(A, B, C).
equal(if(if(A, B, C), D, E), if(A, if(B, D, E), if(C, D, E))).
equal(implies(P, Q), if(P, if(Q, t1, f1), t1)).
equal(b_length(A), B) :-
my_length(A, B).
equal(lessp(A, B), C) :-
lessp1(A, B, C).
equal(my_plus(A, B), C) :-
plus1(A, B, C).
equal(remainder(A, B), C) :-
remainder1(A, B, C).
equal(reverse(append(A, B)), append(reverse(B), reverse(A))).
% :- pred difference1(type_wff, type_wff, type_wff).
:- mode difference1(in, in, out) is semidet.
difference1(X, Y, Z) :-
( if X = Y then
Z = zero
else
( if X = my_plus(A, B), Y = my_plus(A, C) then
Z = difference(B, C)
else if X = my_plus(B, my_plus(Y, C)) then
Z = my_plus(B, C)
else
fail
)
).
% :- pred eq(type_wff, type_wff, type_wff).
:- mode eq(in, in, out) is semidet.
eq(append(A, B), append(A, C), equal1(B, C)).
eq(lessp(X, Y), Z, if(lessp(X, Y), equal1(t1, Z), equal1(f1, Z))).
% :- pred my_length(type_wff, type_wff).
:- mode my_length(in, out) is semidet.
my_length(reverse(X), b_length(X)).
% :- pred lessp1(type_wff, type_wff, type_wff).
:- mode lessp1(in, in, out) is semidet.
lessp1(my_plus(X, Y), my_plus(X, Z), lessp(Y, Z)).
% :- pred plus1(type_wff, type_wff, type_wff).
:- mode plus1(in, in, out) is semidet.
plus1(my_plus(X, Y), Z, my_plus(X, my_plus(Y, Z))).
% :- pred remainder1(type_wff, type_wff, type_wff).
:- mode remainder1(in, in, out) is semidet.
remainder1(U, V, zero) :-
( if U = V then
true
else
U = my_times(A, B),
( if B = V then
true
else
A = V
)
).