| 
									
										
										
										
											2008-03-13 17:43:13 +00:00
										 |  |  | /*  $Id: chr_compiler_utility.pl,v 1.4 2008-03-13 17:43:13 vsc Exp $ | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Part of CHR (Constraint Handling Rules) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Author:        Tom Schrijvers | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  |     E-mail:        Tom.Schrijvers@cs.kuleuven.be | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  |     WWW:           http://www.swi-prolog.org | 
					
						
							|  |  |  |     Copyright (C): 2005-2006, K.U. Leuven | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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 Lesser General Public | 
					
						
							|  |  |  |     License along with this library; if not, write to the Free Software | 
					
						
							|  |  |  |     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  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(chr_compiler_utility, | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | 	[ time/2 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 	, replicate/3 | 
					
						
							|  |  |  | 	, pair_all_with/3 | 
					
						
							|  |  |  | 	, conj2list/2 | 
					
						
							|  |  |  | 	, list2conj/2 | 
					
						
							|  |  |  | 	, disj2list/2 | 
					
						
							|  |  |  | 	, list2disj/2 | 
					
						
							|  |  |  | 	, variable_replacement/3 | 
					
						
							|  |  |  | 	, variable_replacement/4 | 
					
						
							|  |  |  | 	, identical_rules/2 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | 	, identical_guarded_rules/2 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 	, copy_with_variable_replacement/3 | 
					
						
							|  |  |  | 	, my_term_copy/3 | 
					
						
							|  |  |  | 	, my_term_copy/4 | 
					
						
							|  |  |  | 	, atom_concat_list/2 | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | 	, atomic_concat/3 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 	, init/2 | 
					
						
							|  |  |  | 	, member2/3 | 
					
						
							|  |  |  | 	, select2/6 | 
					
						
							|  |  |  | 	, set_elems/2 | 
					
						
							|  |  |  | 	, instrument_goal/4 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | 	, sort_by_key/3 | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | 	, arg1/3 | 
					
						
							|  |  |  | 	, wrap_in_functor/3 | 
					
						
							|  |  |  | 	, tree_set_empty/1 | 
					
						
							|  |  |  | 	, tree_set_memberchk/2 | 
					
						
							|  |  |  | 	, tree_set_add/3 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 	]). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | :- use_module(pairlist). | 
					
						
							|  |  |  | :- use_module(library(lists), [permutation/2]). | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | :- use_module(library(assoc)). | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | %% SICStus begin | 
					
						
							|  |  |  | %% use_module(library(terms),[term_variables/2]). | 
					
						
							|  |  |  | %% SICStus end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | % time(Phase,Goal) :- | 
					
						
							|  |  |  | % 	statistics(runtime,[T1|_]), | 
					
						
							|  |  |  | % 	call(Goal), | 
					
						
							|  |  |  | % 	statistics(runtime,[T2|_]), | 
					
						
							|  |  |  | % 	T is T2 - T1, | 
					
						
							|  |  |  | % 	format('    ~w ~46t ~D~80| ms\n',[Phase,T]), | 
					
						
							|  |  |  | % 	deterministic(Det), | 
					
						
							|  |  |  | % 	( Det == true -> | 
					
						
							|  |  |  | % 		true | 
					
						
							|  |  |  | % 	; | 
					
						
							|  |  |  | % 		format('\t\tNOT DETERMINISTIC!\n',[]) | 
					
						
							|  |  |  | % 	). | 
					
						
							|  |  |  | time(_,Goal) :- call(Goal). | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | replicate(N,E,L) :- | 
					
						
							|  |  |  | 	( N =< 0 -> | 
					
						
							|  |  |  | 		L = [] | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  | 		L = [E|T], | 
					
						
							|  |  |  | 		M is N - 1, | 
					
						
							|  |  |  | 		replicate(M,E,T) | 
					
						
							|  |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | pair_all_with([],_,[]). | 
					
						
							|  |  |  | pair_all_with([X|Xs],Y,[X-Y|Rest]) :- | 
					
						
							|  |  |  | 	pair_all_with(Xs,Y,Rest). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | conj2list(Conj,L) :-				%% transform conjunctions to list | 
					
						
							|  |  |  |   conj2list(Conj,L,[]). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | conj2list(Var,L,T) :- | 
					
						
							|  |  |  | 	var(Var), !, | 
					
						
							|  |  |  | 	L = [Var|T]. | 
					
						
							|  |  |  | conj2list(true,L,L) :- !. | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | conj2list(Conj,L,T) :- | 
					
						
							|  |  |  |   Conj = (G1,G2), !, | 
					
						
							|  |  |  |   conj2list(G1,L,T1), | 
					
						
							|  |  |  |   conj2list(G2,T1,T). | 
					
						
							|  |  |  | conj2list(G,[G | T],T). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | disj2list(Conj,L) :-				%% transform disjunctions to list | 
					
						
							|  |  |  |   disj2list(Conj,L,[]). | 
					
						
							|  |  |  | disj2list(Conj,L,T) :- | 
					
						
							|  |  |  |   Conj = (fail;G2), !, | 
					
						
							|  |  |  |   disj2list(G2,L,T). | 
					
						
							|  |  |  | disj2list(Conj,L,T) :- | 
					
						
							|  |  |  |   Conj = (G1;G2), !, | 
					
						
							|  |  |  |   disj2list(G1,L,T1), | 
					
						
							|  |  |  |   disj2list(G2,T1,T). | 
					
						
							|  |  |  | disj2list(G,[G | T],T). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | list2conj([],true). | 
					
						
							|  |  |  | list2conj([G],X) :- !, X = G. | 
					
						
							|  |  |  | list2conj([G|Gs],C) :- | 
					
						
							|  |  |  | 	( G == true ->				%% remove some redundant trues | 
					
						
							|  |  |  | 		list2conj(Gs,C) | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  | 		C = (G,R), | 
					
						
							|  |  |  | 		list2conj(Gs,R) | 
					
						
							|  |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | list2disj([],fail). | 
					
						
							|  |  |  | list2disj([G],X) :- !, X = G. | 
					
						
							|  |  |  | list2disj([G|Gs],C) :- | 
					
						
							|  |  |  | 	( G == fail ->				%% remove some redundant fails | 
					
						
							|  |  |  | 		list2disj(Gs,C) | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  | 		C = (G;R), | 
					
						
							|  |  |  | 		list2disj(Gs,R) | 
					
						
							|  |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | % check wether two rules are identical | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | identical_guarded_rules(rule(H11,H21,G1,_),rule(H12,H22,G2,_)) :- | 
					
						
							|  |  |  |    G1 == G2, | 
					
						
							|  |  |  |    permutation(H11,P1), | 
					
						
							|  |  |  |    P1 == H12, | 
					
						
							|  |  |  |    permutation(H21,P2), | 
					
						
							|  |  |  |    P2 == H22. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | identical_rules(rule(H11,H21,G1,B1),rule(H12,H22,G2,B2)) :- | 
					
						
							|  |  |  |    G1 == G2, | 
					
						
							|  |  |  |    identical_bodies(B1,B2), | 
					
						
							|  |  |  |    permutation(H11,P1), | 
					
						
							|  |  |  |    P1 == H12, | 
					
						
							|  |  |  |    permutation(H21,P2), | 
					
						
							|  |  |  |    P2 == H22. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | identical_bodies(B1,B2) :- | 
					
						
							|  |  |  |    ( B1 = (X1 = Y1), | 
					
						
							|  |  |  |      B2 = (X2 = Y2) -> | 
					
						
							|  |  |  |      ( X1 == X2, | 
					
						
							|  |  |  |        Y1 == Y2 | 
					
						
							|  |  |  |      ; X1 == Y2, | 
					
						
							|  |  |  |        X2 == Y1 | 
					
						
							|  |  |  |      ), | 
					
						
							|  |  |  |      ! | 
					
						
							|  |  |  |    ; B1 == B2 | 
					
						
							|  |  |  |    ). | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | % replace variables in list | 
					
						
							|  |  |  |     | 
					
						
							|  |  |  | copy_with_variable_replacement(X,Y,L) :- | 
					
						
							|  |  |  |    ( var(X) -> | 
					
						
							|  |  |  |      ( lookup_eq(L,X,Y) -> | 
					
						
							|  |  |  |        true | 
					
						
							|  |  |  |      ; X = Y | 
					
						
							|  |  |  |      ) | 
					
						
							|  |  |  |    ; functor(X,F,A), | 
					
						
							|  |  |  |      functor(Y,F,A), | 
					
						
							|  |  |  |      X =.. [_|XArgs], | 
					
						
							|  |  |  |      Y =.. [_|YArgs], | 
					
						
							|  |  |  |      copy_with_variable_replacement_l(XArgs,YArgs,L) | 
					
						
							|  |  |  |    ). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | copy_with_variable_replacement_l([],[],_). | 
					
						
							|  |  |  | copy_with_variable_replacement_l([X|Xs],[Y|Ys],L) :- | 
					
						
							|  |  |  |    copy_with_variable_replacement(X,Y,L), | 
					
						
							|  |  |  |    copy_with_variable_replacement_l(Xs,Ys,L). | 
					
						
							|  |  |  |     | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | % build variable replacement list | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | variable_replacement(X,Y,L) :- | 
					
						
							|  |  |  |    variable_replacement(X,Y,[],L). | 
					
						
							|  |  |  |     | 
					
						
							|  |  |  | variable_replacement(X,Y,L1,L2) :- | 
					
						
							|  |  |  |    ( var(X) -> | 
					
						
							|  |  |  |      var(Y), | 
					
						
							|  |  |  |      ( lookup_eq(L1,X,Z) -> | 
					
						
							|  |  |  |        Z == Y, | 
					
						
							|  |  |  |        L2 = L1 | 
					
						
							|  |  |  |      ; ( X == Y -> L2=L1 ; L2 = [X-Y,Y-X|L1]) | 
					
						
							|  |  |  |      ) | 
					
						
							|  |  |  |    ; X =.. [F|XArgs], | 
					
						
							|  |  |  |      nonvar(Y), | 
					
						
							|  |  |  |      Y =.. [F|YArgs], | 
					
						
							|  |  |  |      variable_replacement_l(XArgs,YArgs,L1,L2) | 
					
						
							|  |  |  |    ). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | variable_replacement_l([],[],L,L). | 
					
						
							|  |  |  | variable_replacement_l([X|Xs],[Y|Ys],L1,L3) :- | 
					
						
							|  |  |  |    variable_replacement(X,Y,L1,L2), | 
					
						
							|  |  |  |    variable_replacement_l(Xs,Ys,L2,L3). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | my_term_copy(X,Dict,Y) :- | 
					
						
							|  |  |  |    my_term_copy(X,Dict,_,Y). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | my_term_copy(X,Dict1,Dict2,Y) :- | 
					
						
							|  |  |  |    (   var(X) -> | 
					
						
							|  |  |  |        (   lookup_eq(Dict1,X,Y) -> | 
					
						
							|  |  |  |            Dict2 = Dict1 | 
					
						
							|  |  |  |        ;   Dict2 = [X-Y|Dict1] | 
					
						
							|  |  |  |        ) | 
					
						
							|  |  |  |    ;   functor(X,XF,XA), | 
					
						
							|  |  |  |        functor(Y,XF,XA), | 
					
						
							|  |  |  |        X =.. [_|XArgs], | 
					
						
							|  |  |  |        Y =.. [_|YArgs], | 
					
						
							|  |  |  |        my_term_copy_list(XArgs,Dict1,Dict2,YArgs) | 
					
						
							|  |  |  |    ). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | my_term_copy_list([],Dict,Dict,[]). | 
					
						
							|  |  |  | my_term_copy_list([X|Xs],Dict1,Dict3,[Y|Ys]) :- | 
					
						
							|  |  |  |    my_term_copy(X,Dict1,Dict2,Y), | 
					
						
							|  |  |  |    my_term_copy_list(Xs,Dict2,Dict3,Ys). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | atom_concat_list([X],X) :- ! . | 
					
						
							|  |  |  | atom_concat_list([X|Xs],A) :- | 
					
						
							|  |  |  | 	atom_concat_list(Xs,B), | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | 	atomic_concat(X,B,A). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | atomic_concat(A,B,C) :- | 
					
						
							|  |  |  | 	make_atom(A,AA), | 
					
						
							|  |  |  | 	make_atom(B,BB), | 
					
						
							|  |  |  | 	atom_concat(AA,BB,C). | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | make_atom(A,AA) :- | 
					
						
							|  |  |  | 	( | 
					
						
							|  |  |  | 	  atom(A) -> | 
					
						
							|  |  |  | 	  AA = A | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  | 	  number(A) -> | 
					
						
							|  |  |  | 	  number_codes(A,AL), | 
					
						
							|  |  |  | 	  atom_codes(AA,AL) | 
					
						
							|  |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	     | 
					
						
							| 
									
										
										
										
											2005-10-28 17:41:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | set_elems([],_). | 
					
						
							|  |  |  | set_elems([X|Xs],X) :- | 
					
						
							|  |  |  | 	set_elems(Xs,X). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | init([],[]). | 
					
						
							|  |  |  | init([_],[]) :- !. | 
					
						
							|  |  |  | init([X|Xs],[X|R]) :- | 
					
						
							|  |  |  | 	init(Xs,R). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | member2([X|_],[Y|_],X-Y). | 
					
						
							|  |  |  | member2([_|Xs],[_|Ys],P) :- | 
					
						
							|  |  |  | 	member2(Xs,Ys,P). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | select2(X, Y, [X|Xs], [Y|Ys], Xs, Ys). | 
					
						
							|  |  |  | select2(X, Y, [X1|Xs], [Y1|Ys], [X1|NXs], [Y1|NYs]) :- | 
					
						
							|  |  |  | 	select2(X, Y, Xs, Ys, NXs, NYs). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | instrument_goal(Goal,Pre,Post,(Pre,Goal,Post)). | 
					
						
							| 
									
										
										
										
											2007-10-16 23:17:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | sort_by_key(List,Keys,SortedList) :- | 
					
						
							|  |  |  | 	pairup(Keys,List,Pairs), | 
					
						
							|  |  |  | 	sort(Pairs,SortedPairs), | 
					
						
							|  |  |  | 	once(pairup(_,SortedList,SortedPairs)).	 | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | arg1(Term,Index,Arg) :- arg(Index,Term,Arg).	 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | wrap_in_functor(Functor,X,Term) :- | 
					
						
							|  |  |  | 	Term =.. [Functor,X]. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | tree_set_empty(TreeSet) :- empty_assoc(TreeSet). | 
					
						
							|  |  |  | tree_set_memberchk(Element,TreeSet) :- get_assoc(Element,TreeSet,_). | 
					
						
							|  |  |  | tree_set_add(TreeSet,Element,NTreeSet) :- put_assoc(Element,TreeSet,x,NTreeSet). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | :- dynamic | 
					
						
							|  |  |  | 	user:goal_expansion/2. | 
					
						
							|  |  |  | :- multifile | 
					
						
							|  |  |  | 	user:goal_expansion/2. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | user:goal_expansion(arg1(Term,Index,Arg), arg(Index,Term,Arg)). | 
					
						
							|  |  |  | user:goal_expansion(wrap_in_functor(Functor,In,Out), Goal) :- | 
					
						
							|  |  |  | 	( atom(Functor), var(Out) -> | 
					
						
							|  |  |  | 		Out =.. [Functor,In], | 
					
						
							|  |  |  | 		Goal = true | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  | 		Goal = (Out =.. [Functor,In]) | 
					
						
							|  |  |  | 	). | 
					
						
							|  |  |  | 
 |