120 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Prolog
		
	
	
	
	
	
			
		
		
	
	
			120 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Prolog
		
	
	
	
	
	
| /*************************************************************************
 | |
| *									 *
 | |
| *	 YAP Prolog 							 *
 | |
| *									 *
 | |
| *	Yap Prolog was developed at NCCUP - Universidade do Porto	 *
 | |
| *									 *
 | |
| * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 *
 | |
| *									 *
 | |
| **************************************************************************
 | |
| *									 *
 | |
| * File:		eval.yap						 *
 | |
| * Last rev:								 *
 | |
| * mods:									 *
 | |
| * comments:	optimise disjunction handling				 *
 | |
| *									 *
 | |
| *************************************************************************/
 | |
| 
 | |
| %, portray_clause((H:-BF))
 | |
| '$full_clause_optimisation'(H, M, B0, BF) :-
 | |
| 	'$localise_vars_opt'(H, M, B0, BF), !.
 | |
| 
 | |
| % the idea here is to make global variables in disjunctions
 | |
| % local.
 | |
| '$localise_vars_opt'(H, M, (B1;B2), (NB1;NB2)) :-
 | |
| 	'$variables_in_term'(H, [], LV),
 | |
| 	'$localise_vars'(B1, M, NB1, LV, LV, []),
 | |
| 	'$localise_disj_vars'(B2, M, NB2, LV, LV, []).
 | |
| 
 | |
| '$localise_disj_vars'((B;B2), M, (NB ; NB2), LV, LV0, LEqs) :- !,
 | |
| 	'$localise_vars'(B, M, NB, LV, LV0, LEqs),
 | |
| 	'$localise_disj_vars'(B2, M, NB2, LV, LV0, LEqs).	
 | |
| '$localise_disj_vars'(B2, M, NB, LV, LV0, LEqs) :-
 | |
| 	'$localise_vars'(B2, M, NB, LV, LV0, LEqs).
 | |
| 
 | |
| '$localise_vars'((A->B), M, (A->NB), LV, LV0, LEqs) :-
 | |
| 	'$safe_guard'(A, M), !,
 | |
| 	'$variables_in_term'(A, LV, LV1),
 | |
| 	'$localise_vars'(B, M,  NB, LV1, LV0, LEqs).
 | |
| '$localise_vars'((A;B), M, (NA;NB), LV1, LV0, LEqs) :- !,
 | |
| 	'$localise_vars'(A, M,  NA, LV1, LV0, LEqs),
 | |
| 	'$localise_disj_vars'(B, M,  NB, LV1, LV0, LEqs).
 | |
| '$localise_vars'(((A,B),C), M, NG, LV, LV0, LEqs) :- !,
 | |
| 	'$flatten_bd'((A,B),C,NB),
 | |
| 	'$localise_vars'(NB, M, NG, LV, LV0, LEqs).
 | |
| '$localise_vars'((!,B), M, (!,NB), LV, LV0, LEqs) :- !,
 | |
| 	'$localise_vars'(B, M,  NB, LV, LV0, LEqs).
 | |
| '$localise_vars'((X=Y,B), M, (X=Y,NB1), LV, LV0, LEqs) :-
 | |
| 	var(X), var(Y), !,
 | |
| 	'$localise_vars'(B, M,  NB1, LV, LV0, [X,Y|LEqs]).
 | |
| '$localise_vars'((G,B), M, (G,NB1), LV, LV0, LEqs) :-
 | |
| 	'$safe_builtin'(G, M), !,
 | |
| 	'$variables_in_term'(G, LV, LV1),
 | |
| 	'$add_extra_safe'(G, NLV0, LV0),
 | |
| 	'$localise_vars'(B, M,  NB1, LV1, NLV0, LEqs).
 | |
| '$localise_vars'((G1,B1), _, O, LV, LV0, LEqs) :- !,
 | |
| 	terms:variables_within_term(LV, B1, Commons),
 | |
| 	terms:new_variables_in_term(LV, B1, New),
 | |
| 	copy_term(Commons+New+LEqs+B1, NCommons+NNew+NLEqs+NB1),
 | |
| 	NNew = New,
 | |
| 	NLEqs = LEqs,
 | |
| 	'$gen_equals'(Commons, NCommons, LV0, (G1,NB1), O).
 | |
| '$localise_vars'(G, _, G, _, _, _).
 | |
| 
 | |
| '$gen_equals'([], [], _, O, O).
 | |
| '$gen_equals'([V|Commons],[NV|NCommons], LV0, O, NO) :- V == NV, !,
 | |
| 	'$gen_equals'(Commons,NCommons, LV0, O, NO).
 | |
| '$gen_equals'([V|Commons],[NV|NCommons], LV0, O, OO) :-
 | |
| 	'$vmember'(V,LV0),
 | |
|          OO = (V=NV,'$safe'(NV),NO),
 | |
| 	'$gen_equals'(Commons,NCommons, LV0, O, NO).
 | |
| '$gen_equals'([V|Commons],[NV|NCommons], LV0, O, OO) :- 
 | |
|          OO = (V=NV,NO),
 | |
| 	'$gen_equals'(Commons,NCommons, LV0, O, NO).
 | |
| 
 | |
| '$safe_guard'((A,B), M) :- !,
 | |
| 	    '$safe_guard'(A, M),
 | |
| 	    '$safe_guard'(B, M).
 | |
| '$safe_guard'((A;B), M) :- !,
 | |
| 	    '$safe_guard'(A, M),
 | |
| 	    '$safe_guard'(B, M).
 | |
| '$safe_guard'(A, M) :- !,
 | |
| 	    '$safe_builtin'(A, M).
 | |
| 
 | |
| '$safe_builtin'(G, Mod) :-
 | |
| 	'$flags'(G, Mod, Fl, Fl),
 | |
| 	Fl /\ 0x00008880 =\= 0.
 | |
| 
 | |
| '$vmember'(V,[V1|_]) :- V == V1, !.
 | |
| '$vmember'(V,[_|LV0]) :-
 | |
| 	'$vmember'(V,LV0).
 | |
| 
 | |
| '$flatten_bd'((A,B),R,NB) :- !,
 | |
| 	'$flatten_bd'(B,R,R1),
 | |
| 	'$flatten_bd'(A,R1,NB).
 | |
| '$flatten_bd'(A,R,(A,R)).
 | |
| 
 | |
| '$add_extra_safe'('$plus'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$minus'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$times'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$div'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$and'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$or'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$sll'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'('$slr'(_,_,V)) --> !, [V].
 | |
| '$add_extra_safe'(C=D,A,B) :-
 | |
|    !,
 | |
|    ( compound(C) ->
 | |
|      '$variables_in_term'(C,E,A)
 | |
|    ;
 | |
|      E=A
 | |
|    ),
 | |
|    ( compound(D) ->
 | |
|      '$variables_in_term'(D,B,E)
 | |
|    ;
 | |
|      B=E
 | |
|    ).
 | |
| '$add_extra_safe'(_) --> [].
 | |
| 
 | |
| 
 |