894 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
		
		
			
		
	
	
			894 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
|   | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
|  | %  Constraint Handling Rules			      version 2.2 % | ||
|  | %								  % | ||
|  | %  (c) Copyright 1996-98					  % | ||
|  | %  LMU, Muenchen						  % | ||
|  | %								  % | ||
|  | %  File:   chr.pl						  % | ||
|  | %  Author: Christian Holzbaur		christian@ai.univie.ac.at % | ||
|  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
|  | 
 | ||
|  | % | ||
|  | % The CHR runtime system, | ||
|  | % the constraint store. | ||
|  | % | ||
|  | % Two functions: a) storage b) reactivation triggered by bindings | ||
|  | % | ||
|  | % Reactivation is symmetric: if two variables with suspensions | ||
|  | % are unified, both suspensions run. (Both variables got more | ||
|  | % constrained) | ||
|  | % | ||
|  | % *** Sequence of wakeups determines termination of handler leq *** | ||
|  | % | ||
|  | % Another sequence that could matter is the one | ||
|  | % generated by the iterators | ||
|  | % | ||
|  | % Layout: | ||
|  | % | ||
|  | %	   suspension(Id,State,Closure,Generation,PropagationHistory,F|Args) | ||
|  | % | ||
|  | %   Id is 1st to allow for direct comparisons (sort) and avoids | ||
|  | %   unifiability if the Id is nonvar. | ||
|  | %   F is the constraint functor | ||
|  | % | ||
|  | % | ||
|  | 
 | ||
|  | :- module( chr, | ||
|  | 	[ | ||
|  | 	  find_constraint/2, | ||
|  | 	  find_constraint/3, | ||
|  | 	  findall_constraints/2, | ||
|  | 	  findall_constraints/3, | ||
|  | 	  remove_constraint/1, | ||
|  | 	  current_handler/2, | ||
|  | 	  current_constraint/2, | ||
|  | 	  unconstrained/1, | ||
|  | 	  notify_constrained/1, | ||
|  | 
 | ||
|  | 	  chr_trace/0, chr_notrace/0, | ||
|  | 	  chr_debug/0, chr_nodebug/0, chr_debugging/0, | ||
|  | 	  chr_leash/1, chr_spy/1, chr_nospy/1 | ||
|  |       ]). | ||
|  | 
 | ||
|  | :- use_module( library('chr/getval')). | ||
|  | 
 | ||
|  | :- use_module( library(lists), | ||
|  | 	[ | ||
|  | 	  append/3, | ||
|  | 	  member/2, | ||
|  | 	  is_list/1, | ||
|  | 	  nth/3, | ||
|  | 	  select/3 | ||
|  | 	]). | ||
|  | 
 | ||
|  | :- use_module( library(terms), | ||
|  | 	[ | ||
|  | 	  term_variables/2, | ||
|  | 	  subsumes_chk/2, | ||
|  | 	  subsumes/2 | ||
|  | 	]). | ||
|  | 
 | ||
|  | :- use_module( library(assoc),			% propagation history | ||
|  | 	[ | ||
|  | 	  empty_assoc/1, | ||
|  | 	  put_assoc/4, | ||
|  | 	  get_assoc/3, | ||
|  | 	  assoc_to_list/2 | ||
|  |       ]). | ||
|  | 
 | ||
|  | :- use_module(library('chr/sbag')).   % link to sbag_l.pl or sbag_a.pl | ||
|  | :- use_module(library('chr/chrcmp')). | ||
|  | :- use_module(library('chr/trace')). | ||
|  | 
 | ||
|  | :- use_module(library(atts)). | ||
|  | 
 | ||
|  | :- attribute locked/0, exposed/1, dbg_state/1. | ||
|  | 
 | ||
|  | % | ||
|  | % Problem with cyclic structures: | ||
|  | %   error reporters seem to use write ... | ||
|  | % | ||
|  | :- multifile | ||
|  | 	user:portray/1, | ||
|  | 	user:portray_message/2, | ||
|  | 	user:goal_expansion/3. | ||
|  | 
 | ||
|  | :- dynamic | ||
|  | 	user:portray/1, | ||
|  | 	user:portray_message/2, | ||
|  | 	user:goal_expansion/3. | ||
|  | 
 | ||
|  | % | ||
|  | user:portray( Susp) :- | ||
|  | 	Susp =.. [suspension,Id,Mref,_,_,_,_|_], | ||
|  | 	nonvar( Mref), | ||
|  | 	!, | ||
|  | 	write('<c'), write(Id), write('>').	% (c)onstraint | ||
|  | % | ||
|  | user:portray( '$want_duplicates'(_,Term)) :- !, % cf. attribute_goal/2 | ||
|  | 	prolog_flag( toplevel_print_options, Options), | ||
|  | 	write_term( Term, Options). | ||
|  | 
 | ||
|  | :- initialization | ||
|  | 	setval( id, 0).		% counter for portray/debugger | ||
|  | 
 | ||
|  | % | ||
|  | user:portray_message( error, chr(multiple_handlers(Old,New,Module))) :- !, | ||
|  | 	format( user_error, '{CHR ERROR: registering ~p, module ~p already hosts ~p}~n', | ||
|  | 	  [New,Module,Old]). | ||
|  | 
 | ||
|  | % ----------------------------------------------------------------- | ||
|  | 
 | ||
|  | % | ||
|  | % *** MACROS *** | ||
|  | % | ||
|  | % | ||
|  | user:goal_expansion( lock_some(L),		 chr, Exp) :- is_list(L), | ||
|  | 	unravel( L, lock, Exp). | ||
|  | user:goal_expansion( unlock_some(L),		 chr, Exp) :- is_list(L), | ||
|  | 	unravel( L, unlock, Exp). | ||
|  | user:goal_expansion( via([],V), 		 chr, global_term_ref_1(V)). | ||
|  | user:goal_expansion( via([X],V),		 chr, via_1(X,V)). | ||
|  | user:goal_expansion( via([X,Y],V),		 chr, via_2(X,Y,V)). | ||
|  | user:goal_expansion( via([X,Y,Z],V),		 chr, via_3(X,Y,Z,V)). | ||
|  | user:goal_expansion( load_args(S,State,Args),	 chr, Exp) :- | ||
|  | 	is_list( Args), | ||
|  | 	Susp =.. [suspension,_,Mref,_,_,_,_|Args], | ||
|  | 	Exp = ( S=Susp, get_mutable( State, Mref) ). | ||
|  | % | ||
|  | % | ||
|  | % | ||
|  | user:goal_expansion( nd_init_iteration(V,_,_,Att,S), _, Exp) :- | ||
|  | 	arg( 1, Att, Stack), | ||
|  | 	Exp = ( get_atts(V,Att), chr:sbag_member(S,Stack) ). | ||
|  | % | ||
|  | user:goal_expansion( init_iteration(V,_,_,Att,L), _, Exp) :- | ||
|  | 	arg( 1, Att, Stack), | ||
|  | 	Exp = ( get_atts(V,Att), chr:iter_init(Stack,L) ). | ||
|  | 
 | ||
|  | unravel( [],	 _, true). | ||
|  | unravel( [X|Xs], F, (G,Gs)) :- | ||
|  | 	G =.. [F,X], | ||
|  | 	unravel( Xs, F, Gs). | ||
|  | 
 | ||
|  | % ----------------------- runtime user predicates ----------------- | ||
|  | 
 | ||
|  | remove_constraint( Susp) :- | ||
|  | 	nonvar( Susp), | ||
|  | 	functor( Susp, suspension, N), | ||
|  | 	N >= 6, | ||
|  | 	!, | ||
|  | 	debug_event( remove(Susp)), | ||
|  | 	remove_constraint_internal( Susp, Vars), | ||
|  | 	arg( 3, Susp, Module:_), | ||
|  | 	arg( 6, Susp, F), | ||
|  | 	A is N-6, | ||
|  | 	Module:detach( F/A, Susp, Vars).  | ||
|  | remove_constraint( S) :- | ||
|  | 	raise_exception( type_error(remove_constraint(S),1,'a constraint object',S)). | ||
|  | 
 | ||
|  | find_constraint( Term, Susp) :- | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	find_constraint( Global, Term, Susp). | ||
|  | 
 | ||
|  | find_constraint( V, Term, Susp) :- var( V), !, | ||
|  | 	find_constraint_internal( V, Term, Susp, active, _). | ||
|  | find_constraint( A, B, C) :- | ||
|  | 	raise_exception( instantiation_error( find_constraint(A,B,C), 1)). | ||
|  | 
 | ||
|  | find_constraint_internal( V, Term, Susp, State, Module) :- | ||
|  | 	constraint( Handler, F/A, Att), | ||
|  | 	functor( Term, F, A),			% prune some | ||
|  | 	arg( 1, Att, Stack), | ||
|  | 	current_handler( Handler, Module), | ||
|  | 	Module:get_atts( V, Att), | ||
|  | 	length( Args, A), | ||
|  | 	Try =.. [F|Args], | ||
|  | 	sbag_member( Susp, Stack), | ||
|  | 	Susp =.. [suspension,_,Mref,_,_,_,_|Args], | ||
|  | 	get_mutable( State, Mref), | ||
|  | 	subsumes( Term, Try). | ||
|  | 
 | ||
|  | % | ||
|  | % Test for unconstrained var | ||
|  | % Used by some math solvers | ||
|  | % | ||
|  | unconstrained( X) :- | ||
|  | 	% var(X), prolog:'$get_cva'(X,[],_). | ||
|  | 	find_constraint( X, _, _), !, fail. | ||
|  | unconstrained( _). | ||
|  | 
 | ||
|  | findall_constraints( C, L) :- | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	findall_constraints( Global, C, L). | ||
|  | 
 | ||
|  | findall_constraints( V, C, L) :- var( V), !, | ||
|  | 	findall( M:Att, ( | ||
|  | 			    constraint( H, F/A, Att), | ||
|  | 			    functor( C, F, A), | ||
|  | 			    current_handler( H, M) | ||
|  | 			), | ||
|  | 			Agenda), | ||
|  | 	findall_constraints( Agenda, C, V, L, []). | ||
|  | findall_constraints( V, C, L) :- | ||
|  | 	raise_exception( instantiation_error( findall_constraints(V,C,L), 1)). | ||
|  | 
 | ||
|  | findall_constraints( [],		  _, _) --> []. | ||
|  | findall_constraints( [Module:Att|Agenda], C, V) --> | ||
|  | 	( { | ||
|  | 	    arg( 1, Att, Stack), | ||
|  | 	    Module:get_atts( V, Att), | ||
|  | 	    iter_init( Stack, State) | ||
|  | 	  } -> | ||
|  | 	    findall_constraints_( State, C, Module) | ||
|  | 	; | ||
|  | 	    [] | ||
|  | 	), | ||
|  | 	findall_constraints( Agenda, C, V). | ||
|  | 
 | ||
|  | findall_constraints_( State, _,       _) --> {iter_last(State)}. | ||
|  | findall_constraints_( State, General, Module) --> | ||
|  | 	{ | ||
|  | 	    iter_next( State, S, Next) | ||
|  | 	}, | ||
|  | 	( { | ||
|  | 	    S =.. [suspension,_,Mref,_,_,_,F|Args], | ||
|  | 	    get_mutable( active, Mref), | ||
|  | 	    Term =.. [F|Args], | ||
|  | 	    subsumes_chk( General, Term) | ||
|  | 	  } -> | ||
|  | 		[ Term#S ] | ||
|  | 	; | ||
|  | 		[] | ||
|  | 	), | ||
|  | 	findall_constraints_( Next, General, Module). | ||
|  | 
 | ||
|  | % | ||
|  | % Decorate a constraint Term from Module | ||
|  | % with a module prefix if needed. | ||
|  | % | ||
|  | module_wrap( Term, Module, Wrapped) :- | ||
|  | 	prolog_flag( typein_module, Typein), | ||
|  | 	( Module == Typein -> | ||
|  | 	    Wrapped = Term | ||
|  | 	; predicate_property( Typein:Term, imported_from(_)) -> | ||
|  | 	    Wrapped = Term | ||
|  | 	; | ||
|  | 	    Wrapped = Module:Term | ||
|  | 	). | ||
|  | 
 | ||
|  | % ----------------------------------------------------------------- | ||
|  | /* | ||
|  | 
 | ||
|  |    Two namespaces handler/module actually only justified if there | ||
|  |    can be more than one handler per module ... | ||
|  | 
 | ||
|  | */ | ||
|  | 
 | ||
|  | :- dynamic handler/2. | ||
|  | :- dynamic constraint/3. | ||
|  | 
 | ||
|  | current_handler( Handler, Module) :- | ||
|  | 	handler( Handler, Module). | ||
|  | 
 | ||
|  | current_constraint( Handler, C) :- | ||
|  | 	constraint( Handler, C, _). | ||
|  | 
 | ||
|  | register_handler( Handler, Cs, Slots) :- | ||
|  | 	prolog_load_context( module, Module), | ||
|  | 	( handler(Other,Module), | ||
|  | 	  Other \== Handler -> | ||
|  | 	    raise_exception( chr(multiple_handlers(Other,Handler,Module))) | ||
|  | 	; handler( Handler, Module) -> | ||
|  | 	    true				% simple reload | ||
|  | 	; | ||
|  | 	    assert( handler(Handler,Module)) | ||
|  | 	), | ||
|  | 	retractall( constraint(Handler,_,_)), | ||
|  | 	reg_handler( Cs, Slots, Handler). | ||
|  | 
 | ||
|  | reg_handler( [],     [],     _). | ||
|  | reg_handler( [C|Cs], [S|Ss], Handler) :- | ||
|  | 	assert( constraint(Handler,C,S)), | ||
|  | 	reg_handler( Cs, Ss, Handler). | ||
|  | 
 | ||
|  | % ---------------------------------------------------------------- | ||
|  | 
 | ||
|  | notify_constrained( X) :- var( X), | ||
|  | 	findall( M, handler(_,M), Modules), | ||
|  | 	notify_constrained( Modules, X). | ||
|  | notify_constrained( X) :- nonvar( X), | ||
|  | 	raise_exception( instantitation_error( notify_constrained(X),1)). | ||
|  | 
 | ||
|  | notify_constrained( [], _). | ||
|  | notify_constrained( [M|Ms], X) :- | ||
|  | 	M:get_suspensions( X, S), | ||
|  | 	run_suspensions( S), | ||
|  | 	notify_constrained( Ms, X). | ||
|  | 
 | ||
|  | % | ||
|  | % support for verify_attributes/3, notify_constrained/1 | ||
|  | % | ||
|  | % Approximation because debug state might change between calls ... | ||
|  | % | ||
|  | run_suspensions( Slots) :- | ||
|  | 	getval( debug, State), | ||
|  | 	( State == off -> | ||
|  | 	    run_suspensions_loop( Slots) | ||
|  | 	; | ||
|  | 	    run_suspensions_loop_d( Slots) | ||
|  | 	), | ||
|  | 	true. | ||
|  | 
 | ||
|  | run_suspensions_loop( []). | ||
|  | run_suspensions_loop( [A|As]) :- | ||
|  | 	arg( 1, A, Stack), | ||
|  | 	iter_init( Stack, State), | ||
|  | 	run_suspensions_( State), | ||
|  | 	run_suspensions_loop( As). | ||
|  | 
 | ||
|  | run_suspensions_loop_d( []). | ||
|  | run_suspensions_loop_d( [A|As]) :- | ||
|  | 	arg( 1, A, Stack), | ||
|  | 	iter_init( Stack, State), | ||
|  | 	run_suspensions_d( State), | ||
|  | 	run_suspensions_loop_d( As). | ||
|  | 
 | ||
|  | % | ||
|  | % Transition active->triggered->removed instead of | ||
|  | % active->removed is to avoid early gc of suspensions. | ||
|  | % The suspension's generation is incremented to signal | ||
|  | % to the revive scheme that the constraint has been | ||
|  | % processed already. | ||
|  | % | ||
|  | run_suspensions_( State) :- iter_last( State). | ||
|  | run_suspensions_( State) :- | ||
|  | 	iter_next( State, S, Next), | ||
|  | 	arg( 2, S, Mref), | ||
|  | 	get_mutable( Status, Mref), | ||
|  | 	( Status==active -> | ||
|  | 	    update_mutable( triggered, Mref), | ||
|  | 	    arg( 4, S, Gref), | ||
|  | 	    get_mutable( Gen, Gref), | ||
|  | 	    Generation is Gen+1, | ||
|  | 	    update_mutable( Generation, Gref), | ||
|  | 	    arg( 3, S, Goal), | ||
|  | 	    call( Goal), | ||
|  | 	    get_mutable( Post, Mref), | ||
|  | 	    ( Post==triggered -> | ||
|  | 		update_mutable( removed, Mref) | ||
|  | 	    ; | ||
|  | 		true | ||
|  | 	    ) | ||
|  | 	; | ||
|  | 	    true | ||
|  | 	), | ||
|  | 	run_suspensions_( Next). | ||
|  | 
 | ||
|  | run_suspensions_d( State) :- iter_last( State). | ||
|  | run_suspensions_d( State) :- | ||
|  | 	iter_next( State, S, Next), | ||
|  | 	arg( 2, S, Mref), | ||
|  | 	get_mutable( Status, Mref), | ||
|  | 	( Status==active -> | ||
|  | 	    update_mutable( triggered, Mref), | ||
|  | 	    arg( 4, S, Gref), | ||
|  | 	    get_mutable( Gen, Gref), | ||
|  | 	    Generation is Gen+1, | ||
|  | 	    update_mutable( Generation, Gref), | ||
|  | 	    arg( 3, S, Goal), | ||
|  | 	    byrd( S, Goal), | ||
|  | 	    get_mutable( Post, Mref), | ||
|  | 	    ( Post==triggered -> | ||
|  | 		update_mutable( removed, Mref) | ||
|  | 	    ; | ||
|  | 		true | ||
|  | 	    ) | ||
|  | 	; | ||
|  | 	    true | ||
|  | 	), | ||
|  | 	run_suspensions_d( Next). | ||
|  | 
 | ||
|  | byrd( Self, Goal) :- | ||
|  | 	(   debug_event( wake(Self)), call( Goal) | ||
|  | 	;   debug_event( fail(Self)), !, fail | ||
|  | 	), | ||
|  | 	(   debug_event( exit(Self)) | ||
|  | 	;   debug_event( redo(Self)), fail | ||
|  | 	). | ||
|  | 
 | ||
|  | % | ||
|  | % Merge 2 sorted lists of Name/1 terms. | ||
|  | % The argument of each term is a sbag. | ||
|  | % | ||
|  | merge_attributes( [],	  Bs, Bs). | ||
|  | merge_attributes( [A|As], Bs, Cs) :- | ||
|  | 	merge_attributes( Bs, Cs, A, As). | ||
|  | 
 | ||
|  | merge_attributes( [],	  [A|As], A, As). | ||
|  | merge_attributes( [B|Bs], Cs,	  A, As) :- | ||
|  | 	functor( A, NameA, 1), | ||
|  | 	functor( B, NameB, 1), | ||
|  | 	compare( R, NameA, NameB), | ||
|  | 	( R == < -> Cs = [A|Css], merge_attributes( As, Css, B, Bs) | ||
|  | 	; R == > -> Cs = [B|Css], merge_attributes( Bs, Css, A, As) | ||
|  | 	; | ||
|  | 	    Cs = [C|Css], | ||
|  | 	    functor( C, NameA, 1), | ||
|  | 	    arg( 1, A, StackA), | ||
|  | 	    arg( 1, B, StackB), | ||
|  | 	    arg( 1, C, StackC), | ||
|  | 	    sbag_union( StackA, StackB, StackC), | ||
|  | 	    merge_attributes( As, Bs, Css) | ||
|  | 	). | ||
|  | 
 | ||
|  | show_bag( Bag) :- | ||
|  | 	iter_init( Bag, State), | ||
|  | 	show_bag_( State), | ||
|  | 	nl. | ||
|  | 
 | ||
|  | show_bag_( State) :- iter_last( State). | ||
|  | show_bag_( State) :- | ||
|  | 	iter_next( State, S, Next), | ||
|  | 	arg( 2, S, Ref), | ||
|  | 	get_mutable( St, Ref), | ||
|  | 	format( ' ~p:~p', [S,St]), | ||
|  | 	show_bag_( Next). | ||
|  | 
 | ||
|  | % | ||
|  | % Support for attribute_goal/2. | ||
|  | % | ||
|  | % Complication: the Sicstus kernel removes duplicates | ||
|  | % via call_residue/2 - that includes the toplevel. | ||
|  | % We may want to see them -> | ||
|  | %   tag Term with Suspension, 'untag' via portray/1 | ||
|  | % | ||
|  | % Called with a list of slots once per module | ||
|  | % | ||
|  | attribute_goals( L, Goal, Module) :- | ||
|  | 	attribute_goal_loop( L, Module, GL, []), | ||
|  | 	l2c( GL, Goal). | ||
|  | 
 | ||
|  | attribute_goal_loop( [],     _) --> []. | ||
|  | attribute_goal_loop( [A|As], Mod) --> | ||
|  | 	{ | ||
|  | 	    arg( 1, A, Stack), | ||
|  | 	    iter_init( Stack, State) | ||
|  | 	}, | ||
|  | 	attgs_( State, Mod), | ||
|  | 	attribute_goal_loop( As, Mod). | ||
|  | 
 | ||
|  | attgs_( State, _) --> {iter_last( State)}. | ||
|  | attgs_( State, Module) --> | ||
|  | 	{ | ||
|  | 	    iter_next( State, S, Next), | ||
|  | 	    S =.. [suspension,_,Mref,_,_,_,F|Args] | ||
|  | 	}, | ||
|  | 	( {get_mutable(active,Mref)} -> | ||
|  | 	     { | ||
|  | 	       Term =.. [F|Args], | ||
|  | 	       module_wrap( Term, Module, Wrapped) | ||
|  | 	     }, | ||
|  | 	     [ '$want_duplicates'(S,Wrapped) ] | ||
|  | 	; | ||
|  | 	     [] | ||
|  | 	), | ||
|  | 	attgs_( Next, Module). | ||
|  | 
 | ||
|  | % | ||
|  | % fail for empty list | ||
|  | % | ||
|  | l2c( [C], C) :- !. | ||
|  | l2c( [C|Cs], (C,Cj)) :- | ||
|  | 	l2c( Cs, Cj). | ||
|  | 
 | ||
|  | % | ||
|  | % Unlink removed constraints cleanly from all chains | ||
|  | % Still need gc state because of wake, | ||
|  | % but re-insertion = insert because of complete removal. | ||
|  | % | ||
|  | chr_gc :- | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	findall( M, handler(_,M), Modules), | ||
|  | 	chr_gcm( Modules, Global). | ||
|  | 
 | ||
|  | chr_gcm( [],	 _). | ||
|  | chr_gcm( [M|Ms], Global) :- | ||
|  | 	M:get_suspensions( Global, AllS), | ||
|  | 	term_variables( [Global|AllS], Vars),	% AllS may be ground | ||
|  | 	chr_gcv( Vars, M), | ||
|  | 	chr_gcm( Ms, Global). | ||
|  | 
 | ||
|  | % | ||
|  | % Have compiler generated support? | ||
|  | % | ||
|  | chr_gcv( [],	 _). | ||
|  | chr_gcv( [V|Vs], M) :- | ||
|  | 	M:get_suspensions( V, Old), | ||
|  | 	chr_gcb( Old, New), | ||
|  | 	M:put_suspensions( V, New), | ||
|  | 	chr_gcv( Vs, M). | ||
|  | 
 | ||
|  | chr_gcb( [],	 []). | ||
|  | chr_gcb( [S|Ss], [Sgc|Ts]) :- | ||
|  | 	arg( 1, S, Bag), | ||
|  | 	iter_init( Bag, State), | ||
|  | 	functor( S, N, 1), | ||
|  | 	functor( T, N, 1), | ||
|  | 	gc_bag( State, Lgc), | ||
|  | 	( Lgc==[] -> | ||
|  | 	    Sgc = -T | ||
|  | 	; | ||
|  | 	    Sgc = T, | ||
|  | 	    list_to_sbag( Lgc, BagGc), | ||
|  | 	    arg( 1, T, BagGc) | ||
|  | 	), | ||
|  | 	chr_gcb( Ss, Ts). | ||
|  | 
 | ||
|  | gc_bag( State, []) :- iter_last( State). | ||
|  | gc_bag( State, L) :- | ||
|  | 	iter_next( State, Susp, Next), | ||
|  | 	arg( 2, Susp, Mref), | ||
|  | 	get_mutable( SuspState, Mref), | ||
|  | 	( SuspState==removed -> | ||
|  | 	    L = Tail, | ||
|  | 	    update_mutable( gc, Mref) | ||
|  | 	; SuspState==gc -> | ||
|  | 	    L = Tail | ||
|  | 	; | ||
|  | 	    L = [Susp|Tail] | ||
|  | 	), | ||
|  | 	gc_bag( Next, Tail). | ||
|  | 
 | ||
|  | % -------------------------------------------------------------------- | ||
|  | % | ||
|  | % Incremental allocation & activation of constraints. | ||
|  | % Attachment code of closures to variables is generated | ||
|  | % by the compiler. | ||
|  | % | ||
|  | % States {passive(Term),inactive,triggered,active,removed,gc} | ||
|  | % | ||
|  | % | ||
|  | 
 | ||
|  | :- meta_predicate allocate_constraint(:,-,+,+). | ||
|  | % | ||
|  | allocate_constraint( Closure, Self, F, Args) :- | ||
|  | 	empty_history( History), | ||
|  | 	create_mutable( passive(Args), Mref), | ||
|  | 	create_mutable( 0, Gref), | ||
|  | 	create_mutable( History, Href), | ||
|  | 	gen_id( Id), | ||
|  | 	Self =.. [suspension,Id,Mref,Closure,Gref,Href,F|Args]. | ||
|  | 
 | ||
|  | % | ||
|  | % activate_constraint( -, +, -). | ||
|  | % | ||
|  | % The transition gc->active should be rare | ||
|  | % | ||
|  | activate_constraint( Vars, Susp, Generation) :- | ||
|  | 	arg( 2, Susp, Mref), | ||
|  | 	get_mutable( State, Mref), | ||
|  | 	update_mutable( active, Mref), | ||
|  | 	( nonvar(Generation) ->			% aih | ||
|  | 	    true | ||
|  | 	; | ||
|  | 	    arg( 4, Susp, Gref), | ||
|  | 	    get_mutable( Gen, Gref), | ||
|  | 	    Generation is Gen+1, | ||
|  | 	    update_mutable( Generation, Gref) | ||
|  | 	), | ||
|  | 	( compound(State) ->			% passive/1 | ||
|  | 	    term_variables( State, Vs), | ||
|  | 	    none_locked( Vs), | ||
|  | 	    global_term_ref_1( Global), | ||
|  | 	    Vars = [Global|Vs] | ||
|  | 	; State==gc ->				% removed from all chains | ||
|  | 	    Susp =.. [_,_,_,_,_,_,_|Args], | ||
|  | 	    term_variables( Args, Vs), | ||
|  | 	    global_term_ref_1( Global), | ||
|  | 	    Vars = [Global|Vs] | ||
|  | 	; State==removed ->			% the price for eager removal ... | ||
|  | 	    Susp =.. [_,_,_,_,_,_,_|Args], | ||
|  | 	    term_variables( Args, Vs), | ||
|  | 	    global_term_ref_1( Global), | ||
|  | 	    Vars = [Global|Vs] | ||
|  | 	; | ||
|  | 	    Vars = [] | ||
|  | 	). | ||
|  | 
 | ||
|  | % | ||
|  | % Combination of the prev. two | ||
|  | % | ||
|  | :- meta_predicate insert_constraint_internal(-,-,:,+,+). | ||
|  | % | ||
|  | insert_constraint_internal( [Global|Vars], Self, Closure, F, Args) :- | ||
|  | 	term_variables( Args, Vars), | ||
|  | 	none_locked( Vars), | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	empty_history( History), | ||
|  | 	create_mutable( active, Mref), | ||
|  | 	create_mutable( 0, Gref), | ||
|  | 	create_mutable( History, Href), | ||
|  | 	gen_id( Id), | ||
|  | 	Self =.. [suspension,Id,Mref,Closure,Gref,Href,F|Args]. | ||
|  | 
 | ||
|  | :- meta_predicate insert_constraint_internal(-,-,?,:,+,+). | ||
|  | % | ||
|  | insert_constraint_internal( [Global|Vars], Self, Term, Closure, F, Args) :- | ||
|  | 	term_variables( Term, Vars), | ||
|  | 	none_locked( Vars), | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	empty_history( History), | ||
|  | 	create_mutable( active, Mref), | ||
|  | 	create_mutable( 0, Gref), | ||
|  | 	create_mutable( History, Href), | ||
|  | 	gen_id( Id), | ||
|  | 	Self =.. [suspension,Id,Mref,Closure,Gref,Href,F|Args]. | ||
|  | 
 | ||
|  | gen_id( Id) :- | ||
|  | 	incval( id, Id). | ||
|  |         /* no undo/1 in sicstus3.7 | ||
|  | 	( Id =:= 1 ->				% first time called | ||
|  | 	    undo( setval(id,0)) | ||
|  | 	; | ||
|  | 	    true | ||
|  | 	). | ||
|  | 	*/ | ||
|  | 
 | ||
|  | % | ||
|  | % Eager removal from all chains. | ||
|  | % | ||
|  | remove_constraint_internal( Susp, Agenda) :- | ||
|  | 	arg( 2, Susp, Mref), | ||
|  | 	get_mutable( State, Mref), | ||
|  | 	update_mutable( removed, Mref),		% mark in any case | ||
|  | 	( compound(State) ->			% passive/1 | ||
|  | 	    Agenda = [] | ||
|  | 	; State==removed -> | ||
|  | 	    Agenda = [] | ||
|  | 	; State==triggered -> | ||
|  | 	    Agenda = [] | ||
|  | 	; | ||
|  |             Susp =.. [_,_,_,_,_,_,_|Args], | ||
|  | 	    term_variables( Args, Vars), | ||
|  | 	    global_term_ref_1( Global), | ||
|  | 	    Agenda = [Global|Vars] | ||
|  | 	). | ||
|  | 
 | ||
|  | % | ||
|  | % Protect the goal against any binding | ||
|  | % or attachment of constraints. The latter is | ||
|  | % via the notify_constrained/1 convention. | ||
|  | % | ||
|  | lock( T) :- var(T), put_atts( T, locked). | ||
|  | lock( T) :- nonvar( T), | ||
|  | 	functor( T, _, N), | ||
|  | 	lock_arg( N, T). | ||
|  | 
 | ||
|  | lock_arg( 0, _) :- !. | ||
|  | lock_arg( 1, T) :- !, arg( 1, T, A), lock( A). | ||
|  | lock_arg( 2, T) :- !, arg( 1, T, A), lock( A), arg( 2, T, B), lock( B). | ||
|  | lock_arg( N, T) :- | ||
|  | 	arg( N, T, A), | ||
|  | 	lock( A), | ||
|  | 	M is N-1, | ||
|  | 	lock_arg( M, T). | ||
|  | 
 | ||
|  | unlock( T) :- var(T), put_atts( T, -locked). | ||
|  | unlock( T) :- nonvar( T), | ||
|  | 	functor( T, _, N), | ||
|  | 	unlock_arg( N, T). | ||
|  | 
 | ||
|  | unlock_arg( 0, _) :- !. | ||
|  | unlock_arg( 1, T) :- !, arg( 1, T, A), unlock( A). | ||
|  | unlock_arg( 2, T) :- !, arg( 1, T, A), unlock( A), arg( 2, T, B), unlock( B). | ||
|  | unlock_arg( N, T) :- | ||
|  | 	arg( N, T, A), | ||
|  | 	unlock( A), | ||
|  | 	M is N-1, | ||
|  | 	unlock_arg( M, T). | ||
|  | 
 | ||
|  | verify_attributes( X, Y, []) :- | ||
|  | 	get_atts( X, locked), | ||
|  | 	!, | ||
|  | 	var(Y), | ||
|  | 	get_atts( Y, -locked), | ||
|  | 	put_atts( Y,  locked). | ||
|  | verify_attributes( _, _, []). | ||
|  | 
 | ||
|  | none_locked( []). | ||
|  | none_locked( [V|Vs]) :- | ||
|  | 	not_locked( V), | ||
|  | 	none_locked( Vs). | ||
|  | 
 | ||
|  | not_locked( V) :- var( V), get_atts( V, -locked). | ||
|  | not_locked( V) :- nonvar( V). | ||
|  | 
 | ||
|  | % -------------------------- access to constraints ------------------ | ||
|  | 
 | ||
|  | % | ||
|  | % Try a list of candidates. V may be nonvar but | ||
|  | % bound to a term with variables in it. | ||
|  | % | ||
|  | via( L, V) :- | ||
|  | 	member( X, L), | ||
|  | 	var( X), | ||
|  | 	!, | ||
|  | 	V = X. | ||
|  | via( L, V) :- | ||
|  | 	compound( L), | ||
|  | 	nonground( L, V), | ||
|  | 	!. | ||
|  | via( _, V) :- | ||
|  | 	global_term_ref_1( V). | ||
|  | 
 | ||
|  | % | ||
|  | % specialization(s) | ||
|  | % | ||
|  | via_1( X, V) :- var(X), !, X=V. | ||
|  | via_1( T, V) :- compound(T), nonground( T, V), !. | ||
|  | via_1( _, V) :- global_term_ref_1( V). | ||
|  | 
 | ||
|  | via_2( X, _, V) :- var(X), !, X=V. | ||
|  | via_2( _, Y, V) :- var(Y), !, Y=V. | ||
|  | via_2( T, _, V) :- compound(T), nonground( T, V), !. | ||
|  | via_2( _, T, V) :- compound(T), nonground( T, V), !. | ||
|  | via_2( _, _, V) :- global_term_ref_1( V). | ||
|  | 
 | ||
|  | via_3( X, _, _, V) :- var(X), !, X=V. | ||
|  | via_3( _, Y, _, V) :- var(Y), !, Y=V. | ||
|  | via_3( _, _, Z, V) :- var(Z), !, Z=V. | ||
|  | via_3( T, _, _, V) :- compound(T), nonground( T, V), !. | ||
|  | via_3( _, T, _, V) :- compound(T), nonground( T, V), !. | ||
|  | via_3( _, _, T, V) :- compound(T), nonground( T, V), !. | ||
|  | via_3( _, _, _, V) :- global_term_ref_1( V). | ||
|  | 
 | ||
|  | 
 | ||
|  | % | ||
|  | % The second arg is a witness. | ||
|  | % The formulation with term_variables/2 is | ||
|  | % cycle safe, but it finds a list of all vars. | ||
|  | % We need only one, and no list in particular. | ||
|  | % | ||
|  | nonground( Term, V) :- | ||
|  | 	term_variables( Term, Vs), | ||
|  | 	Vs = [V|_]. | ||
|  | 
 | ||
|  | /* | ||
|  | nonground( Term, V) :- var( Term), V=Term. | ||
|  | nonground( Term, V) :- compound( Term), | ||
|  | 	functor( Term, _, N), | ||
|  | 	nonground( N, Term, V). | ||
|  | 
 | ||
|  | % | ||
|  | % assert: N > 0 | ||
|  | % | ||
|  | nonground( 1, Term, V) :- !, | ||
|  | 	arg( 1, Term, Arg), | ||
|  | 	nonground( Arg, V). | ||
|  | nonground( 2, Term, V) :- !, | ||
|  | 	arg( 2, Term, Arg2), | ||
|  | 	( nonground( Arg2, V) -> | ||
|  | 	    true | ||
|  | 	; | ||
|  | 	    arg( 1, Term, Arg1), | ||
|  | 	    nonground( Arg1, V) | ||
|  | 	). | ||
|  | nonground( N, Term, V) :- | ||
|  | 	arg( N, Term, Arg), | ||
|  | 	( nonground( Arg, V) -> | ||
|  | 	    true | ||
|  | 	; | ||
|  | 	    M is N-1, | ||
|  | 	    nonground( M, Term, V) | ||
|  | 	). | ||
|  | */ | ||
|  | 
 | ||
|  | constraint_generation( Susp, State, Generation) :- | ||
|  | 	arg( 2, Susp, Mref), | ||
|  | 	get_mutable( State, Mref), | ||
|  | 	arg( 4, Susp, Gref), | ||
|  | 	get_mutable( Generation, Gref). 	% not incremented meanwhile | ||
|  | 
 | ||
|  | change_state( Susp, State) :- | ||
|  | 	arg( 2, Susp, Mref), | ||
|  | 	update_mutable( State, Mref). | ||
|  | 
 | ||
|  | :- meta_predicate expose(-,+,+,+,:). | ||
|  | % | ||
|  | expose_active( Ref, Head, Tid, Heads, Continuation) :- | ||
|  | 	get_exposed( Ref), | ||
|  | 	get_mutable( Exposed, Ref), | ||
|  | 	update_mutable( [active(Head,Tid,Heads,Continuation)|Exposed], Ref). | ||
|  | 
 | ||
|  | expose_passive( Ref, Heads) :- | ||
|  | 	get_exposed( Ref), | ||
|  | 	get_mutable( Exposed, Ref), | ||
|  | 	update_mutable( [passive(Heads)|Exposed], Ref). | ||
|  | 
 | ||
|  | de_expose( Ref) :- | ||
|  | 	get_mutable( [_|Exposed], Ref), | ||
|  | 	update_mutable( Exposed, Ref). | ||
|  | 
 | ||
|  | % | ||
|  | % Prefer passive over active (cheaper to deal with). | ||
|  | % | ||
|  | is_exposed( Constraint, Suspension, Continuation) :- | ||
|  | 	get_exposed( Ref), | ||
|  | 	get_mutable( Exposed, Ref), | ||
|  | 	is_exposed( Exposed, Constraint, Suspension, Continuation). | ||
|  | 
 | ||
|  | is_exposed( [E|Es], Constraint, Suspension, Continuation) :- | ||
|  | 	is_exposed( E, Constraint, Suspension, Continuation, Es). | ||
|  | 
 | ||
|  | is_exposed( active(Head,Susp,Heads,Cont), Constraint, Suspension, Continuation, Es) :- | ||
|  | 	( member( C#Suspension, Heads), | ||
|  | 	  Constraint == C -> | ||
|  | 	    Continuation = true | ||
|  | 	; Constraint == Head -> | ||
|  | 	    ( is_exposed( Es, Constraint, Suspension, true) -> % prefer | ||
|  | 		Continuation = true | ||
|  | 	    ; | ||
|  | 		Continuation = Cont, | ||
|  | 		Suspension = Susp | ||
|  | 	    ) | ||
|  | 	; | ||
|  | 	    is_exposed( Es, Constraint, Suspension, Continuation) | ||
|  | 	). | ||
|  | is_exposed( passive(Heads), Constraint, Suspension, Continuation, Es) :- | ||
|  | 	( member( C#Suspension, Heads), | ||
|  | 	  Constraint == C -> | ||
|  | 	    Continuation = true | ||
|  | 	; | ||
|  | 	    is_exposed( Es, Constraint, Suspension, Continuation) | ||
|  | 	). | ||
|  | 
 | ||
|  | get_exposed( Ref) :- | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	( get_atts( Global, exposed(Ref)) -> | ||
|  | 	    true | ||
|  | 	; | ||
|  | 	    create_mutable( [], Ref), | ||
|  | 	    put_atts( Global, exposed(Ref)) | ||
|  | 	). | ||
|  | 
 | ||
|  | get_dbg_state( Ref) :- | ||
|  | 	global_term_ref_1( Global), | ||
|  | 	( get_atts( Global, dbg_state(Ref)) -> | ||
|  | 	    true | ||
|  | 	; | ||
|  | 	    create_mutable( [], Ref), | ||
|  | 	    put_atts( Global, dbg_state(Ref)) | ||
|  | 	). | ||
|  | 
 | ||
|  | % ------------------- abstract data type for propagation rules ------------- | ||
|  | 
 | ||
|  | empty_history( E) :- empty_assoc( E). | ||
|  | 
 | ||
|  | % | ||
|  | % assert: constraints/tuples are comparable directly | ||
|  | % | ||
|  | novel_production( Self, Tuple) :- | ||
|  | 	arg( 5, Self, Ref), | ||
|  | 	get_mutable( History, Ref), | ||
|  | 	( get_assoc( Tuple, History, _) -> | ||
|  | 	    fail | ||
|  | 	; | ||
|  | 	    true | ||
|  | 	). | ||
|  | 
 | ||
|  | % | ||
|  | % Not folded with novel_production/2 because guard checking | ||
|  | % goes in between the two calls. | ||
|  | % | ||
|  | extend_history( Self, Tuple) :- | ||
|  | 	arg( 5, Self, Ref), | ||
|  | 	get_mutable( History, Ref), | ||
|  | 	put_assoc( Tuple, History, x, NewHistory), | ||
|  | 	update_mutable( NewHistory, Ref). | ||
|  | 
 | ||
|  | :- load_foreign_resource(library(system(chr))). | ||
|  | 
 | ||
|  | end_of_file. | ||
|  | 
 |