| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | /************************************************************************* | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	 YAP Prolog 	%W% %G% | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | ************************************************************************** | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * File:		setof.pl						 * | 
					
						
							|  |  |  | * Last rev:								 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	set predicates						 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %   The "existential quantifier" symbol is only significant to bagof | 
					
						
							|  |  |  | %   and setof, which it stops binding the quantified variable. | 
					
						
							|  |  |  | %   op(200, xfy, ^) is defined during bootstrap. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % this is used by the all predicate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | :- op(50,xfx,same). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-10-30 16:42:05 +00:00
										 |  |  | _^Goal :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$execute'(Goal). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %   findall/3 is a simplified version of bagof which has an implicit | 
					
						
							|  |  |  | %   existential quantifier on every variable. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | findall(Template, Generator, Answers) :- | 
					
						
							| 
									
										
										
										
											2002-01-07 06:28:04 +00:00
										 |  |  | 	'$check_list_for_bags'(Answers, findall(Template, Generator, Answers)), | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	'$findall'(Template, Generator, [], Answers). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % If some answers have already been found | 
					
						
							|  |  |  | findall(Template, Generator, Answers, SoFar) :- | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	'$findall'(Template, Generator, SoFar, Answers). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | % starts by calling the generator, | 
					
						
							|  |  |  | % and recording the answers | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | '$findall'(Template, Generator, SoFar, Answers) :- | 
					
						
							| 
									
										
										
										
											2006-09-28 17:24:40 +00:00
										 |  |  | 	nb:nb_queue(Ref), | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	( | 
					
						
							|  |  |  | 	  '$execute'(Generator), | 
					
						
							| 
									
										
										
										
											2006-09-28 17:24:40 +00:00
										 |  |  | 	  nb:nb_queue_enqueue(Ref, Template), | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	  fail | 
					
						
							|  |  |  | 	; | 
					
						
							| 
									
										
										
										
											2006-09-28 17:24:40 +00:00
										 |  |  | 	  nb:nb_queue_close(Ref, Answers, SoFar) | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % findall_with_key is very similar to findall, but uses the SICStus | 
					
						
							|  |  |  | % algorithm to guarantee that variables will have the same names. | 
					
						
							|  |  |  | % | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | '$findall_with_common_vars'(Template, Generator, Answers) :- | 
					
						
							| 
									
										
										
										
											2006-09-28 17:24:40 +00:00
										 |  |  | 	nb:nb_queue(Ref), | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	( | 
					
						
							|  |  |  | 	  '$execute'(Generator), | 
					
						
							| 
									
										
										
										
											2006-09-28 17:24:40 +00:00
										 |  |  | 	  nb:nb_queue_enqueue(Ref, Template), | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	  fail | 
					
						
							|  |  |  | 	; | 
					
						
							| 
									
										
										
										
											2006-09-28 17:24:40 +00:00
										 |  |  | 	  nb:nb_queue_close(Ref, Answers, []), | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 	  '$collect_with_common_vars'(Answers, _) | 
					
						
							|  |  |  | 	). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-22 16:12:46 +00:00
										 |  |  | '$collect_with_common_vars'([], _). | 
					
						
							|  |  |  | '$collect_with_common_vars'([Key-_|Answers], VarList) :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$variables_in_term'(Key, _, VarList), | 
					
						
							| 
									
										
										
										
											2006-08-22 16:12:46 +00:00
										 |  |  | 	'$collect_with_common_vars'(Answers, VarList). | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | % This is the setof predicate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | setof(Template, Generator, Set) :- | 
					
						
							| 
									
										
										
										
											2002-01-07 06:28:04 +00:00
										 |  |  | 	'$check_list_for_bags'(Set, setof(Template, Generator, Set)), | 
					
						
							| 
									
										
										
										
											2001-12-28 16:11:09 +00:00
										 |  |  | 	'$bagof'(Template, Generator, Bag), | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$sort'(Bag, Set). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % And this is bagof | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % Either we have excess of variables | 
					
						
							| 
									
										
										
										
											2002-01-09 17:19:36 +00:00
										 |  |  | % and we need to find the solutions for each instantiation | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | % of these variables | 
					
						
							| 
									
										
										
										
											2001-12-28 16:11:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | bagof(Template, Generator, Bag) :- | 
					
						
							| 
									
										
										
										
											2001-12-28 16:11:09 +00:00
										 |  |  | 	'$bagof'(Template, Generator, Bag). | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | '$bagof'(Template, Generator, Bag) :- | 
					
						
							| 
									
										
										
										
											2002-01-07 06:28:04 +00:00
										 |  |  | 	'$check_list_for_bags'(Bag, bagof(Template, Generator, Bag)), | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$variables_in_term'(Template, [], TemplateV), | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | 	'$excess_vars'(Generator, StrippedGenerator, TemplateV, [], FreeVars), | 
					
						
							| 
									
										
										
										
											2005-11-26 02:57:25 +00:00
										 |  |  | 	( FreeVars \== [] -> | 
					
						
							|  |  |  | 		'$variables_in_term'(FreeVars, [], LFreeVars), | 
					
						
							|  |  |  | 		Key =.. ['$'|LFreeVars], | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 		'$findall_with_common_vars'(Key-Template, StrippedGenerator, Bags0), | 
					
						
							| 
									
										
										
										
											2005-11-26 02:57:25 +00:00
										 |  |  | 		'$keysort'(Bags0, Bags), | 
					
						
							|  |  |  | 		'$pick'(Bags, Key, Bag) | 
					
						
							|  |  |  | 	; | 
					
						
							| 
									
										
										
										
											2006-08-23 12:12:14 +00:00
										 |  |  | 		'$findall'(Template, StrippedGenerator, [], Bag0), | 
					
						
							| 
									
										
										
										
											2005-11-26 02:57:25 +00:00
										 |  |  | 		Bag0 \== [], | 
					
						
							|  |  |  | 		Bag = Bag0 | 
					
						
							|  |  |  | 	). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % picks a solution attending to the free variables | 
					
						
							|  |  |  | '$pick'([K-X|Bags], Key, Bag) :- | 
					
						
							|  |  |  | 	'$parade'(Bags, K, Bag1, Bags1), | 
					
						
							|  |  |  | 	'$decide'(Bags1, [X|Bag1], K, Key, Bag). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | '$parade'([K-X|L1], Key, [X|B], L) :- K == Key, !, | 
					
						
							|  |  |  | 	'$parade'(L1, Key, B, L). | 
					
						
							|  |  |  | '$parade'(L, _, [], L). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | % The first argument to decide gives if solutions still left; | 
					
						
							|  |  |  | % The second gives the solution currently found; | 
					
						
							|  |  |  | % The third gives the free variables that are supposed to be bound; | 
					
						
							|  |  |  | % The fourth gives the free variables being currently used. | 
					
						
							|  |  |  | % The fifth  outputs the current solution. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | '$decide'([], Bag, Key, Key, Bag) :- !. | 
					
						
							|  |  |  | '$decide'(_, Bag, Key, Key, Bag). | 
					
						
							|  |  |  | '$decide'(Bags, _, _, Key, Bag) :- | 
					
						
							|  |  |  | 	'$pick'(Bags, Key, Bag). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | % Detect free variables in the source term | 
					
						
							|  |  |  | % | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | '$excess_vars'(V, V, X, L0, L) :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	var(V), | 
					
						
							|  |  |  | 	!, | 
					
						
							|  |  |  | 	(   '$doesnt_include'(X, V) -> L = [V|L0] | 
					
						
							|  |  |  | 	;   L = L0 | 
					
						
							|  |  |  | 	). | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | '$excess_vars'(A, A, _, L, L) :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	atomic(A),  !. | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | '$excess_vars'(X^P, NP, Y, L0, L) :- !, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$variables_in_term'(X+Y, [], NY), | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | 	'$excess_vars'(P, NP, NY, L0, L). | 
					
						
							|  |  |  | '$excess_vars'(setof(X,P,S), setof(X,P,S), Y, L0, L) :- !, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$variables_in_term'(X+Y, [], NY), | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | 	'$excess_vars'((P,S), _, NY, L0, L). | 
					
						
							|  |  |  | '$excess_vars'(bagof(X,P,S), bagof(X,P,S), Y, L0, L) :- !, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$variables_in_term'(X+Y, [], NY), | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | 	'$excess_vars'((P,S), _,  NY, L0, L). | 
					
						
							|  |  |  | '$excess_vars'(findall(X,P,S), findall(X,P,S), Y, L0, L) :- !, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$excess_vars'(S, Y, L0, L). | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | '$excess_vars'(findall(X,P,S0,S), (X,P,S0,S), Y, L0, L) :- !, | 
					
						
							|  |  |  | 	'$excess_vars'(S, _, Y, L0, L). | 
					
						
							|  |  |  | '$excess_vars'(\+G, \+G, _, L0, LF) :- !, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	L0 = LF. | 
					
						
							| 
									
										
										
										
											2005-11-10 01:27:13 +00:00
										 |  |  | '$excess_vars'(_:G1, M:NG, Y, L0, LF) :- nonvar(G1), G1 = M:G, !, | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | 	'$excess_vars'(G, NG, Y, L0, LF). | 
					
						
							|  |  |  | '$excess_vars'(M:G, M:NG, Y, L0, LF) :- !, | 
					
						
							|  |  |  | 	'$excess_vars'(G, NG, Y, L0, LF). | 
					
						
							|  |  |  | '$excess_vars'(T, T, X, L0, L) :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	T =.. [_|LArgs], | 
					
						
							|  |  |  | 	'$recurse_for_excess_vars'(LArgs, X, L0, L). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | '$recurse_for_excess_vars'([], _, L, L). | 
					
						
							|  |  |  | '$recurse_for_excess_vars'([T1|LArgs], X, L0, L) :- | 
					
						
							| 
									
										
										
										
											2005-11-05 03:02:33 +00:00
										 |  |  | 	'$excess_vars'(T1, _, X, L0, L1), | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	'$recurse_for_excess_vars'(LArgs, X, L1, L). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-10-30 16:42:05 +00:00
										 |  |  | '$doesnt_include'([], _). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | '$doesnt_include'([Y|L], X) :- | 
					
						
							|  |  |  | 	Y \== X, | 
					
						
							|  |  |  | 	'$doesnt_include'(L, X). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % as an alternative to setof you can use the predicate all(Term,Goal,Solutions) | 
					
						
							|  |  |  | % But this version of all does not allow for repeated answers | 
					
						
							|  |  |  | % if you want them use findall	 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | all(T,G same X,S) :- !, all(T same X,G,Sx), '$$produce'(Sx,S,X). | 
					
						
							| 
									
										
										
										
											2004-04-22 03:23:12 +00:00
										 |  |  | all(T,G,S) :-  | 
					
						
							|  |  |  | 	'$init_db_queue'(Ref), | 
					
						
							|  |  |  | 	( '$catch'(Error,'$clean_findall'(Ref,Error),_), | 
					
						
							|  |  |  | 	  '$execute'(G), | 
					
						
							|  |  |  | 	  '$db_enqueue'(Ref, T), | 
					
						
							|  |  |  | 	  fail | 
					
						
							|  |  |  |         ; | 
					
						
							|  |  |  | 	  '$$set'(S,Ref) | 
					
						
							|  |  |  |         ). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | % $$set does its best to preserve space | 
					
						
							| 
									
										
										
										
											2004-04-22 03:23:12 +00:00
										 |  |  | '$$set'(S,R) :-  | 
					
						
							| 
									
										
										
										
											2004-05-13 20:54:58 +00:00
										 |  |  |        '$$build'(S0,_,R), | 
					
						
							| 
									
										
										
										
											2007-01-25 22:11:55 +00:00
										 |  |  |         S0 = [_|_], | 
					
						
							| 
									
										
										
										
											2004-04-22 20:06:24 +00:00
										 |  |  | 	S = S0. | 
					
						
							| 
									
										
										
										
											2004-04-22 03:23:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-04-27 15:03:06 +00:00
										 |  |  | '$$build'(Ns,S0,R) :- '$db_dequeue'(R,X), !, | 
					
						
							|  |  |  | 	'$$build2'(Ns,S0,R,X). | 
					
						
							|  |  |  | '$$build'([],_,_). | 
					
						
							| 
									
										
										
										
											2004-04-22 03:23:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-04-27 15:03:06 +00:00
										 |  |  | '$$build2'([X|Ns],Hash,R,X) :- | 
					
						
							| 
									
										
										
										
											2004-05-13 20:54:58 +00:00
										 |  |  | 	'$$new'(Hash,X), !, | 
					
						
							|  |  |  | 	'$$build'(Ns,Hash,R). | 
					
						
							| 
									
										
										
										
											2006-03-24 16:26:31 +00:00
										 |  |  | '$$build2'(Ns,Hash,R,_) :- | 
					
						
							| 
									
										
										
										
											2004-04-27 15:03:06 +00:00
										 |  |  | 	'$$build'(Ns,Hash,R). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-05-13 20:54:58 +00:00
										 |  |  | '$$new'(V,El) :- var(V), !, V = n(_,El,_). | 
					
						
							|  |  |  | '$$new'(n(R,El0,L),El) :-  | 
					
						
							|  |  |  | 	compare(C,El0,El), | 
					
						
							|  |  |  | 	'$$new'(C,R,L,El). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | '$$new'(=,_,_,_) :- !, fail. | 
					
						
							|  |  |  | '$$new'(<,R,_,El) :- '$$new'(R,El). | 
					
						
							|  |  |  | '$$new'(>,_,L,El) :- '$$new'(L,El). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | '$$produce'([T1 same X1|Tn],S,X) :- '$$split'(Tn,T1,X1,S1,S2), | 
					
						
							|  |  |  | 	( S=[T1|S1], X=X1; | 
					
						
							|  |  |  | 	  !, produce(S2,S,X) ). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | '$$split'([],_,_,[],[]). | 
					
						
							|  |  |  | '$$split'([T same X|Tn],T,X,S1,S2) :- '$$split'(Tn,T,X,S1,S2). | 
					
						
							|  |  |  | '$$split'([T1 same X|Tn],T,X,[T1|S1],S2) :- '$$split'(Tn,T,X,S1,S2). | 
					
						
							|  |  |  | '$$split'([T1|Tn],T,X,S1,[T1|S2]) :- '$$split'(Tn,T,X,S1,S2). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-07 06:28:04 +00:00
										 |  |  | '$check_list_for_bags'(V, _) :- var(V), !. | 
					
						
							|  |  |  | '$check_list_for_bags'([], _) :- !. | 
					
						
							|  |  |  | '$check_list_for_bags'([_|B], T) :- !, | 
					
						
							|  |  |  | 	'$check_list_for_bags'(B,T). | 
					
						
							|  |  |  | '$check_list_for_bags'(S, T) :- | 
					
						
							| 
									
										
										
										
											2002-09-09 17:40:12 +00:00
										 |  |  | 	'$do_error'(type_error(list,S),T). | 
					
						
							| 
									
										
										
										
											2002-01-07 06:28:04 +00:00
										 |  |  | 
 |