File: varnumbers.pl

package info (click to toggle)
swi-prolog 6.6.6-1~bpo70%2B1
  • links: PTS, VCS
  • area: main
  • in suites: wheezy-backports
  • size: 82,312 kB
  • sloc: ansic: 322,250; perl: 245,822; sh: 6,651; java: 5,254; makefile: 4,423; cpp: 4,153; ruby: 1,594; yacc: 843; xml: 82; sed: 12; sql: 6
file content (158 lines) | stat: -rw-r--r-- 4,758 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
/*  Part of SWI-Prolog

    Author:        Jan Wielemaker
    E-mail:        J.Wielemaker@cs.vu.nl
    WWW:           http://www.swi-prolog.org
    Copyright (C): 2011, VU University Amsterdam

    This program is free software; you can redistribute it and/or
    modify it under the terms of the GNU General Public License
    as published by the Free Software Foundation; either version 2
    of the License, or (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public
    License along with this library; if not, write to the Free Software
    Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA

    As a special exception, if you link this library with other files,
    compiled with a Free Software compiler, to produce an executable, this
    library does not by itself cause the resulting executable to be covered
    by the GNU General Public License. This exception does not however
    invalidate any other reasons why the executable file might be covered by
    the GNU General Public License.
*/

:- module(varnumbers,
	  [ numbervars/1,			% +Term
	    varnumbers/2,			% +Term, -Copy
	    max_var_number/3,			% +Term, +Start, -Max
	    varnumbers/3			% +Term, +No, -Copy
	  ]).
:- use_module(library(error)).

/** <module> Utilities for numbered terms

This  library  provides  the  inverse   functionality  of  the  built-in
numbervars/3. Note that this library suffers  from the known issues that
'$VAR'(X) is a normal Prolog term and, -unlike the built-in numbervars-,
the inverse predicates do _not_  process   cyclic  terms.  The following
predicate is true for  any  acyclic   term  that  contains no '$VAR'(X),
integer(X) terms and no constraint variables:

  ==
  always_true(X) :-
	copy_term(X, X2),
	numbervars(X),
	varnumbers(X, Copy),
	Copy =@= X2.
  ==

@see	numbervars/4, =@=/2 (variant/2).
@compat	This library was introduced by Quintus and available in
	many related implementations, although not with exactly the
	same set of predicates.
*/

%%	numbervars(+Term) is det.
%
%	Number  variables  in  Term   using    $VAR(N).   Equivalent  to
%	numbervars(Term, 0, _).
%
%	@see numbervars/3, numbervars/4

numbervars(Term) :-
	numbervars(Term, 0, _).

%%	varnumbers(+Term, -Copy) is det.
%
%	Inverse  of  numbervars/1.  Equivalent  to  varnumbers(Term,  0,
%	Copy).

varnumbers(Term, Copy) :-
	varnumbers(Term, 0, Copy).

%%	varnumbers(+Term, +Start, -Copy) is det.
%
%	Inverse of numbervars/3. True when Copy is   a copy of Term with
%	all variables numbered >= Start   consistently replaced by fresh
%	variables. Variables in Term are _shared_  with Copy rather than
%	replaced by fresh variables.
%
%	@error domain_error(acyclic_term, Term) if Term is cyclic.
%	@compat Quintus, SICStus.  Not in YAP version of this library

varnumbers(Term, Min, Copy) :-
	must_be(acyclic, Term),
	MaxStart is Min-1,
	max_var_number(Term, MaxStart, Max),
	NVars is Max-MaxStart,
	(   NVars == 0
	->  Copy = Term
	;   roundup_next_power_two(NVars, Len),
	    functor(Vars, v, Len),
	    varnumbers(Term, MaxStart, Vars, Copy)
	).


varnumbers(Var, _, _, Copy) :-
	var(Var), !,
	Copy = Var.
varnumbers(Var, _, _, Copy) :-
	atomic(Var), !,
	Copy = Var.
varnumbers('$VAR'(I), Min, Vars, Copy) :-
	integer(I),
	I > Min, !,
	Index is I-Min,
	arg(Index, Vars, Copy).
varnumbers(Term, Min, Vars, Copy) :-
	functor(Term, Name, Arity),
	functor(Copy, Name, Arity),
	varnumbers_args(1, Arity, Term, Min, Vars, Copy).

varnumbers_args(I, Arity, Term, Min, Vars, Copy) :-
	I =< Arity, !,
	arg(I, Term, AT),
	arg(I, Copy, CT),
	varnumbers(AT, Min, Vars, CT),
	I2 is I + 1,
	varnumbers_args(I2, Arity, Term, Min, Vars, Copy).
varnumbers_args(_, _, _, _, _, _).


%%	roundup_next_power_two(+Int, -NextPower) is det.
%
%	NextPower is I**2, such that NextPower >= Int.

roundup_next_power_two(1, 1) :- !.
roundup_next_power_two(N, L) :-
	L is 1<<(msb(N-1)+1).

%%	max_var_number(+Term, +Start, -Max) is det.
%
%	True when Max is the  max  of   Start  and  the highest numbered
%	$VAR(N) term.
%
%	@author Vitor Santos Costa
%	@compat YAP

max_var_number(V, Max, Max) :-
	var(V), !.
max_var_number('$VAR'(I), Max0, Max) :-
	integer(I), !,
        Max is max(I,Max0).
max_var_number(S, Max0, Max) :-
        functor(S, _, Ar),
        max_var_numberl(Ar, S, Max0, Max).

max_var_numberl(0, _, Max, Max) :- !.
max_var_numberl(I, T, Max0, Max) :-
	arg(I, T, Arg),
	I2 is I-1,
	max_var_number(Arg, Max0, Max1),
	max_var_numberl(I2, T, Max1, Max).