2001-04-09 20:54:03 +01:00
/*************************************************************************
* *
* YAP Prolog *
* *
* Yap Prolog was developed at NCCUP - Universidade do Porto *
* *
* Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 *
* *
**************************************************************************
* *
* File: modules.pl *
* Last rev: *
* mods: *
* comments: module support *
* *
*************************************************************************/
2014-04-09 12:39:29 +01:00
2014-07-17 18:19:38 +01:00
2015-12-15 09:01:44 +00:00
/**
@file modules.yap
2014-07-17 18:19:38 +01:00
**/
2014-04-09 12:39:29 +01:00
:- system_module( '$_modules', [abolish_module/1,
add_import_module/3,
current_module/1,
current_module/2,
delete_import_module/2,
expand_goal/2,
export/1,
export_list/2,
export_resource/1,
import_module/2,
ls_imports/0,
module/1,
module_property/2,
set_base_module/1,
source_module/1,
use_module/1,
2014-07-27 01:43:16 +01:00
use_module/2,
use_module/3], ['$add_to_imports'/3,
2014-04-09 12:39:29 +01:00
'$clean_cuts'/2,
'$convert_for_export'/7,
'$do_import'/3,
'$extend_exports'/3,
'$get_undefined_pred'/4,
2015-07-06 12:04:42 +01:00
'$imported_predicate'/4,
2015-12-15 09:01:44 +00:00
'$meta_expand'/6,
2014-04-09 12:39:29 +01:00
'$meta_predicate'/2,
'$meta_predicate'/4,
'$module'/3,
'$module'/4,
2014-10-05 23:53:05 +01:00
'$module_expansion'/6,
2014-04-09 12:39:29 +01:00
'$module_transparent'/2,
'$module_transparent'/4]).
2015-12-15 09:01:44 +00:00
2014-04-09 12:39:29 +01:00
:- use_system_module( '$_arith', ['$c_built_in'/3]).
:- use_system_module( '$_consult', ['$lf_opt'/3,
'$load_files'/3]).
:- use_system_module( '$_debug', ['$skipeol'/1]).
:- use_system_module( '$_errors', ['$do_error'/2]).
:- use_system_module( '$_eval', ['$full_clause_optimisation'/4]).
2001-04-09 20:54:03 +01:00
2015-12-15 09:01:44 +00:00
:- multifile '$system_module'/1.
2014-04-06 17:05:17 +01:00
:- '$purge_clauses'(module(_,_), prolog).
:- '$purge_clauses'('$module'(_,_), prolog).
:- '$purge_clauses'(use_module(_), prolog).
:- '$purge_clauses'(use_module(_,_), prolog).
%
% start using default definition of module.
%
2014-07-17 18:19:38 +01:00
/**
\pred use_module( +Files ) is directive
loads a module file
This predicate loads the file specified by _Files_, importing all
their public predicates into the current type-in module. It is
implemented as if by:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
2014-04-06 17:05:17 +01:00
use_module(F) :-
2014-07-17 18:19:38 +01:00
load_files(F, [if(not_loaded),must_be_module(true)]).
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Notice that _Files_ may be a single file, or a list with a number
files. The _Files_ are loaded in YAP only once, even if they have been
updated meanwhile. YAP should also verify whether the files actually
define modules. Please consult load_files/3 for other options when
loading a file.
Predicate name clashes between two different modules may arise, either
when trying to import predicates that are also defined in the current
type-in module, or by trying to import the same predicate from two
different modules.
In the first case, the local predicate is considered to have priority
and use_module/1 simply gives a warning. As an example, if the file
`a.pl` contains:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
:- module( a, [a/1] ).
:- use_module(b).
a(1).
2014-07-27 01:14:15 +01:00
a(X) :- b(X).
2014-07-17 18:19:38 +01:00
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
and the file `b.pl` contains:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
:- module( b, [a/1,b/1] ).
a(2).
b(1).
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
YAP will execute as follows:
2014-07-27 01:14:15 +01:00
2014-07-17 18:19:38 +01:00
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
2014-07-27 01:14:15 +01:00
?- [a].
% consulting .../a.pl...
% consulting .../b.pl...
% consulted .../b.pl in module b, 0 msec 0 bytes
% consulted .../a.pl in module a, 1 msec 0 bytes
2014-07-17 18:19:38 +01:00
true.
?- a(X).
2014-07-27 01:14:15 +01:00
X = 1 ? ;
2014-07-17 18:19:38 +01:00
X = 1.
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
The example shows that the query `a(X)`has a single answer, the one
defined in `a.pl`. Calls to `a(X)`succeed in the top-level, because
the module `a` was loaded into `user`. On the other hand, `b(X)`is not
exported by `a.pl`, and is not available to calls, although it can be
accessed as a predicate in the module 'a' by using the `:` operator.
Next, consider the three files `c.pl`, `d1.pl`, and `d2.pl`:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
% c.pl
:- module( c, [a/1] ).
:- use_module([d1, d2]).
a(X) :-
b(X).
a(X) :-
c(X).
a(X) :-
d(X).
% d1.pl
:- module( d1, [b/1,c/1] ).
2015-12-15 09:01:44 +00:00
vvb(2).
2014-07-17 18:19:38 +01:00
c(3).
% d2.pl
:- module( d2, [b/1,d/1] ).
b(1).
d(4).
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
The result is as follows:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
./yap -l c
YAP 6.3.4 (x86_64-darwin13.3.0): Tue Jul 15 10:42:11 CDT 2014
2015-02-14 11:39:07 +00:00
2014-07-17 18:19:38 +01:00
ERROR!!
2014-07-27 01:14:15 +01:00
at line 3 in o/d2.pl,
PERMISSION ERROR- loading .../c.pl: modules d1 and d2 both define b/1
2014-07-17 18:19:38 +01:00
?- a(X).
X = 2 ? ;
ERROR!!
EXISTENCE ERROR- procedure c/1 is undefined, called from context prolog:$user_call/2
Goal was c:c(_131290)
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2014-07-27 01:14:15 +01:00
The state of the module system after this error is undefined.
2014-07-17 18:19:38 +01:00
**/
use_module(F) :- '$load_files'(F,
2014-07-27 01:14:15 +01:00
[if(not_loaded),must_be_module(true)], use_module(F)).
2014-07-17 18:19:38 +01:00
2015-02-14 11:39:07 +00:00
2014-07-17 18:19:38 +01:00
/**
\pred use_module(+Files, +Imports)
loads a module file but only imports the named predicates
2014-07-27 01:43:16 +01:00
2014-07-17 18:19:38 +01:00
This predicate loads the file specified by _Files_, importing their
public predicates specified by _Imports_ into the current type-in
module. It is implemented as if by:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
use_module(Files, Imports) :-
load_files(Files, [if(not_loaded),must_be_module(true),imports(Imports)]).
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
The _Imports_ argument may be use to specify which predicates one
wants to load. It can also be used to give the predicates a different name. As an example,
the graphs library is implemented on top of the red-black trees library, and some predicates are just aliases:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~pl
:- use_module(library(rbtrees), [
rb_min/3 as min_assoc,
rb_max/3 as max_assoc,
2014-07-27 01:14:15 +01:00
2014-07-17 18:19:38 +01:00
...]).
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Unfortunately it is still not possible to change argument order.
**/
2014-04-06 17:05:17 +01:00
use_module(F,Is) :-
'$load_files'(F, [if(not_loaded),must_be_module(true),imports(Is)], use_module(F,Is)).
2014-07-17 18:19:38 +01:00
2001-04-19 18:12:18 +01:00
'$module'(O,N,P,Opts) :- !,
'$module'(O,N,P),
'$process_module_decls_options'(Opts,module(Opts,N,P)).
2015-02-14 11:39:07 +00:00
2001-04-19 18:12:18 +01:00
'$process_module_decls_options'(Var,Mod) :-
2005-07-06 16:51:01 +01:00
var(Var), !,
2002-09-09 18:40:12 +01:00
'$do_error'(instantiation_error,Mod).
2005-07-06 16:51:01 +01:00
'$process_module_decls_options'([],_) :- !.
'$process_module_decls_options'([H|L],M) :- !,
2001-04-19 18:12:18 +01:00
'$process_module_decls_option'(H,M),
'$process_module_decls_options'(L,M).
'$process_module_decls_options'(T,M) :-
2002-09-09 18:40:12 +01:00
'$do_error'(type_error(list,T),M).
2001-04-19 18:12:18 +01:00
2015-02-14 11:39:07 +00:00
'$process_module_decls_option'(Var,M) :-
var(Var),
2002-09-09 18:40:12 +01:00
'$do_error'(instantiation_error,M).
2015-02-14 11:39:07 +00:00
'$process_module_decls_option'(At,M) :-
2005-07-06 16:51:01 +01:00
atom(At), !,
2014-08-20 15:09:20 +01:00
use_module(M:At).
'$process_module_decls_option'(library(L),M) :- !,
use_module(M:library(L)).
2005-07-06 16:51:01 +01:00
'$process_module_decls_option'(hidden(Bool),M) :- !,
2001-04-19 18:12:18 +01:00
'$process_hidden_module'(Bool, M).
2015-02-14 11:39:07 +00:00
'$process_module_decls_option'(Opt,M) :-
2002-09-09 18:40:12 +01:00
'$do_error'(domain_error(module_decl_options,Opt),M).
2001-04-19 18:12:18 +01:00
'$process_hidden_module'(TNew,M) :-
'$convert_true_off_mod3'(TNew, New, M),
source_mode(Old, New),
'$prepare_restore_hidden'(Old,New).
2013-11-08 08:24:06 +00:00
'$convert_true_off_mod3'(true, off, _) :- !.
'$convert_true_off_mod3'(false, on, _) :- !.
2001-04-19 18:12:18 +01:00
'$convert_true_off_mod3'(X, _, M) :-
2002-09-09 18:40:12 +01:00
'$do_error'(domain_error(module_decl_options,hidden(X)),M).
2001-04-19 18:12:18 +01:00
'$prepare_restore_hidden'(Old,Old) :- !.
'$prepare_restore_hidden'(Old,New) :-
2015-11-05 17:21:48 +00:00
recorda('$system_initialization', source_mode(New,Old), _).
2001-04-19 18:12:18 +01:00
2014-07-17 18:19:38 +01:00
'$extend_exports'(HostF, Exports, DonorF ) :-
2014-09-25 08:50:26 +01:00
( recorded('$module','$module'( DonorF, DonorM, _,DonorExports, _),_) -> true ; DonorF = user_input ),
( recorded('$module','$module'( HostF, HostM, SourceF, _, _),_) -> true ; HostF = user_input ),
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(HostF, HostM, _, AllExports, _Line), R), erase(R),
2014-10-07 01:35:41 +01:00
'$convert_for_export'(Exports, DonorExports, DonorM, HostM, _TranslationTab, AllReExports, reexport(DonorF, Exports)),
2014-07-17 18:19:38 +01:00
lists:append( AllReExports, AllExports, Everything0 ),
sort( Everything0, Everything ),
( source_location(_, Line) -> true ; Line = 0 ),
2014-09-22 18:13:35 +01:00
recorda('$module','$module'(HostF,HostM,SourceF, Everything, Line),_).
2004-02-13 13:31:44 +00:00
2013-11-08 00:02:38 +00:00
'$module_produced by'(M, M0, N, K) :-
2007-12-05 12:17:25 +00:00
recorded('$import','$import'(M,M0,_,_,N,K),_), !.
2013-11-08 00:02:38 +00:00
'$module_produced by'(M, M0, N, K) :-
2007-12-05 12:17:25 +00:00
recorded('$import','$import'(MI,M0,G1,_,N,K),_),
functor(G1, N1, K1),
2015-02-14 11:39:07 +00:00
'$module_produced by'(M,MI,N1,K1).
2005-10-28 18:38:50 +01:00
2001-04-09 20:54:03 +01:00
2015-12-15 09:01:44 +00:00
/** @pred current_module( ? Mod:atom) is nondet
2014-09-11 20:06:57 +01:00
2015-12-15 09:01:44 +00:00
Succeeds if _M_ is a user-visible modules. A module is defined as soon as some
2014-09-11 20:06:57 +01:00
predicate defined in the module is loaded, as soon as a goal in the
module is called, or as soon as it becomes the current type-in module.
2015-02-14 11:39:07 +00:00
2014-07-27 01:14:15 +01:00
*/
current_module(Mod) :-
'$all_current_modules'(Mod),
2015-12-15 09:01:44 +00:00
\+ prolog:'$system_module'(Mod).
2014-09-11 20:06:57 +01:00
2015-12-15 09:01:44 +00:00
/** @pred current_module( ? Mod:atom, ? _F_ : file ) is nondet
2014-09-11 20:06:57 +01:00
2015-12-15 09:01:44 +00:00
Succeeds if _M_ is a module associated with the file _F_, that is, if _File_ is the source for _M_. If _M_ is not declared in a file, _F_ unifies with `user`.
2014-09-11 20:06:57 +01:00
*/
2014-07-27 01:14:15 +01:00
current_module(Mod,TFN) :-
'$all_current_modules'(Mod),
2014-09-22 18:13:35 +01:00
( recorded('$module','$module'(TFN,Mod,_,_Publics, _),_) -> true ; TFN = user ).
2014-07-27 01:14:15 +01:00
2001-05-28 20:54:53 +01:00
'$trace_module'(X) :-
2001-04-09 20:54:03 +01:00
telling(F),
tell('P0:debug'),
write(X),nl,
tell(F), fail.
2001-10-30 16:42:05 +00:00
'$trace_module'(_).
2001-04-09 20:54:03 +01:00
2001-05-28 20:54:53 +01:00
'$trace_module'(X,Y) :- X==Y, !.
'$trace_module'(X,Y) :-
2001-04-09 20:54:03 +01:00
telling(F),
tell('~/.dbg.modules'),
write('***************'), nl,
portray_clause(X),
portray_clause(Y),
tell(F),fail.
2001-10-30 16:42:05 +00:00
'$trace_module'(_,_).
2001-04-09 20:54:03 +01:00
2015-02-14 11:39:07 +00:00
2015-12-15 09:01:44 +00:00
'$continue_imported'(Mod,Mod,Pred,Pred) :-
'$pred_exists'(Pred, Mod),
2014-10-11 12:45:54 +01:00
!.
2015-12-15 09:01:44 +00:00
'$continue_imported'(FM,Mod,FPred,Pred) :-
recorded('$import','$import'(IM,Mod,IPred,Pred,_,_),_), !,
'$continue_imported'(FM, IM, FPred, IPred).
'$continue_imported'(FM,Mod,FPred,Pred) :-
prolog:'$parent_module'(Mod,IM),
'$continue_imported'(FM, IM, FPred, Pred).
2001-04-09 20:54:03 +01:00
2008-03-13 14:38:02 +00:00
% be careful here not to generate an undefined exception.
2015-12-15 09:01:44 +00:00
'$imported_predicate'(G, ImportingMod, G, prolog) :-
'$system_predicate'(G, prolog), !.
2015-07-06 12:04:42 +01:00
'$imported_predicate'(G, ImportingMod, G0, ExportingMod) :-
2015-04-13 13:28:17 +01:00
( var(G) -> true ;
2015-09-21 23:05:36 +01:00
var(ImportingMod) -> true ;
'$undefined'(G, ImportingMod)
),
2009-12-04 11:00:13 +00:00
'$get_undefined_pred'(G, ImportingMod, G0, ExportingMod),
2015-12-15 09:01:44 +00:00
ExportingMod \= ImportingMod,
!.
2001-04-09 20:54:03 +01:00
2009-12-04 11:00:13 +00:00
'$get_undefined_pred'(G, ImportingMod, G0, ExportingMod) :-
recorded('$import','$import'(ExportingModI,ImportingMod,G0I,G,_,_),_),
2015-12-15 09:01:44 +00:00
'$continue_imported'(ExportingMod, ExportingModI, G0, G0I),
!.
2010-04-22 12:16:37 +01:00
% SWI builtin
2015-02-14 11:39:07 +00:00
'$get_undefined_pred'(G, _ImportingMod, G, user) :-
2015-02-14 17:05:48 +00:00
nonvar(G),
2015-02-14 11:39:07 +00:00
'$pred_exists'(G, user), !.
2015-03-28 19:42:10 +00:00
'$get_undefined_pred'(G, ImportingMod, G0, ExportingMod) :-
2015-07-06 12:04:42 +01:00
recorded('$dialect',swi,_),
get_prolog_flag(autoload, true),
2015-12-15 09:01:44 +00:00
prolog_flag(unknown, OldUnk, fail),
(
'$autoload'(G, ImportingMod, ExportingModI, swi)
->
prolog_flag(unknown, _, OldUnk)
;
prolog_flag(unknown, _, OldUnk),
fail
),
2015-07-06 12:04:42 +01:00
'$continue_imported'(ExportingMod, ExportingModI, G0, G).
% autoload
2014-11-25 12:03:48 +00:00
% parent module mechanism
2009-12-04 11:00:13 +00:00
'$get_undefined_pred'(G, ImportingMod, G0, ExportingMod) :-
2015-12-15 09:01:44 +00:00
'$parent_module'(ImportingMod,ExportingModI),
2009-12-04 11:00:13 +00:00
'$continue_imported'(ExportingMod, ExportingModI, G0, G).
2015-07-06 12:04:42 +01:00
'$autoload'(G, _ImportingMod, ExportingMod, Dialect) :-
functor(G, Name, Arity),
2015-12-15 09:01:44 +00:00
call(Dialect:index(Name,Arity,ExportingMod,_)),
!.
2015-07-06 12:04:42 +01:00
'$autoload'(G, ImportingMod, ExportingMod, _Dialect) :-
functor(G, N, K),
functor(G0, N, K),
'$autoloader_find_predicate'(G0,ExportingMod),
ExportingMod \= ImportingMod,
(recordzifnot('$import','$import'(ExportingMod,ImportingMod,G0,G0, N ,K),_) -> true ; true ).
2013-10-30 09:41:45 +00:00
'$autoloader_find_predicate'(G,ExportingModI) :-
2015-12-15 09:01:44 +00:00
'$nb_getval'('$autoloader_set', true, false), !,
2013-10-30 09:41:45 +00:00
autoloader:find_predicate(G,ExportingModI).
'$autoloader_find_predicate'(G,ExportingModI) :-
2015-12-15 09:01:44 +00:00
yap_flag(autoload, true, false),
yap_flag( unknown, Unknown, fast_fail),
yap_flag(debug, Debug, false), !,
2013-10-30 09:41:45 +00:00
load_files([library(autoloader),
autoloader:library('INDEX'),
swi:library('dialect/swi/INDEX')],
2013-11-10 11:18:00 +00:00
[autoload(true),if(not_loaded)]),
2013-10-30 09:41:45 +00:00
nb_setval('$autoloader_set', true),
2015-12-15 09:01:44 +00:00
yap_flag(autoload, _, true),
yap_flag( unknown, _, Unknown),
yap_flag( debug, _, Debug),
2013-10-30 09:41:45 +00:00
autoloader:find_predicate(G,ExportingModI).
2014-07-27 01:14:15 +01:00
/**
be associated to a new file.
\param[in] _Module_ is the name of the module to declare
\param[in] _MSuper_ is the name of the context module. Use `prolog`or `system`
if you do not need a context.
\param[in] _File_ is the canonical name of the file from which the module is loaded
\param[in] Line is the line-number of the :- module/2 directive.
\param[in] If _Redefine_ `true`, allow associating the module to a new file
*/
'$declare_module'(Name, _Super, Context, _File, _Line) :-
add_import_module(Name, Context, start).
/**
\pred abolish_module( + Mod) is det
get rid of a module and of all predicates included in the module.
*/
2006-04-05 01:16:55 +01:00
abolish_module(Mod) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(_,Mod,_,_,_),R), erase(R),
2006-04-05 01:16:55 +01:00
fail.
abolish_module(Mod) :-
2013-11-26 09:40:00 +00:00
recorded('$import','$import'(Mod,_,_,_,_,_),R), erase(R),
2006-04-05 01:16:55 +01:00
fail.
2008-06-02 18:20:28 +01:00
abolish_module(Mod) :-
2014-11-25 12:03:48 +00:00
'$current_predicate'(Na,Mod,S,_),
functor(S, Na, Ar),
2008-06-02 18:20:28 +01:00
abolish(Mod:Na/Ar),
fail.
2006-04-05 01:16:55 +01:00
abolish_module(_).
2007-12-05 12:17:25 +00:00
2012-10-22 10:18:26 +01:00
export(Resource) :-
var(Resource),
2015-02-14 11:39:07 +00:00
'$do_error'(instantiation_error,export(Resource)).
2012-10-22 10:18:26 +01:00
export([]) :- !.
export([Resource| Resources]) :- !,
export_resource(Resource),
export(Resources).
export(Resource) :-
export_resource(Resource).
export_resource(Resource) :-
2014-07-17 18:19:38 +01:00
var(Resource), !,
2015-02-14 11:39:07 +00:00
'$do_error'(instantiation_error,export(Resource)).
2012-10-22 10:18:26 +01:00
export_resource(P) :-
P = F/N, atom(F), number(N), N >= 0, !,
2015-02-14 11:39:07 +00:00
'$current_module'(Mod),
2014-09-22 18:13:35 +01:00
( recorded('$module','$module'(File,Mod,SourceF,ExportedPreds,Line),R) ->
2015-02-14 11:39:07 +00:00
erase(R),
2014-09-25 08:50:26 +01:00
recorda('$module','$module'(File,Mod,SourceF,[P|ExportedPreds],Line),_)
2012-10-22 15:04:52 +01:00
; prolog_load_context(file, File) ->
2014-09-22 18:13:35 +01:00
recorda('$module','$module'(File,Mod,SourceF,[P],Line),_)
; recorda('$module','$module'(user_input,Mod,user_input,[P],1),_)
2012-02-14 09:54:56 +00:00
).
2012-10-22 10:18:26 +01:00
export_resource(P0) :-
P0 = F//N, atom(F), number(N), N >= 0, !,
2012-02-14 09:54:56 +00:00
N1 is N+2, P = F/N1,
2015-02-14 11:39:07 +00:00
'$current_module'(Mod),
2014-09-22 18:13:35 +01:00
( recorded('$module','$module'(File,Mod,SourceF,ExportedPreds,Line),R) ->
2015-02-14 11:39:07 +00:00
erase(R),
2014-09-22 18:13:35 +01:00
recorda('$module','$module'(File,Mod,SourceF,[P|ExportedPreds],Line ),_)
2012-10-22 15:04:52 +01:00
; prolog_load_context(file, File) ->
2014-09-22 18:13:35 +01:00
recorda('$module','$module'(File,Mod,SourceF,[P],Line),_)
; recorda('$module','$module'(user_input,Mod,user_input,[P],1),_)
2012-02-14 09:54:56 +00:00
).
2012-10-22 10:18:26 +01:00
export_resource(op(Prio,Assoc,Name)) :- !,
2015-02-14 11:39:07 +00:00
op(Prio,Assoc,prolog:Name).
export_resource(op(Prio,Assoc,Name)) :- !,
op(Prio,Assoc,user:Name).
2012-10-22 10:18:26 +01:00
export_resource(Resource) :-
'$do_error'(type_error(predicate_indicator,Resource),export(Resource)).
2015-02-14 11:39:07 +00:00
2012-02-14 09:54:56 +00:00
export_list(Module, List) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(_,Module,_,List,_),_).
2012-02-14 09:54:56 +00:00
2014-07-17 18:19:38 +01:00
2007-12-05 12:17:25 +00:00
'$add_to_imports'([], _, _).
2013-11-08 00:02:38 +00:00
% no need to import from the actual module
2015-02-14 11:39:07 +00:00
'$add_to_imports'([T|Tab], Module, ContextModule) :-
2013-11-08 00:02:38 +00:00
'$do_import'(T, Module, ContextModule),
'$add_to_imports'(Tab, Module, ContextModule).
'$do_import'(op(Prio,Assoc,Name), _Mod, ContextMod) :-
op(Prio,Assoc,ContextMod:Name).
'$do_import'(N0/K0-N0/K0, Mod, Mod) :- !.
2015-07-06 12:04:42 +01:00
'$do_import'(N0/K0-N0/K0, _Mod, prolog) :- !.
2013-11-08 00:02:38 +00:00
'$do_import'(_N/K-N1/K, _Mod, ContextMod) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(_F, ContextMod, _SourceF, MyExports,_),_),
2013-11-08 00:02:38 +00:00
once(lists:member(N1/K, MyExports)),
functor(S, N1, K),
% reexport predicates if they are undefined in the current module.
\+ '$undefined'(S,ContextMod), !.
'$do_import'( N/K-N1/K, Mod, ContextMod) :-
functor(G,N,K),
'$follow_import_chain'(Mod,G,M0,G0),
2014-10-09 10:47:41 +01:00
G0=..[_N0|Args],
2007-12-05 12:17:25 +00:00
G1=..[N1|Args],
2013-11-08 00:02:38 +00:00
( '$check_import'(M0,ContextMod,N1,K) ->
2015-04-13 13:28:17 +01:00
( ContextMod == prolog ->
recordzifnot('$import','$import'(M0,user,G0,G1,N1,K),_),
fail
2013-11-08 00:02:38 +00:00
;
2015-04-13 13:28:17 +01:00
recordaifnot('$import','$import'(M0,ContextMod,G0,G1,N1,K),_),
fail
;
true
2013-11-08 00:02:38 +00:00
)
;
true
).
'$follow_import_chain'(M,G,M0,G0) :-
recorded('$import','$import'(M1,M,G1,G,_,_),_), M \= M1, !,
'$follow_import_chain'(M1,G1,M0,G0).
'$follow_import_chain'(M,G,M,G).
% trying to import Mod:N/K into ContextM
'$check_import'(Mod, ContextM, N, K) :-
recorded('$import','$import'(MI, ContextM, _, _, N,K),_R),
2015-02-14 11:39:07 +00:00
% dereference MI to M1, in order to find who
2013-11-08 00:02:38 +00:00
% is actually generating
( '$module_produced by'(M1, MI, N, K) -> true ; MI = M1 ),
2014-07-17 18:19:38 +01:00
( '$module_produced by'(M2, Mod, N, K) -> true ; Mod = M2 ),
2013-11-08 00:02:38 +00:00
M2 \= M1, !,
2015-12-15 09:01:44 +00:00
'$redefine_import'( M1, M2, Mod, ContextM, N/K).
2013-11-08 00:02:38 +00:00
'$check_import'(_,_,_,_).
2015-12-15 09:01:44 +00:00
'$redefine_import'( M1, M2, Mod, ContextM, N/K) :-
'$nb_getval'('$lf_status', TOpts, fail),
'$lf_opt'(redefine_module, TOpts, Action), !,
'$redefine_action'(Action, M1, M2, Mod, ContextM, N/K).
'$redefine_import'( M1, M2, Mod, ContextM, N/K) :-
'$redefine_action'(false, M1, M2, Mod, ContextM, N/K).
2014-07-17 18:19:38 +01:00
'$redefine_action'(ask, M1, M2, M, _, N/K) :-
2013-11-08 00:02:38 +00:00
stream_property(user_input,tty(true)), !,
format(user_error,'NAME CLASH: ~w was already imported to module ~w;~n',[M1:N/K,M2]),
format(user_error,' Do you want to import it from ~w ? [y, n, e or h] ',M),
'$mod_scan'(C),
2014-08-20 14:08:21 +01:00
( C == e -> halt(1) ;
2015-02-14 11:39:07 +00:00
C == y ).
2014-07-17 18:19:38 +01:00
'$redefine_action'(true, M1, _, _, _, _) :- !,
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(F, M1, _, _MyExports,_Line),_),
2013-11-08 00:02:38 +00:00
unload_file(F).
2014-10-07 01:35:41 +01:00
'$redefine_action'(false, M1, M2, _M, ContextM, N/K) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(F, ContextM, _, _MyExports,_Line),_),
2014-07-17 18:19:38 +01:00
'$current_module'(_, M2),
'$do_error'(permission_error(import,M1:N/K,redefined,M2),F).
2008-02-12 17:03:59 +00:00
2014-08-20 14:08:21 +01:00
'$mod_scan'(C) :-
get_char(C),
'$skipeol'(C),
(C == y -> true; C == n).
2014-07-27 01:14:15 +01:00
/**
2014-12-24 15:32:29 +00:00
@pred set_base_module( +ExportingModule ) is det
2015-02-14 11:39:07 +00:00
All exported predicates from _ExportingModule_ are automatically available to the
2014-07-27 01:14:15 +01:00
current source module.
This built-in was introduced by SWI-Prolog. In YAP, by default, modules only
inherit from `prolog`. This extension allows predicates in the current
module (see module/2 and module/1) to inherit from `user` or other modules.
*/
2009-12-04 11:00:13 +00:00
set_base_module(ExportingModule) :-
var(ExportingModule),
'$do_error'(instantiation_error,set_base_module(ExportingModule)).
set_base_module(ExportingModule) :-
atom(ExportingModule), !,
'$current_module'(Mod),
2013-11-08 00:02:38 +00:00
retractall(prolg:'$parent_module'(Mod,_)),
2009-12-04 11:00:13 +00:00
asserta(prolog:'$parent_module'(Mod,ExportingModule)).
set_base_module(ExportingModule) :-
'$do_error'(type_error(atom,ExportingModule),set_base_module(ExportingModule)).
2014-07-27 01:14:15 +01:00
/**
2014-12-24 15:32:29 +00:00
@pred import_module( +ImportingModule, +ExportingModule ) is det
2015-02-14 11:39:07 +00:00
All exported predicates from _ExportingModule_ are automatically available to the
2014-07-27 01:14:15 +01:00
source module _ImportModule_.
This innovation was introduced by SWI-Prolog. By default, modules only
2015-02-14 11:39:07 +00:00
inherit from `prolog` and `user`. This extension allows predicates in
any module to inherit from `user` and other modules.
2014-07-27 01:14:15 +01:00
*/
2009-12-04 11:00:13 +00:00
import_module(Mod, ImportModule) :-
var(Mod),
'$do_error'(instantiation_error,import_module(Mod, ImportModule)).
import_module(Mod, ImportModule) :-
atom(Mod), !,
prolog:'$parent_module'(Mod,ImportModule).
import_module(Mod, EM) :-
'$do_error'(type_error(atom,Mod),import_module(Mod, EM)).
2014-11-02 12:10:32 +00:00
2014-12-24 15:32:29 +00:00
/**
@pred add_import_module( + _Module_, + _ImportModule_ , +_Pos_) is det
2014-11-02 12:10:32 +00:00
Add all exports in _ImportModule_ as available to _Module_.
2015-02-14 11:39:07 +00:00
All exported predicates from _ExportModule_ are made available to the
source module _ImportModule_. If _Position_ is bound to `start` the
module _ImportModule_ is tried first, if _Position_ is bound to `end`,
2014-11-02 12:10:32 +00:00
the module is consulted last.
*/
2009-12-04 11:00:13 +00:00
add_import_module(Mod, ImportModule, Pos) :-
var(Mod),
'$do_error'(instantiation_error,add_import_module(Mod, ImportModule, Pos)).
add_import_module(Mod, ImportModule, Pos) :-
var(Pos),
'$do_error'(instantiation_error,add_import_module(Mod, ImportModule, Pos)).
add_import_module(Mod, ImportModule, start) :-
atom(Mod), !,
retractall(prolog:'$parent_module'(Mod,ImportModule)),
asserta(prolog:'$parent_module'(Mod,ImportModule)).
add_import_module(Mod, ImportModule, end) :-
atom(Mod), !,
retractall(prolog:'$parent_module'(Mod,ImportModule)),
assertz(prolog:'$parent_module'(Mod,ImportModule)).
add_import_module(Mod, ImportModule, Pos) :-
\+ atom(Mod), !,
'$do_error'(type_error(atom,Mod),add_import_module(Mod, ImportModule, Pos)).
add_import_module(Mod, ImportModule, Pos) :-
'$do_error'(domain_error(start_end,Pos),add_import_module(Mod, ImportModule, Pos)).
2009-12-04 21:32:36 +00:00
2014-12-24 15:32:29 +00:00
/**
@pred delete_import_module( + _ExportModule_, + _ImportModule_ ) is det
2014-11-02 12:10:32 +00:00
Exports in _ImportModule_ are no longer available to _Module_.
2015-02-14 11:39:07 +00:00
All exported predicates from _ExportModule_ are discarded from the
2014-11-02 12:10:32 +00:00
ones used vy the source module _ImportModule_.
*/
2009-12-04 21:32:36 +00:00
delete_import_module(Mod, ImportModule) :-
var(Mod),
'$do_error'(instantiation_error,delete_import_module(Mod, ImportModule)).
delete_import_module(Mod, ImportModule) :-
var(ImportModule),
'$do_error'(instantiation_error,delete_import_module(Mod, ImportModule)).
delete_import_module(Mod, ImportModule) :-
atom(Mod),
atom(ImportModule), !,
retractall(prolog:'$parent_module'(Mod,ImportModule)).
delete_import_module(Mod, ImportModule) :-
\+ atom(Mod), !,
'$do_error'(type_error(atom,Mod),delete_import_module(Mod, ImportModule)).
delete_import_module(Mod, ImportModule) :-
'$do_error'(type_error(atom,ImportModule),delete_import_module(Mod, ImportModule)).
2011-06-14 09:04:03 +01:00
'$set_source_module'(Source0, SourceF) :-
prolog_load_context(module, Source0), !,
module(SourceF).
'$set_source_module'(Source0, SourceF) :-
current_module(Source0, SourceF).
2014-12-24 15:32:29 +00:00
/**
@pred module_property( + _Module_, ? _Property_ ) is nondet
2014-11-02 12:10:32 +00:00
Enumerate non-deterministically the main properties of _Module_ .
Reports the following properties of _Module_:
+ `class`( ?_Class_ ): whether it is a `system`, `library`, or `user` module.
+ `line_count`(?_Ls_): number of lines in source file.
+ `file`(?_F_): source file for _Module_.
+ `exports`(-Es): list of all predicate symbols and
operator symbols exported or re-exported by this module.
*/
2014-06-16 15:00:04 +01:00
module_property(Mod, class(L)) :-
'$module_class'(Mod, L).
2013-11-25 15:50:57 +00:00
module_property(Mod, line_count(L)) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(_F,Mod,_,_,L),_).
2011-06-14 09:04:03 +01:00
module_property(Mod, file(F)) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(F,Mod,_,_,_),_).
2011-06-14 09:04:03 +01:00
module_property(Mod, exports(Es)) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(_,Mod,_,Es,_),_).
2011-06-14 09:04:03 +01:00
2014-06-17 15:06:23 +01:00
'$module_class'(Mod, system) :- '$system_module'( Mod ).
'$module_class'(Mod, library) :- '$library_module'( Mod ).
'$module_class'(Mod, user) :- '$user_module'( Mod ).
'$module_class'(_, temporary) :- fail.
'$module_class'(_, test) :- fail.
'$module_class'(_, development) :- fail.
2014-06-16 15:00:04 +01:00
'$library_module'(M1) :-
2014-09-22 18:13:35 +01:00
recorded('$module','$module'(F, M1, _, _MyExports,_Line),_),
2014-06-17 15:06:23 +01:00
user:library_directory(D),
2014-06-16 15:00:04 +01:00
sub_atom(F, 0, _, _, D).
2015-02-14 11:39:07 +00:00
'$user_module'( Mod ) :-
2014-06-17 15:06:23 +01:00
\+ '$library_module'( Mod),
\+ '$system_module'( Mod).
2014-06-16 15:00:04 +01:00
2013-11-08 00:02:38 +00:00
ls_imports :-
recorded('$import','$import'(M0,M,G0,G,_N,_K),_R),
numbervars(G0+G, 0, _),
format('~a:~w <- ~a:~w~n', [M, G, M0, G0]),
fail.
ls_imports.
2014-09-25 08:50:26 +01:00
unload_module(Mod) :-
clause( '$meta_predicate'(_F,Mod,_N,_P), _, R),
erase(R),
fail.
unload_module(Mod) :-
recorded('$multifile_defs','$defined'(_FileName,_Name,_Arity,Mod), R),
erase(R),
fail.
unload_module(Mod) :-
recorded( '$foreign', Mod:_Foreign, R),
erase(R),
fail.
% remove imported modules
unload_module(Mod) :-
setof( M, recorded('$import',_G0^_G^_N^_K^_R^'$import'(Mod,M,_G0,_G,_N,_K),_R), Ms),
2014-10-07 01:35:41 +01:00
recorded('$module','$module'( _, Mod, _, _, Exports), _),
2014-09-25 08:50:26 +01:00
lists:member(M, Ms),
current_op(X, Y, M:Op),
lists:member( op(X, Y, Op), Exports ),
op(X, 0, M:Op),
fail.
unload_module(Mod) :-
2014-10-07 01:35:41 +01:00
recorded('$module','$module'( _, Mod, _, _, Exports), _),
lists:member( op(X, _Y, Op), Exports ),
2014-09-25 08:50:26 +01:00
op(X, 0, Mod:Op),
fail.
unload_module(Mod) :-
current_predicate(Mod:P),
abolish(P),
fail.
unload_module(Mod) :-
recorded('$import','$import'(Mod,_M,_G0,_G,_N,_K),R),
erase(R),
fail.
unload_module(Mod) :-
recorded('$module','$module'( _, Mod, _, _, _), R),
erase(R),
fail.
2015-09-21 23:05:36 +01:00
/* debug */
module_state :-
recorded('$module','$module'(HostF,HostM,SourceF, Everything, Line),_),
format('HostF ~a, HostM ~a, SourceF ~w, Everything ~w, Line ~d.~n', [HostF,HostM,SourceF, Everything, Line]),
fail.
module_state.