1708 lines
40 KiB
Prolog
1708 lines
40 KiB
Prolog
/*************************************************************************
|
|
* *
|
|
* YAP Prolog *
|
|
* *
|
|
* Yap Prolog was developed at NCCUP - Universidade do Porto *
|
|
* *
|
|
* Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-2014 *
|
|
* *
|
|
**************************************************************************
|
|
* Ptv *
|
|
* File: boot.yap *
|
|
* Last rev: 8/2/88 *
|
|
* mods: *
|
|
* comments: boot file for Prolog *
|
|
* *
|
|
*************************************************************************/
|
|
|
|
/**
|
|
@file boot.yap
|
|
@defgroup YAPControl Control Predicates
|
|
@ingroup builtins
|
|
|
|
*/
|
|
|
|
%% @{
|
|
|
|
|
|
|
|
/** @pred :_P_ ; :_Q_ is iso
|
|
Disjunction of goals (or).
|
|
|
|
Example:
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
p(X) :- q(X); r(X).
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
should be read as "p( _X_) if q( _X_) or r( _X_)".
|
|
|
|
|
|
*/
|
|
|
|
/** @pred \+ :_P_ is iso
|
|
Negation by failure.
|
|
|
|
Goal _P_ is not provable. The execution of this predicate fails if
|
|
and only if the goal _P_ finitely succeeds. It is not a true logical
|
|
negation, which is impossible in standard Prolog, but
|
|
"negation-by-failure".
|
|
|
|
This predicate might be defined as:
|
|
|
|
~~~~~~~~~~~~
|
|
\+(P) :- P, !, fail.
|
|
\+(_).
|
|
~~~~~~~~~~~~
|
|
if _P_ did not include "cuts".
|
|
|
|
If _P_ includes cuts, the cuts are defined to be scoped by _P_: they cannot cut over the calling prredicate.
|
|
|
|
~~~~~~~~~~~~
|
|
go(P).
|
|
|
|
:- \+ P, !, fail.
|
|
\+(_).
|
|
~~~~~~~~~~~~
|
|
|
|
*/
|
|
|
|
/** @pred :_Condition__ -> :_Action_ is iso
|
|
|
|
|
|
Read as "if-then-else" or "commit". This operator is similar to the
|
|
conditional operator of imperative languages and can be used alone or
|
|
with an else part as follows:
|
|
|
|
|
|
~~~~~
|
|
+P -> +Q
|
|
~~~~~
|
|
|
|
"if P then Q".
|
|
|
|
|
|
~~~~~
|
|
+P -> +Q; +R
|
|
~~~~~
|
|
|
|
"if P then Q else R".
|
|
|
|
These two predicates could be defined respectively in Prolog as:
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
(P -> Q) :- P, !, Q.
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
and
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
(P -> Q; R) :- P, !, Q.
|
|
(P -> Q; R) :- R.
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
if there were no "cuts" in _P_, _Q_ and _R_.
|
|
|
|
Note that the commit operator works by "cutting" any alternative
|
|
solutions of _P_.
|
|
|
|
Note also that you can use chains of commit operators like:
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
P -> Q ; R -> S ; T.
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
Note that `(->)/2` does not affect the scope of cuts in its
|
|
arguments.
|
|
|
|
|
|
*/
|
|
|
|
/** @pred :_Condition_ *-> :_Action_ is iso
|
|
|
|
This construct implements the so-called <em>soft-cut</em>. The control is
|
|
defined as follows:
|
|
+ If _Condition_ succeeds at least once, the
|
|
semantics is the same as ( _Condition_, _Action_).
|
|
|
|
+ If
|
|
_Condition_ does not succeed, the semantics is that of (\\+
|
|
_Condition_, _Else_).
|
|
|
|
In other words, if _Condition_
|
|
succeeds at least once, simply behave as the conjunction of
|
|
_Condition_ and _Action_, otherwise execute _Else_.
|
|
|
|
The construct _A *-> B_, i.e. without an _Else_ branch, is
|
|
translated as the normal conjunction _A_, _B_.
|
|
|
|
|
|
*/
|
|
|
|
/** @pred ! is iso
|
|
|
|
|
|
Read as "cut". Cuts any choices taken in the current procedure.
|
|
When first found "cut" succeeds as a goal, but if backtracking should
|
|
later return to it, the parent goal (the one which matches the head of
|
|
the clause containing the "cut", causing the clause activation) will
|
|
fail. This is an extra-logical predicate and cannot be explained in
|
|
terms of the declarative semantics of Prolog.
|
|
|
|
example:
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
member(X,[X|_]).
|
|
member(X,[_|L]) :- member(X,L).
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
With the above definition
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
?- member(X,[1,2,3]).
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
will return each element of the list by backtracking. With the following
|
|
definition:
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
member(X,[X|_]) :- !.
|
|
member(X,[_|L]) :- member(X,L).
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
the same query would return only the first element of the
|
|
list, since backtracking could not "pass through" the cut.
|
|
|
|
*/
|
|
|
|
system_module(_Mod, _SysExps, _Decls) :- ! .
|
|
% new_system_module(Mod).
|
|
|
|
use_system_module(_init, _SysExps) :- !.
|
|
|
|
private(_).
|
|
|
|
%
|
|
% boootstrap predicates.
|
|
%
|
|
:- system_module( '$_boot', [
|
|
bootstrap/1,
|
|
call/1,
|
|
catch/3,
|
|
catch_ball/2,
|
|
expand_term/2,
|
|
import_system_module/2,
|
|
incore/1,
|
|
(not)/1,
|
|
repeat/0,
|
|
throw/1,
|
|
true/0], ['$$compile'/4,
|
|
'$call'/4,
|
|
'$catch'/3,
|
|
'$check_callable'/2,
|
|
'$check_head_and_body'/4,
|
|
'$check_if_reconsulted'/2,
|
|
'$clear_reconsulting'/0,
|
|
'$command'/4,
|
|
'$cut_by'/1,
|
|
'$disable_debugging'/0,
|
|
'$do_live'/0,
|
|
'$'/0,
|
|
'$find_goal_definition'/4,
|
|
'$handle_throw'/3,
|
|
'$head_and_body'/3,
|
|
'$inform_as_reconsulted'/2,
|
|
'$init_system'/0,
|
|
'$init_win_graphics'/0,
|
|
'$live'/0,
|
|
'$loop'/2,
|
|
'$meta_call'/2,
|
|
'$prompt_alternatives_on'/1,
|
|
'$run_at_thread_start'/0,
|
|
'$system_catch'/4,
|
|
'$undefp'/1,
|
|
'$version'/0]).
|
|
|
|
:- use_system_module( '$_absf', ['$system_library_directories'/2]).
|
|
|
|
:- use_system_module( '$_checker', ['$check_term'/5,
|
|
'$sv_warning'/2]).
|
|
|
|
:- use_system_module( '$_consult', ['$csult'/2]).
|
|
|
|
:- use_system_module( '$_control', ['$run_atom_goal'/1]).
|
|
|
|
:- use_system_module( '$_directives', ['$all_directives'/1,
|
|
'$exec_directives'/5]).
|
|
|
|
:- use_system_module( '$_errors', ['$do_error'/2]).
|
|
|
|
:- use_system_module( '$_grammar', ['$translate_rule'/2]).
|
|
|
|
:- use_system_module( '$_modules', ['$get_undefined_pred'/4,
|
|
'$meta_expansion'/6,
|
|
'$module_expansion'/6]).
|
|
|
|
:- use_system_module( '$_preddecls', ['$dynamic'/2]).
|
|
|
|
:- use_system_module( '$_preds', ['$assert_static'/5,
|
|
'$assertz_dynamic'/4,
|
|
'$init_preds'/0,
|
|
'$unknown_error'/1,
|
|
'$unknown_warning'/1]).
|
|
|
|
:- use_system_module( '$_qly', ['$init_state'/0]).
|
|
|
|
:- use_system_module( '$_strict_iso', ['$check_iso_strict_clause'/1,
|
|
'$iso_check_goal'/2]).
|
|
|
|
|
|
|
|
'$early_print_message'(_, absolute_file_path(X, Y)) :- !,
|
|
format(user_error, X, Y), nl(user_error).
|
|
'$early_print_message'(_, loading( C, F)) :- !,
|
|
format(user_error, '~*|% ~a ~w...~n', [2,C,F]).
|
|
'$early_print_message'(_, loaded(F,C,M,T,H)) :- !,
|
|
format(user_error, '~*|% ~a:~w ~a ~d bytes in ~d seconds...~n', [2, M, F ,C, H, T]).
|
|
'$early_print_message'(Level, Msg) :-
|
|
source_location(F0, L),
|
|
!,
|
|
format(user_error, '~a:~d:0: unprocessed ~a ~w ~n', [F0, L,Level,Msg]).
|
|
'$early_print_message'(Level, Msg) :-
|
|
format(user_error, 'unprocessed ~a ~w ~n', [Level,Msg]).
|
|
|
|
'$handle_error'(_Action,_G0,_M0) :- fail.
|
|
|
|
% cases where we cannot afford to ever fail.
|
|
'$undefp'([_|print_message(Context, Msg)], true) :- !,
|
|
'$early_print_message'(Context, Msg).
|
|
% undef handler
|
|
'$undefp'([M0|G0], Action) :-
|
|
% make sure we do not loop on undefined predicates
|
|
'$stop_creeping'(Current),
|
|
yap_flag( unknown, Action, fail),
|
|
% yap_flag( debug, Debug, false),
|
|
(
|
|
'$undefp_search'(M0:G0, NM:NG),
|
|
( M0 \== NM -> true ; G0 \== NG ),
|
|
NG \= fail
|
|
->
|
|
yap_flag( unknown, _, Action),
|
|
% yap_flag( debug, _, Debug),
|
|
(
|
|
Current == true
|
|
->
|
|
% carry on signal processing
|
|
'$start_creep'([NM|NG], creep)
|
|
;
|
|
'$execute0'(NG, NM)
|
|
)
|
|
;
|
|
yap_flag( unknown, _, Action),
|
|
'$handle_error'(Action,G0,M0)
|
|
).
|
|
|
|
/*
|
|
'$undefp'([M0|G0], Default) :-
|
|
G0 \= '$imported_predicate'(_,_,_,_),
|
|
G0 \= '$full_clause_optimisation'(_H, _M, _B0, _BF),
|
|
G0 \= '$expand_a_clause'(_,_,_,_),
|
|
G0 \= '$all_directives'(_),
|
|
format(user_error, 'ERROR: undefined ~a:~q.~n', [M0, G0]), fail.
|
|
*/
|
|
'$prepare_goals'((A,B),(NA,NB),Any) :-
|
|
!,
|
|
'$prepare_goals'(A,NA,Any),
|
|
'$prepare_goals'(B,NB,Any).
|
|
'$prepare_goals'((A;B),(NA;NB),Any) :-
|
|
!,
|
|
'$prepare_goals'(A,NA,Any),
|
|
'$prepare_goals'(B,NB,Any).
|
|
'$prepare_goals'((A->B),(NA->NB),Any) :-
|
|
!,
|
|
'$prepare_goals'(A,NA,Any),
|
|
'$prepare_goals'(B,NB,Any).
|
|
'$prepare_goals'((A*->B),(NA*->NB),Any) :-
|
|
!,
|
|
'$prepare_goals'(A,NA,Any),
|
|
'$prepare_goals'(B,NB,Any).
|
|
'$prepare_goals'((\+ A),(\+ NA),Any) :-
|
|
!,
|
|
'$prepare_goals'(A,NA,Any).
|
|
'$prepare_goals'('$do_error'(Error,Goal),
|
|
(clause_location(Call, Caller),
|
|
source_module(M),
|
|
strip_module(M:Goal,M1,NGoal),
|
|
throw(error(Error, [[g|g(M1:NGoal)],[p|Call],[e|Caller],[h|g(Head)]]))
|
|
),
|
|
true) :-
|
|
!,
|
|
'$head_and_body'(NGoal,Head,_Body).
|
|
'$prepare_goals'(X is AOB,
|
|
is(X, IOp, A, B ),
|
|
true) :-
|
|
var(X),
|
|
functor(AOB, Op, 2),
|
|
arg(1, AOB, A),
|
|
arg(2, AOB, B),
|
|
!,
|
|
'$inbrary_op_as_integer'(Op,IOp).
|
|
'$prepare_goals'((A,B),(A,B),_Any).
|
|
|
|
'$prepare_clause'((H :- B), (H:-NB)) :-
|
|
'$prepare_goals'(B,NB,Any),
|
|
Any==true.
|
|
|
|
|
|
|
|
%
|
|
%
|
|
%
|
|
|
|
/** @pred true is iso
|
|
Succeed.
|
|
|
|
Succeeds once.
|
|
*/
|
|
true :- true.
|
|
|
|
'$live' :-
|
|
'$init_system',
|
|
'$do_live'.
|
|
|
|
'$init_prolog' :-
|
|
'$init_system'.
|
|
|
|
'$do_live' :-
|
|
repeat,
|
|
'$current_module'(Module),
|
|
( Module==user ->
|
|
true % '$compile_mode'(_,0)
|
|
;
|
|
format(user_error,'[~w]~n', [Module])
|
|
),
|
|
'$system_catch'('$enter_top_level',Module,Error,'$Error'(Error)).
|
|
|
|
|
|
'$init_system' :-
|
|
get_value('$yap_inited', true), !.
|
|
'$init_system' :-
|
|
set_value('$yap_inited', true),
|
|
% do catch as early as possible
|
|
(
|
|
current_prolog_flag(halt_after_consult, false),
|
|
current_prolog_flag(verbose, normal)
|
|
% \+ '$uncaught_throw'
|
|
->
|
|
'$version'
|
|
;
|
|
true
|
|
),
|
|
% '$init_preds', % needs to be done before library_directory
|
|
% (
|
|
% retractall(user:library_directory(_)),
|
|
% '$system_library_directories'(D),
|
|
% assertz(user:library_directory(D)),
|
|
% fail
|
|
% ;
|
|
% true
|
|
% ),
|
|
current_prolog_flag(file_name_variables, OldF),
|
|
set_prolog_flag(file_name_variables, true),
|
|
'$init_consult',
|
|
set_prolog_flag(file_name_variables, OldF),
|
|
'$init_globals',
|
|
set_prolog_flag(fileerrors, true),
|
|
set_value('$gc',on),
|
|
('$exit_undefp' -> true ; true),
|
|
prompt1(' ?- '),
|
|
set_prolog_flag(debug, false),
|
|
% simple trick to find out if this is we are booting from Prolog.
|
|
% boot from a saved state
|
|
(
|
|
'$undefined'('$init_preds',prolog)
|
|
->
|
|
get_value('$consult_on_boot',X),
|
|
(
|
|
X \= []
|
|
->
|
|
bootstrap(X),
|
|
module( user ),
|
|
qsave_program( 'startup.yss')
|
|
;
|
|
true
|
|
)
|
|
;
|
|
'$init_state'
|
|
),
|
|
'$db_clean_queues'(0),
|
|
% this must be executed from C-code.
|
|
% '$startup_saved_state',
|
|
set_input(user_input),
|
|
set_output(user_output),
|
|
'$init_or_threads',
|
|
'$run_at_thread_start'.
|
|
|
|
'$init_globals' :-
|
|
% set_prolog_flag(break_level, 0),
|
|
% '$set_read_error_handler'(error), let the user do that
|
|
nb_setval('$chr_toplevel_show_store',false).
|
|
|
|
'$init_consult' :-
|
|
set_value('$open_expands_filename',true),
|
|
nb_setval('$assert_all',off),
|
|
nb_setval('$if_level',0),
|
|
nb_setval('$endif',off),
|
|
nb_setval('$initialization_goals',off),
|
|
nb_setval('$included_file',[]),
|
|
\+ '$undefined'('$init_preds',prolog),
|
|
'$init_preds',
|
|
fail.
|
|
'$init_consult'.
|
|
|
|
'$init_win_graphics' :-
|
|
'$undefined'(window_title(_,_), system), !.
|
|
'$init_win_graphics' :-
|
|
load_files([library(win_menu)], [silent(true),if(not_loaded)]),
|
|
fail.
|
|
'$init_win_graphics'.
|
|
|
|
'$init_or_threads' :-
|
|
'$c_yapor_workers'(W), !,
|
|
'$start_orp_threads'(W).
|
|
'$init_or_threads'.
|
|
|
|
'$start_orp_threads'(1) :- !.
|
|
'$start_orp_threads'(W) :-
|
|
thread_create('$c_worker',_,[detached(true)]),
|
|
W1 is W-1,
|
|
'$start_orp_threads'(W1).
|
|
|
|
|
|
% Start file for yap
|
|
|
|
/* I/O predicates */
|
|
|
|
/* meaning of flags for '$write' is
|
|
1 quote illegal atoms
|
|
2 ignore operator declarations
|
|
4 output '$VAR'(N) terms as A, B, C, ...
|
|
8 use portray(_)
|
|
*/
|
|
|
|
/* main execution loop */
|
|
'$read_toplevel'(Goal, Bindings) :-
|
|
'$prompt',
|
|
'$system_catch'(read_term(user_input,
|
|
Goal,
|
|
[variable_names(Bindings), syntax_errors(dec10)]),
|
|
prolog, E, '$handle_toplevel_error'( E) ).
|
|
|
|
'$handle_toplevel_error'( syntax_error(_)) :-
|
|
!,
|
|
fail.
|
|
'$handle_toplevel_error'( error(io_error(read,user_input),_)) :-
|
|
!.
|
|
'$handle_toplevel_error'(_, E) :-
|
|
throw(E).
|
|
|
|
|
|
/** @pred stream_property( _Stream_, _Prop_)
|
|
|
|
*/
|
|
|
|
% reset alarms when entering top-level.
|
|
'$enter_top_level' :-
|
|
'$alarm'(0, 0, _, _),
|
|
fail.
|
|
'$enter_top_level' :-
|
|
'$clean_up_dead_clauses',
|
|
fail.
|
|
'$enter_top_level' :-
|
|
get_value('$top_level_goal',GA), GA \= [], !,
|
|
set_value('$top_level_goal',[]),
|
|
'$run_atom_goal'(GA),
|
|
current_prolog_flag(break_level, BreakLevel),
|
|
(
|
|
BreakLevel \= 0
|
|
->
|
|
true
|
|
;
|
|
'$pred_exists'(halt(_), user)
|
|
->
|
|
halt(0)
|
|
;
|
|
'$halt'(0)
|
|
).
|
|
'$enter_top_level' :-
|
|
flush_output,
|
|
'$run_toplevel_hooks',
|
|
prompt1(' ?- '),
|
|
'$read_toplevel'(Command,Varnames),
|
|
nb_setval('$spy_gn',1),
|
|
% stop at spy-points if debugging is on.
|
|
nb_setval('$debug_run',off),
|
|
nb_setval('$debug_jump',off),
|
|
'$command'(Command,Varnames,_Pos,top),
|
|
current_prolog_flag(break_level, BreakLevel),
|
|
(
|
|
BreakLevel \= 0
|
|
->
|
|
true
|
|
;
|
|
'$pred_exists'(halt(_), user)
|
|
-> halt(0)
|
|
;
|
|
'$halt'(0)
|
|
).
|
|
|
|
|
|
'$erase_sets' :-
|
|
eraseall('$'),
|
|
eraseall('$$set'),
|
|
eraseall('$$one'),
|
|
eraseall('$reconsulted'), fail.
|
|
'$erase_sets' :- \+ recorded('$path',_,_), recorda('$path',"",_).
|
|
'$erase_sets'.
|
|
|
|
'$version' :-
|
|
current_prolog_flag(version_git,VersionGit),
|
|
current_prolog_flag(compiled_at,AT),
|
|
current_prolog_flag(version_data, yap(Mj, Mi, Patch, _) ),
|
|
sub_atom( VersionGit, 0, 8, _, VERSIONGIT ),
|
|
format(user_error, '% YAP ~d.~d.~d-~a (compiled ~a)~n', [Mj,Mi, Patch, VERSIONGIT, AT]),
|
|
fail.
|
|
'$version'.
|
|
|
|
/** @pred repeat is iso
|
|
Succeeds repeatedly.
|
|
|
|
In the next example, `repeat` is used as an efficient way to implement
|
|
a loop. The next example reads all terms in a file:
|
|
~~~~~~~~~~~~~{.prolog}
|
|
a :- repeat, read(X), write(X), nl, X=end_of_file, !.
|
|
~~~~~~~~~~~~~
|
|
the loop is effectively terminated by the cut-goal, when the test-goal
|
|
`X=end` succeeds. While the test fails, the goals `read(X)`,
|
|
`write(X)`, and `nl` are executed repeatedly, because
|
|
backtracking is caught by the `repeat` goal.
|
|
|
|
The built-in `repeat/0` could be defined in Prolog by:
|
|
|
|
~~~~~{.prolog}
|
|
repeat.
|
|
repeat :- repeat.
|
|
~~~~~
|
|
|
|
The predicate between/3 can be used to iterate for a pre-defined
|
|
number of steps.
|
|
|
|
*/
|
|
repeat :- '$repeat'.
|
|
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat'.
|
|
'$repeat' :- '$repeat'.
|
|
|
|
'$start_corouts' :-
|
|
eraseall('$corout'),
|
|
eraseall('$result'),
|
|
eraseall('$actual'),
|
|
fail.
|
|
'$start_corouts' :- recorda('$actual',main,_),
|
|
recordz('$corout','$corout'(main,main,'$corout'([],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[])),_Ref),
|
|
recorda('$result',going,_).
|
|
|
|
'$command'(C,VL,Pos,Con) :-
|
|
current_prolog_flag(strict_iso, true), !, /* strict_iso on */
|
|
'$execute_command'(C,VL,Pos,Con,_Source).
|
|
'$command'(C,VL,Pos,Con) :-
|
|
( (Con = top ; var(C) ; C = [_|_]) ->
|
|
'$execute_command'(C,VL,Pos,Con,C), ! ;
|
|
% do term expansion
|
|
expand_term(C, EC),
|
|
% execute a list of commands
|
|
'$execute_commands'(EC,VL,Pos,Con,_Source),
|
|
% succeed only if the *original* was at end of file.
|
|
C == end_of_file
|
|
).
|
|
|
|
%
|
|
% Hack in case expand_term has created a list of commands.
|
|
%
|
|
'$execute_commands'(V,_,_,_,Source) :- var(V), !,
|
|
'$do_error'(instantiation_error,meta_call(Source)).
|
|
'$execute_commands'([],_,_,_,_) :- !.
|
|
'$execute_commands'([C|Cs],VL,Pos,Con,Source) :- !,
|
|
(
|
|
'$system_catch'('$execute_command'(C,VL,Pos,Con,C),prolog,Error,'$LoopError'(Error, Con)),
|
|
fail
|
|
;
|
|
'$execute_commands'(Cs,VL,Pos,Con,Source)
|
|
).
|
|
'$execute_commands'(C,VL,Pos,Con,Source) :-
|
|
'$execute_command'(C,VL,Pos,Con,Source).
|
|
|
|
%
|
|
%
|
|
%
|
|
|
|
'$execute_command'(C,_,_,top,Source) :-
|
|
var(C),
|
|
!,
|
|
'$do_error'(instantiation_error,meta_call(Source)).
|
|
'$execute_command'(C,_,_,top,Source) :-
|
|
number(C),
|
|
!,
|
|
'$do_error'(type_error(callable,C),meta_call(Source)).
|
|
'$execute_command'(R,_,_,top,Source) :-
|
|
db_reference(R),
|
|
!,
|
|
'$do_error'(type_error(callable,R),meta_call(Source)).
|
|
'$execute_command'(end_of_file,_,_,_,_) :- !.
|
|
'$execute_command'(Command,_,_,_,_) :-
|
|
'$nb_getval'('$if_skip_mode', skip, fail),
|
|
\+ '$if_directive'(Command),
|
|
!.
|
|
'$execute_command'((:-G),VL,Pos,Option,_) :-
|
|
% !,
|
|
Option \= top, !,
|
|
'$current_module'(M),
|
|
% allow user expansion
|
|
expand_term((:- M:G), O),
|
|
(
|
|
O = (:- G1)
|
|
->
|
|
'$process_directive'(G1, Option, M, VL, Pos)
|
|
;
|
|
'$execute_commands'(O,VL,Pos,Option,O)
|
|
).
|
|
'$execute_command'((?-G), VL, Pos, Option, Source) :-
|
|
Option \= top,
|
|
!,
|
|
'$execute_command'(G, VL, Pos, top, Source).
|
|
'$execute_command'(G, VL, Pos, Option, Source) :-
|
|
'$continue_with_command'(Option, VL, Pos, G, Source).
|
|
|
|
%
|
|
% This command is very different depending on the language mode we are in.
|
|
%
|
|
% ISO only wants directives in files
|
|
% SICStus accepts everything in files
|
|
% YAP accepts everything everywhere
|
|
%
|
|
'$process_directive'(G, top, M, VL, Pos) :-
|
|
current_prolog_flag(language_mode, yap), !, /* strict_iso on */
|
|
'$process_directive'(G, consult, M, VL, Pos).
|
|
'$process_directive'(G, top, _, _, _) :-
|
|
!,
|
|
'$do_error'(context_error((:- G),clause),query).
|
|
%
|
|
% allow modules
|
|
%
|
|
'$process_directive'(M:G, Mode, _, VL, Pos) :- !,
|
|
'$process_directive'(G, Mode, M, VL, Pos).
|
|
%
|
|
% default case
|
|
%
|
|
'$process_directive'(Gs, Mode, M, VL, Pos) :-
|
|
'$all_directives'(Gs), !,
|
|
'$exec_directives'(Gs, Mode, M, VL, Pos).
|
|
|
|
%
|
|
% ISO does not allow goals (use initialization).
|
|
%
|
|
'$process_directive'(D, _, M, _VL, _Pos) :-
|
|
current_prolog_flag(language_mode, iso),
|
|
!, % ISO Prolog mode, go in and do it,
|
|
'$do_error'(context_error((:- M:D),query),directive).
|
|
%
|
|
% but YAP and SICStus does.
|
|
%
|
|
'$process_directive'(G, Mode, M, VL, Pos) :-
|
|
( '$undefined'('$save_directive'(G, Mode, M, VL, Pos),prolog) ->
|
|
true
|
|
;
|
|
'$save_directive'(G, Mode, M, VL, Pos)
|
|
->
|
|
true
|
|
;
|
|
true
|
|
),
|
|
(
|
|
'$execute'(M:G)
|
|
->
|
|
true
|
|
;
|
|
format(user_error,':- ~w:~w failed.~n',[M,G])
|
|
).
|
|
|
|
'$continue_with_command'(Where,V,'$stream_position'(C,_P,A1,A2,A3),'$source_location'(_F,L):G,Source) :-
|
|
!,
|
|
'$continue_with_command'(Where,V,'$stream_position'(C,L,A1,A2,A3),G,Source).
|
|
'$continue_with_command'(reconsult,V,Pos,G,Source) :-
|
|
% writeln(G),
|
|
'$go_compile_clause'(G,V,Pos,reconsult,Source),
|
|
fail.
|
|
'$continue_with_command'(consult,V,Pos,G,Source) :-
|
|
'$go_compile_clause'(G,V,Pos,consult,Source),
|
|
fail.
|
|
'$continue_with_command'(top,V,_,G,_) :-
|
|
'$query'(G,V).
|
|
|
|
%%
|
|
% @pred '$go_compile_clause'(G,Vs,Pos, Where, Source) is det
|
|
%
|
|
% interfaces the loader and the compiler
|
|
% not 100% compatible with SICStus Prolog, as SICStus Prolog would put
|
|
% module prefixes all over the place, although unnecessarily so.
|
|
%
|
|
% @param [in] _G_ is the clause to compile
|
|
% @param [in] _Vs_ a list of variables and their name
|
|
% @param [in] _Pos_ the source-code position
|
|
% @param [in] _N_ a flag telling whether to add first or last
|
|
% @param [out] _Source_ the user-tranasformed clause
|
|
'$go_compile_clause'(G, _Vs, _Pos, Where, Source) :-
|
|
'$precompile_term'(G, Source, G1),
|
|
!,
|
|
'$$compile'(G1, Where, Source, _).
|
|
'$go_compile_clause'(G,_Vs,_Pos, _Where, _Source) :-
|
|
throw(error(system, compilation_failed(G))).
|
|
|
|
'$$compile'(C, Where, C0, R) :-
|
|
'$head_and_body'( C, MH, B ),
|
|
strip_module( MH, Mod, H),
|
|
(
|
|
'$undefined'(H, Mod)
|
|
->
|
|
'$init_pred'(H, Mod, Where)
|
|
;
|
|
true
|
|
),
|
|
% writeln(Mod:((H:-B))),
|
|
'$compile'((H:-B), Where, C0, Mod, R).
|
|
|
|
'$init_pred'(H, Mod, _Where ) :-
|
|
recorded('$import','$import'(NM,Mod,NH,H,_,_),RI),
|
|
% NM \= Mod,
|
|
functor(NH,N,Ar),
|
|
'$early_print'(warning,redefine_imported(Mod,NM,Mod:N/Ar)),
|
|
erase(RI),
|
|
fail.
|
|
'$init_pred'(H, Mod, Where ) :-
|
|
'$init_as_dynamic'(Where),
|
|
!,
|
|
functor(H, Na, Ar),
|
|
'$dynamic'(Na/Ar, Mod).
|
|
'$init_pred'(_H, _Mod, _Where ).
|
|
|
|
'$init_as_dynamic'( asserta ).
|
|
'$init_as_dynamic'( assertz ).
|
|
'$init_as_dynamic'( consult ) :-
|
|
'$nb_getval'('$assert_all',on,fail).
|
|
'$init_as_dynamic'( reconsult ) :-
|
|
'$nb_getval'('$assert_all',on,fail).
|
|
|
|
'$check_if_reconsulted'(N,A) :-
|
|
once(recorded('$reconsulted',N/A,_)),
|
|
recorded('$reconsulted',X,_),
|
|
( X = N/A , !;
|
|
X = '$', !, fail;
|
|
fail
|
|
).
|
|
|
|
'$inform_as_reconsulted'(N,A) :-
|
|
recorda('$reconsulted',N/A,_).
|
|
|
|
'$clear_reconsulting' :-
|
|
recorded('$reconsulted',X,Ref),
|
|
erase(Ref),
|
|
X == '$',
|
|
!,
|
|
( recorded('$reconsulting',_,R) -> erase(R) ).
|
|
|
|
'$prompt_alternatives_on'(determinism).
|
|
|
|
/* Executing a query */
|
|
|
|
'$query'(end_of_file,_).
|
|
'$query'(G,[]) :-
|
|
'$prompt_alternatives_on'(OPT),
|
|
( OPT = groundness ; OPT = determinism),
|
|
!,
|
|
'$yes_no'(G,(?-)).
|
|
'$query'(G,V) :-
|
|
(
|
|
'$current_choice_point'(CP),
|
|
'$current_module'(M),
|
|
'$user_call'(G, M),
|
|
'$current_choice_point'(NCP),
|
|
'$delayed_goals'(G, V, NV, LGs, DCP),
|
|
'$write_answer'(NV, LGs, Written),
|
|
'$write_query_answer_true'(Written),
|
|
(
|
|
'$prompt_alternatives_on'(determinism), CP == NCP, DCP = 0
|
|
->
|
|
format(user_error, '.~n', []),
|
|
!
|
|
;
|
|
'$another',
|
|
!
|
|
),
|
|
fail
|
|
;
|
|
'$out_neg_answer'
|
|
).
|
|
|
|
'$yes_no'(G,C) :-
|
|
'$current_module'(M),
|
|
'$do_yes_no'(G,M),
|
|
'$delayed_goals'(G, [], NV, LGs, _),
|
|
'$write_answer'(NV, LGs, Written),
|
|
( Written = [] ->
|
|
!,'$present_answer'(C, true)
|
|
;
|
|
'$another', !
|
|
),
|
|
fail.
|
|
'$yes_no'(_,_) :-
|
|
'$out_neg_answer'.
|
|
|
|
'$add_env_and_fail' :- fail.
|
|
|
|
%
|
|
% *-> at this point would require compiler support, which does not exist.
|
|
%
|
|
'$delayed_goals'(G, V, NV, LGs, NCP) :-
|
|
(
|
|
CP is '$last_choice_pt',
|
|
'$current_choice_point'(NCP1),
|
|
'$attributes':delayed_goals(G, V, NV, LGs),
|
|
'$current_choice_point'(NCP2),
|
|
'$clean_ifcp'(CP),
|
|
NCP is NCP2-NCP1
|
|
;
|
|
copy_term_nat(V, NV),
|
|
LGs = [],
|
|
% term_factorized(V, NV, LGs),
|
|
NCP = 0
|
|
).
|
|
|
|
'$out_neg_answer' :-
|
|
'$early_print'( help, false),
|
|
fail.
|
|
|
|
|
|
'$do_yes_no'([X|L], M) :-
|
|
!,
|
|
'$csult'([X|L], M).
|
|
'$do_yes_no'(G, M) :-
|
|
'$user_call'(G, M).
|
|
|
|
'$write_query_answer_true'([]) :- !,
|
|
format(user_error,true,[]).
|
|
'$write_query_answer_true'(_).
|
|
|
|
|
|
%
|
|
% present_answer has three components. First it flushes the streams,
|
|
% then it presents the goals, and last it shows any goals frozen on
|
|
% the arguments.
|
|
%
|
|
'$present_answer'(_,_):-
|
|
flush_output,
|
|
fail.
|
|
'$present_answer'((?-), Answ) :-
|
|
current_prolog_flag(break_level, BL ),
|
|
( BL \= 0 -> format(user_error, '[~p] ',[BL]) ;
|
|
true ),
|
|
( current_prolog_flag(toplevel_print_options, Opts) ->
|
|
write_term(user_error,Answ,Opts) ;
|
|
format(user_error,'~w',[Answ])
|
|
),
|
|
format(user_error,'.~n', []).
|
|
|
|
'$another' :-
|
|
format(user_error,' ? ',[]),
|
|
get0(user_input,C),
|
|
'$do_another'(C).
|
|
|
|
'$do_another'(C) :-
|
|
( C== 0'; -> skip(user_input,10), %'
|
|
% '$add_nl_outside_console',
|
|
fail
|
|
;
|
|
C== 10
|
|
->
|
|
'$add_nl_outside_console',
|
|
(
|
|
'$undefined'('$early_print'(_,_),prolog)
|
|
->
|
|
format(user_error,'yes~n', [])
|
|
;
|
|
'$early_print'(help,yes)
|
|
)
|
|
;
|
|
C== 13
|
|
->
|
|
get0(user_input,NC),
|
|
'$do_another'(NC)
|
|
;
|
|
C== -1
|
|
->
|
|
halt
|
|
;
|
|
skip(user_input,10), '$ask_again_for_another'
|
|
).
|
|
|
|
%'$add_nl_outside_console' :-
|
|
% '$is_same_tty'(user_input, user_error), !.
|
|
'$add_nl_outside_console' :-
|
|
format(user_error,'~n',[]).
|
|
|
|
'$ask_again_for_another' :-
|
|
format(user_error,'Action (\";\" for more choices, <return> for exit)', []),
|
|
'$another'.
|
|
|
|
'$write_answer'(_,_,_) :-
|
|
flush_output,
|
|
fail.
|
|
'$write_answer'(Vs, LBlk, FLAnsw) :-
|
|
'$purge_dontcares'(Vs,IVs),
|
|
'$sort'(IVs, NVs),
|
|
'$prep_answer_var_by_var'(NVs, LAnsw, LBlk),
|
|
'$name_vars_in_goals'(LAnsw, Vs, NLAnsw),
|
|
'$write_vars_and_goals'(NLAnsw, first, FLAnsw).
|
|
|
|
'$purge_dontcares'([],[]).
|
|
'$purge_dontcares'([Name=_|Vs],NVs) :-
|
|
atom_codes(Name, [C|_]), C is "_", !,
|
|
'$purge_dontcares'(Vs,NVs).
|
|
'$purge_dontcares'([V|Vs],[V|NVs]) :-
|
|
'$purge_dontcares'(Vs,NVs).
|
|
|
|
|
|
'$prep_answer_var_by_var'([], L, L).
|
|
'$prep_answer_var_by_var'([Name=Value|L], LF, L0) :-
|
|
'$delete_identical_answers'(L, Value, NL, Names),
|
|
'$prep_answer_var'([Name|Names], Value, LF, LI),
|
|
'$prep_answer_var_by_var'(NL, LI, L0).
|
|
|
|
% fetch all cases that have the same solution.
|
|
'$delete_identical_answers'([], _, [], []).
|
|
'$delete_identical_answers'([(Name=Value)|L], Value0, FL, [Name|Names]) :-
|
|
Value == Value0, !,
|
|
'$delete_identical_answers'(L, Value0, FL, Names).
|
|
'$delete_identical_answers'([VV|L], Value0, [VV|FL], Names) :-
|
|
'$delete_identical_answers'(L, Value0, FL, Names).
|
|
|
|
% now create a list of pairs that will look like goals.
|
|
'$prep_answer_var'(Names, Value, LF, L0) :- var(Value), !,
|
|
'$prep_answer_unbound_var'(Names, LF, L0).
|
|
'$prep_answer_var'(Names, Value, [nonvar(Names,Value)|L0], L0).
|
|
|
|
% ignore unbound variables
|
|
'$prep_answer_unbound_var'([_], L, L) :- !.
|
|
'$prep_answer_unbound_var'(Names, [var(Names)|L0], L0).
|
|
|
|
'$gen_name_string'(I,L,[C|L]) :- I < 26, !, C is I+65.
|
|
'$gen_name_string'(I,L0,LF) :-
|
|
I1 is I mod 26,
|
|
I2 is I // 26,
|
|
C is I1+65,
|
|
'$gen_name_string'(I2,[C|L0],LF).
|
|
|
|
'$write_vars_and_goals'([], _, []).
|
|
'$write_vars_and_goals'([nl,G1|LG], First, NG) :- !,
|
|
nl(user_error),
|
|
'$write_goal_output'(G1, First, NG, Next, IG),
|
|
'$write_vars_and_goals'(LG, Next, IG).
|
|
'$write_vars_and_goals'([G1|LG], First, NG) :-
|
|
'$write_goal_output'(G1, First, NG, Next, IG),
|
|
'$write_vars_and_goals'(LG, Next, IG).
|
|
|
|
'$goal_to_string'(Format, G, String) :-
|
|
format(codes(String),Format,G).
|
|
|
|
'$write_goal_output'(var([V|VL]), First, [var([V|VL])|L], next, L) :- !,
|
|
( First = first -> true ; format(user_error,',~n',[]) ),
|
|
format(user_error,'~a',[V]),
|
|
'$write_output_vars'(VL).
|
|
'$write_goal_output'(nonvar([V|VL],B), First, [nonvar([V|VL],B)|L], next, L) :- !,
|
|
( First = first -> true ; format(user_error,',~n',[]) ),
|
|
format(user_error,'~a',[V]),
|
|
'$write_output_vars'(VL),
|
|
format(user_error,' = ', []),
|
|
( yap_flag(toplevel_print_options, Opts) ->
|
|
write_term(user_error,B,[priority(699)|Opts]) ;
|
|
write_term(user_error,B,[priority(699)])
|
|
).
|
|
'$write_goal_output'(nl, First, NG, First, NG) :- !,
|
|
format(user_error,'~n',[]).
|
|
'$write_goal_output'(Format-G, First, NG, Next, IG) :- !,
|
|
G = [_|_], !,
|
|
% dump on string first so that we can check whether we actually
|
|
% had any output from the solver.
|
|
'$goal_to_string'(Format, G, String),
|
|
( String == [] ->
|
|
% we didn't
|
|
IG = NG, First = Next
|
|
;
|
|
% we did
|
|
( First = first -> true ; format(user_error,',~n',[]) ),
|
|
format(user_error, '~s', [String]),
|
|
NG = [G|IG]
|
|
).
|
|
'$write_goal_output'(_-G, First, [G|NG], next, NG) :- !,
|
|
( First = first -> true ; format(user_error,',~n',[]) ),
|
|
( yap_flag(toplevel_print_options, Opts) ->
|
|
write_term(user_error,G,Opts) ;
|
|
format(user_error,'~w',[G])
|
|
).
|
|
'$write_goal_output'(_M:G, First, [G|NG], next, NG) :- !,
|
|
( First = first -> true ; format(user_error,',~n',[]) ),
|
|
( yap_flag(toplevel_print_options, Opts) ->
|
|
write_term(user_error,G,Opts) ;
|
|
format(user_error,'~w',[G])
|
|
).
|
|
'$write_goal_output'(G, First, [M:G|NG], next, NG) :-
|
|
'$current_module'(M),
|
|
( First = first -> true ; format(user_error,',~n',[]) ),
|
|
( yap_flag(toplevel_print_options, Opts) ->
|
|
write_term(user_error,G,Opts) ;
|
|
format(user_error,'~w',[G])
|
|
).
|
|
|
|
'$name_vars_in_goals'(G, VL0, G) :-
|
|
'$name_well_known_vars'(VL0),
|
|
'$variables_in_term'(G, [], GVL),
|
|
'$name_vars_in_goals1'(GVL, 0, _).
|
|
|
|
'$name_well_known_vars'([]).
|
|
'$name_well_known_vars'([Name=V|NVL0]) :-
|
|
var(V), !,
|
|
V = '$VAR'(Name),
|
|
'$name_well_known_vars'(NVL0).
|
|
'$name_well_known_vars'([_|NVL0]) :-
|
|
'$name_well_known_vars'(NVL0).
|
|
|
|
'$name_vars_in_goals1'([], I, I).
|
|
'$name_vars_in_goals1'([V|NGVL], I0, IF) :-
|
|
I is I0+1,
|
|
'$gen_name_string'(I0,[],SName), !,
|
|
atom_codes(Name, [95|SName]),
|
|
V = '$VAR'(Name),
|
|
'$name_vars_in_goals1'(NGVL, I, IF).
|
|
'$name_vars_in_goals1'([NV|NGVL], I0, IF) :-
|
|
nonvar(NV),
|
|
'$name_vars_in_goals1'(NGVL, I0, IF).
|
|
|
|
'$write_output_vars'([]).
|
|
'$write_output_vars'([V|VL]) :-
|
|
format(user_error,' = ~a',[V]),
|
|
'$write_output_vars'(VL).
|
|
|
|
|
|
/** @pred + _P_ is nondet
|
|
|
|
The same as `call( _P_)`. This feature has been kept to provide
|
|
compatibility with C-Prolog. When compiling a goal, YAP
|
|
generates a `call( _X_)` whenever a variable _X_ is found as
|
|
a goal.
|
|
|
|
~~~~~{.prolog}
|
|
a(X) :- X.
|
|
~~~~~
|
|
is converted to:
|
|
|
|
~~~~~{.prolog}
|
|
a(X) :- call(X).
|
|
~~~~~
|
|
|
|
|
|
*/
|
|
|
|
/** @pred call(+ _P_) is iso
|
|
Meta-call predicate.
|
|
|
|
If _P_ is instantiated to an atom or a compound term, the goal `call(
|
|
_P_)` is executed as if the clause was originally written as _P_
|
|
instead as call( _P_ ), except that any "cut" occurring in _P_ only
|
|
cuts alternatives in the execution of _P_.
|
|
|
|
|
|
*/
|
|
call(G) :- '$execute'(G).
|
|
|
|
/** @pred incore(+ _P_)
|
|
|
|
|
|
The same as call/1.
|
|
|
|
|
|
*/
|
|
incore(G) :- '$execute'(G).
|
|
|
|
%
|
|
% standard meta-call, called if $execute could not do everything.
|
|
%
|
|
'$meta_call'(G, M) :-
|
|
'$current_choice_point'(CP),
|
|
'$call'(G, CP, G, M).
|
|
|
|
'$user_call'(G, M) :-
|
|
(
|
|
'$$save_by'(CP),
|
|
'$enable_debugging',
|
|
'$call'(G, CP, M:G, M),
|
|
'$$save_by'(CP2),
|
|
(
|
|
CP == CP2
|
|
->
|
|
!
|
|
;
|
|
( true ; '$enable_debugging', fail )
|
|
),
|
|
'$disable_debugging'
|
|
;
|
|
'$disable_debugging',
|
|
fail
|
|
).
|
|
|
|
|
|
|
|
% enable creeping
|
|
'$enable_debugging':-
|
|
current_prolog_flag(debug, false), !.
|
|
'$enable_debugging' :-
|
|
'$trace_on', !,
|
|
'$creep'.
|
|
'$enable_debugging'.
|
|
|
|
'$trace_on' :-
|
|
'$nb_getval'('$trace', on, fail).
|
|
|
|
'$trace_off' :-
|
|
'$nb_getval'('$trace', off, fail).
|
|
|
|
|
|
/** @pred :_P_ , :_Q_ is iso, meta
|
|
Conjunction of goals (and).
|
|
|
|
The conjunction is a fundamental construct of Prolog. Example:
|
|
|
|
~~~~~~~
|
|
p(X) :- q(X), r(X).
|
|
~~~~~~~
|
|
|
|
should be read as `p( _X_) if q( _X_) and r( _X_).
|
|
|
|
|
|
*/
|
|
','(X,Y) :-
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
'$call'(X,CP,(X,Y),M),
|
|
'$call'(Y,CP,(X,Y),M).
|
|
';'((X->A),Y) :- !,
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
( '$execute'(X)
|
|
->
|
|
'$call'(A,CP,(X->A;Y),M)
|
|
;
|
|
'$call'(Y,CP,(X->A;Y),M)
|
|
).
|
|
';'((X*->A),Y) :- !,
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
(
|
|
'$current_choice_point'(DCP),
|
|
'$execute'(X),
|
|
yap_hacks:cut_at(DCP),
|
|
'$call'(A,CP,((X*->A),Y),M)
|
|
;
|
|
'$call'(Y,CP,((X*->A),Y),M)
|
|
).
|
|
';'(X,Y) :-
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
( '$call'(X,CP,(X;Y),M) ; '$call'(Y,CP,(X;Y),M) ).
|
|
'|'(X,Y) :-
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
( '$call'(X,CP,(X|Y),M) ; '$call'(Y,CP,(X|Y),M) ).
|
|
'->'(X,Y) :-
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
( '$call'(X,CP,(X->Y),M) -> '$call'(Y,CP,(X->Y),M) ).
|
|
'*->'(X,Y) :-
|
|
yap_hacks:env_choice_point(CP),
|
|
'$current_module'(M),
|
|
( '$call'(X,CP,(X*->Y),M), '$call'(Y,CP,(X*->Y),M) ).
|
|
\+(G) :- \+ '$execute'(G).
|
|
not(G) :- \+ '$execute'(G).
|
|
|
|
'$cut_by'(CP) :- '$$cut_by'(CP).
|
|
|
|
%
|
|
% do it in ISO mode.
|
|
%
|
|
'$meta_call'(G,_ISO,M) :-
|
|
'$iso_check_goal'(G,G),
|
|
'$current_choice_point'(CP),
|
|
'$call'(G, CP, G, M).
|
|
|
|
'$meta_call'(G, CP, G0, M) :-
|
|
'$call'(G, CP, G0, M).
|
|
|
|
'$call'(G, CP, G0, _, M) :- /* iso version */
|
|
'$iso_check_goal'(G,G0),
|
|
'$call'(G, CP, G0, M).
|
|
|
|
|
|
'$call'(M:_,_,G0,_) :- var(M), !,
|
|
'$do_error'(instantiation_error,call(G0)).
|
|
'$call'(M:G,CP,G0,_) :- !,
|
|
'$call'(G,CP,G0,M).
|
|
'$call'((X,Y),CP,G0,M) :- !,
|
|
'$call'(X,CP,G0,M),
|
|
'$call'(Y,CP,G0,M).
|
|
'$call'((X->Y),CP,G0,M) :- !,
|
|
(
|
|
'$call'(X,CP,G0,M)
|
|
->
|
|
'$call'(Y,CP,G0,M)
|
|
).
|
|
'$call'((X*->Y),CP,G0,M) :- !,
|
|
'$call'(X,CP,G0,M),
|
|
'$call'(Y,CP,G0,M).
|
|
'$call'((X->Y; Z),CP,G0,M) :- !,
|
|
(
|
|
'$call'(X,CP,G0,M)
|
|
->
|
|
'$call'(Y,CP,G0,M)
|
|
;
|
|
'$call'(Z,CP,G0,M)
|
|
).
|
|
'$call'((X*->Y; Z),CP,G0,M) :- !,
|
|
(
|
|
'$current_choice_point'(DCP),
|
|
'$call'(X,CP,G0,M),
|
|
yap_hacks:cut_at(DCP),
|
|
'$call'(Y,CP,G0,M)
|
|
;
|
|
'$call'(Z,CP,G0,M)
|
|
).
|
|
'$call'((A;B),CP,G0,M) :- !,
|
|
(
|
|
'$call'(A,CP,G0,M)
|
|
;
|
|
'$call'(B,CP,G0,M)
|
|
).
|
|
'$call'((X->Y| Z),CP,G0,M) :- !,
|
|
(
|
|
'$call'(X,CP,G0,M)
|
|
->
|
|
'$call'(Y,CP,G0,M)
|
|
;
|
|
'$call'(Z,CP,G0,M)
|
|
).
|
|
'$call'((X*->Y| Z),CP,G0,M) :- !,
|
|
(
|
|
'$current_choice_point'(DCP),
|
|
'$call'(X,CP,G0,M),
|
|
yap_hacks:cut_at(DCP),
|
|
'$call'(Y,CP,G0,M)
|
|
;
|
|
'$call'(Z,CP,G0,M)
|
|
).
|
|
'$call'((A|B),CP, G0,M) :- !,
|
|
(
|
|
'$call'(A,CP,G0,M)
|
|
;
|
|
'$call'(B,CP,G0,M)
|
|
).
|
|
'$call'(\+ X, _CP, G0, M) :- !,
|
|
\+ ('$current_choice_point'(CP),
|
|
'$call'(X,CP,G0,M) ).
|
|
'$call'(not(X), _CP, G0, M) :- !,
|
|
\+ ('$current_choice_point'(CP),
|
|
'$call'(X,CP,G0,M) ).
|
|
'$call'(!, CP, _,_) :- !,
|
|
'$$cut_by'(CP).
|
|
'$call'([A|B], _, _, M) :- !,
|
|
'$csult'([A|B], M).
|
|
'$call'(G, _CP, _G0, CurMod) :-
|
|
(
|
|
'$is_metapredicate'(G,CurMod)
|
|
->
|
|
'$disable_debugging',
|
|
( '$expand_meta_call'(CurMod:G, [], NG) -> true ; true ),
|
|
'$enable_debugging'
|
|
;
|
|
NG = G
|
|
),
|
|
'$execute0'(NG, CurMod).
|
|
|
|
'$check_callable'(V,G) :- var(V), !,
|
|
'$do_error'(instantiation_error,G).
|
|
'$check_callable'(M:_G1,G) :- var(M), !,
|
|
'$do_error'(instantiation_error,G).
|
|
'$check_callable'(_:G1,G) :- !,
|
|
'$check_callable'(G1,G).
|
|
'$check_callable'(A,G) :- number(A), !,
|
|
'$do_error'(type_error(callable,A),G).
|
|
'$check_callable'(R,G) :- db_reference(R), !,
|
|
'$do_error'(type_error(callable,R),G).
|
|
'$check_callable'(_,_).
|
|
|
|
|
|
bootstrap(F) :-
|
|
% '$open'(F, '$csult', Stream, 0, 0, F),
|
|
% '$file_name'(Stream,File),
|
|
open(F, read, Stream),
|
|
stream_property(Stream, [file_name(File)]),
|
|
'$start_consult'(consult, File, LC),
|
|
file_directory_name(File, Dir),
|
|
working_directory(OldD, Dir),
|
|
(
|
|
current_prolog_flag(verbose_load, silent)
|
|
->
|
|
true
|
|
;
|
|
H0 is heapused, '$cputime'(T0,_),
|
|
format(user_error, '~*|% consulting ~w...~n', [LC,F])
|
|
),
|
|
'$loop'(Stream,consult),
|
|
working_directory(_, OldD),
|
|
'$current_module'(_, prolog),
|
|
'$end_consult',
|
|
(
|
|
current_prolog_flag(verbose_load, silent)
|
|
->
|
|
true
|
|
;
|
|
H is heapused-H0, '$cputime'(TF,_), T is TF-T0,
|
|
format(user_error, '~*|% ~w consulted ~w bytes in ~d msecs~n', [LC,F,H,T])
|
|
),
|
|
!,
|
|
close(Stream).
|
|
|
|
% '$undefp'([M0|G0], Default) :-
|
|
% writeln(M0:G0),
|
|
% fail.
|
|
|
|
|
|
'$loop'(Stream,exo) :-
|
|
prolog_flag(agc_margin,Old,0),
|
|
prompt1(': '), prompt(_,' '),
|
|
'$current_module'(OldModule),
|
|
repeat,
|
|
'$system_catch'(dbload_from_stream(Stream, OldModule, exo), '$db_load', Error,
|
|
user:'$LoopError'(Error, top)),
|
|
prolog_flag(agc_margin,_,Old),
|
|
!.
|
|
'$loop'(Stream,db) :-
|
|
prolog_flag(agc_margin,Old,0),
|
|
prompt1(': '), prompt(_,' '),
|
|
'$current_module'(OldModule),
|
|
repeat,
|
|
'$system_catch'(dbload_from_stream(Stream, OldModule, db), '$db_load', Error,
|
|
user:'$LoopError'(Error, top)),
|
|
prolog_flag(agc_margin,_,Old),
|
|
!.
|
|
'$loop'(Stream,Status) :-
|
|
% start_low_level_trace,
|
|
'$current_module'( OldModule ),
|
|
repeat,
|
|
'$system_catch'( '$enter_command'(Stream,OldModule,Status),
|
|
OldModule, Error,
|
|
user:'$LoopError'(Error, Status)
|
|
),
|
|
!.
|
|
|
|
'$enter_command'(Stream, Mod, Status) :-
|
|
prompt1(': '), prompt(_,' '),
|
|
Options = [module(Mod), syntax_errors(dec10),variable_names(Vars), term_position(Pos)],
|
|
(
|
|
Status == top
|
|
->
|
|
read_term(Stream, Command, Options)
|
|
;
|
|
read_clause(Stream, Command, Options)
|
|
),
|
|
'$command'(Command,Vars,Pos, Status).
|
|
|
|
/** @pred user:expand_term( _T_,- _X_) is dynamic,multifile.
|
|
|
|
This user-defined predicate is called by YAP after
|
|
reading goals and clauses.
|
|
|
|
- _Module_:`expand_term(` _T_ , _X_) is called first on the
|
|
current source module _Module_ ; if i
|
|
- `user:expand_term(` _T_ , _X_ `)` is available on every module.
|
|
|
|
*/
|
|
|
|
/* General purpose predicates */
|
|
|
|
'$head_and_body'((H:-B),H,B) :- !.
|
|
'$head_and_body'(H,H,true).
|
|
|
|
%
|
|
% split head and body, generate an error if body is unbound.
|
|
%
|
|
'$check_head_and_body'(C,M,H,B,P) :-
|
|
'$yap_strip_module'(C,M1,(MH:-B0)),
|
|
!,
|
|
'$yap_strip_module'(M1:MH,M,H),
|
|
( M == M1 -> B = B0 ; B = M1:B0),
|
|
is_callable(M:H,P).
|
|
|
|
'$check_head_and_body'(MH, M, H, true, P) :-
|
|
'$yap_strip_module'(MH,M,H),
|
|
error:is_callable(M:H,P).
|
|
% term expansion
|
|
%
|
|
% return two arguments: Expanded0 is the term after "USER" expansion.
|
|
% Expanded is the final expanded term.
|
|
%
|
|
'$precompile_term'(Term, ExpandedUser, Expanded) :-
|
|
%format('[ ~w~n',[Term]),
|
|
'$expand_clause'(Term, ExpandedUser, ExpandedI),
|
|
!,
|
|
%format(' -> ~w~n',[Expanded0]),
|
|
(
|
|
current_prolog_flag(strict_iso, true) /* strict_iso on */
|
|
->
|
|
Expanded = ExpandedI,
|
|
'$check_iso_strict_clause'(ExpandedUser)
|
|
;
|
|
'$expand_array_accesses_in_term'(ExpandedI,Expanded)
|
|
-> true
|
|
;
|
|
Expanded = ExpandedI
|
|
).
|
|
'$precompile_term'(Term, Term, Term).
|
|
|
|
'$expand_clause'(InputCl, C1, CO) :-
|
|
source_module(SM),
|
|
'$yap_strip_module'(SM:InputCl, M, ICl),
|
|
'$expand_a_clause'( M:ICl, SM, C1, CO),
|
|
!.
|
|
'$expand_clause'(Cl, Cl, Cl).
|
|
|
|
/** @pred expand_term( _T_,- _X_)
|
|
|
|
This predicate is used by YAP for preprocessing each top level
|
|
term read when consulting a file and before asserting or executing it.
|
|
It rewrites a term _T_ to a term _X_ according to the following
|
|
rules: first try term_expansion/2 in the current module, and then try to use the user defined predicate user:term_expansion/2`. If this call fails then the translating process
|
|
for DCG rules is applied, together with the arithmetic optimizer
|
|
whenever the compilation of arithmetic expressions is in progress.
|
|
|
|
|
|
*/
|
|
expand_term(Term,Expanded) :-
|
|
(
|
|
'$do_term_expansion'(Term,Expanded)
|
|
->
|
|
true
|
|
;
|
|
'$expand_term_grammar'(Term,Expanded)
|
|
).
|
|
|
|
%
|
|
% Grammar Rules expansion
|
|
%
|
|
'$expand_term_grammar'((A-->B), C) :-
|
|
prolog:'$translate_rule'((A-->B),C), !.
|
|
'$expand_term_grammar'(A, A).
|
|
|
|
%
|
|
% Arithmetic expansion
|
|
%
|
|
'$expand_array_accesses_in_term'(Expanded0,ExpandedF) :-
|
|
'$array_refs_compiled',
|
|
'$arrays':'$c_arrays'(Expanded0,ExpandedF), !.
|
|
'$expand_array_accesses_in_term'(Expanded,Expanded).
|
|
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
|
% catch/throw implementation
|
|
|
|
% at each catch point I need to know:
|
|
% what is ball;
|
|
% where was the previous catch
|
|
/** @pred catch( : _Goal_,+ _Exception_,+ _Action_) is iso
|
|
|
|
|
|
The goal `catch( _Goal_, _Exception_, _Action_)` tries to
|
|
execute goal _Goal_. If during its execution, _Goal_ throws an
|
|
exception _E'_ and this exception unifies with _Exception_, the
|
|
exception is considered to be caught and _Action_ is executed. If
|
|
the exception _E'_ does not unify with _Exception_, control
|
|
again throws the exception.
|
|
|
|
The top-level of YAP maintains a default exception handler that
|
|
is responsible to capture uncaught exceptions.
|
|
|
|
|
|
*/
|
|
catch(G, C, A) :-
|
|
'$catch'(C,A,_),
|
|
'$$save_by'(CP0),
|
|
'$execute'(G),
|
|
'$$save_by'(CP1),
|
|
(CP0 == CP1 -> !; true ).
|
|
|
|
% makes sure we have an environment.
|
|
'$true'.
|
|
|
|
|
|
% system_catch is like catch, but it avoids the overhead of a full
|
|
% meta-call by calling '$execute0' instead of $execute.
|
|
% This way it
|
|
% also avoids module preprocessing and goal_expansion
|
|
%
|
|
'$system_catch'(G, M, C, A) :-
|
|
% check current trail
|
|
'$catch'(C,A,_),
|
|
'$$save_by'(CP0),
|
|
'$execute_nonstop'(G, M),
|
|
'$$save_by'(CP1),
|
|
(CP0 == CP1 -> !; true ).
|
|
|
|
%
|
|
% throw has to be *exactly* after system catch!
|
|
%
|
|
/** @pred throw(+ _Ball_) is iso
|
|
|
|
|
|
The goal `throw( _Ball_)` throws an exception. Execution is
|
|
stopped, and the exception is sent to the ancestor goals until reaching
|
|
a matching catch/3, or until reaching top-level.
|
|
|
|
*/
|
|
throw(_Ball) :-
|
|
% use existing ball
|
|
'$get_exception'(Ball),
|
|
!,
|
|
'$jump_env_and_store_ball'(Ball).
|
|
throw(Ball) :-
|
|
( var(Ball) ->
|
|
'$do_error'(instantiation_error,throw(Ball))
|
|
;
|
|
% get current jump point
|
|
'$jump_env_and_store_ball'(Ball)
|
|
).
|
|
|
|
|
|
% just create a choice-point
|
|
'$catch'(_,_,_).
|
|
'$catch'(_,_,_) :- fail.
|
|
|
|
'$handle_throw'(_, _, _).
|
|
'$handle_throw'(C, A, _Ball) :-
|
|
'$reset_exception'(Ball),
|
|
% reset info
|
|
(catch_ball(Ball, C) ->
|
|
'$execute'(A)
|
|
;
|
|
throw(Ball)
|
|
).
|
|
|
|
catch_ball(Abort, _) :-
|
|
Abort == '$abort', !, fail.
|
|
% system defined throws should be ignored by user, unless the
|
|
% user is hacking away.
|
|
catch_ball(Ball, V) :-
|
|
var(V),
|
|
nonvar(Ball),
|
|
Ball = error(Type,_), % internal error ??
|
|
functor(Type, Name, _),
|
|
atom_codes(Name, [0'$|_]), %'0
|
|
!, fail.
|
|
catch_ball(C, C).
|
|
|
|
'$run_toplevel_hooks' :-
|
|
current_prolog_flag(break_level, 0 ),
|
|
recorded('$toplevel_hooks',H,_),
|
|
H \= fail, !,
|
|
( call(user:H) -> true ; true).
|
|
'$run_toplevel_hooks'.
|
|
|
|
'$run_at_thread_start' :-
|
|
recorded('$thread_initialization',M:D,_),
|
|
'$meta_call'(D, M),
|
|
fail.
|
|
'$run_at_thread_start'.
|
|
|
|
log_event( String, Args ) :-
|
|
format( atom( M ), String, Args),
|
|
log_event( M ).
|
|
|
|
'$early_print'( Lev, Msg ) :-
|
|
( '$undefined'(print_message(_,_),prolog) ->
|
|
'$show'(Lev, Msg)
|
|
;
|
|
print_message(Lev, Msg)
|
|
).
|
|
|
|
'$show'(_,Msg) :-
|
|
format(user_error, '~w~n', [Msg]).
|
|
|
|
|
|
'$prompt' :-
|
|
current_prolog_flag(break_level, BreakLevel),
|
|
(
|
|
BreakLevel == 0
|
|
->
|
|
LF = LD
|
|
;
|
|
LF = ['Break (level ', BreakLevel, ')'|LD]
|
|
),
|
|
current_prolog_flag(debug, DBON),
|
|
(
|
|
'$trace_on'
|
|
->
|
|
(
|
|
var(LF)
|
|
->
|
|
LD = ['trace'|LP]
|
|
;
|
|
LD = [', trace '|LP]
|
|
)
|
|
;
|
|
DBON == true
|
|
->
|
|
(var(LF)
|
|
->
|
|
LD = ['debug'|LP]
|
|
;
|
|
LD = [', debug'|LP]
|
|
)
|
|
;
|
|
LD = LP
|
|
),
|
|
(
|
|
var(LF)
|
|
->
|
|
LP = [P]
|
|
;
|
|
LP = [' ',P]
|
|
),
|
|
yap_flag(toplevel_prompt, P),
|
|
atomic_concat(LF, PF),
|
|
prompt1(PF),
|
|
prompt(_,' '),
|
|
'$ensure_prompting'.
|
|
|
|
/**
|
|
@}
|
|
*/
|