more SWI emulation.
This commit is contained in:
		| @@ -24,13 +24,19 @@ INSTALL_PROGRAM=@INSTALL_PROGRAM@ | ||||
| srcdir=@srcdir@ | ||||
| YAP_EXTRAS=@YAP_EXTRAS@  | ||||
|  | ||||
| PROGRAMS= $(srcdir)/debug.pl \ | ||||
| PROGRAMS= $(srcdir)/base64.pl \ | ||||
| 	$(srcdir)/broadcast.pl \ | ||||
| 	$(srcdir)/debug.pl \ | ||||
| 	$(srcdir)/maplist.pl \ | ||||
| 	$(srcdir)/operators.pl \ | ||||
| 	$(srcdir)/option.pl \ | ||||
| 	$(srcdir)/prolog_source.pl \ | ||||
| 	$(srcdir)/prolog_xref.pl \ | ||||
| 	$(srcdir)/shlib.pl | ||||
| 	$(srcdir)/quintus.pl \ | ||||
| 	$(srcdir)/settings.pl \ | ||||
| 	$(srcdir)/shlib.pl \ | ||||
| 	$(srcdir)/url.pl \ | ||||
| 	$(srcdir)/utf8.pl | ||||
|  | ||||
|  | ||||
| install: $(PROGRAMS) | ||||
|   | ||||
							
								
								
									
										230
									
								
								LGPL/base64.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										230
									
								
								LGPL/base64.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,230 @@ | ||||
| /*  $Id$ | ||||
|  | ||||
|     Part of SWI-Prolog | ||||
|  | ||||
|     Author:        Jan Wielemaker | ||||
|     E-mail:        wielemak@science.uva.nl | ||||
|     WWW:           http://www.swi-prolog.org | ||||
|     Copyright (C): 1985-2007, University of Amsterdam | ||||
|  | ||||
|     This program is free software; you can redistribute it and/or | ||||
|     modify it under the terms of the GNU General Public License | ||||
|     as published by the Free Software Foundation; either version 2 | ||||
|     of the License, or (at your option) any later version. | ||||
|  | ||||
|     This program is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public | ||||
|     License along with this library; if not, write to the Free Software | ||||
|     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  | ||||
|     As a special exception, if you link this library with other files, | ||||
|     compiled with a Free Software compiler, to produce an executable, this | ||||
|     library does not by itself cause the resulting executable to be covered | ||||
|     by the GNU General Public License. This exception does not however | ||||
|     invalidate any other reasons why the executable file might be covered by | ||||
|     the GNU General Public License. | ||||
| */ | ||||
|  | ||||
| :- module(base64, | ||||
| 	  [ base64/2,			% ?PlainText, ?Encoded | ||||
| 	    base64//1			% ?PlainText | ||||
| 	  ]). | ||||
|  | ||||
| /** <module> Base64 encoding and decoding | ||||
|  | ||||
| Prolog-based base64 encoding using  DCG   rules.  Encoding  according to | ||||
| rfc2045. For example: | ||||
|  | ||||
| == | ||||
| 1 ?- base64('Hello World', X). | ||||
|  | ||||
| X = 'SGVsbG8gV29ybGQ=' | ||||
|  | ||||
| Yes | ||||
| 2 ?- base64(H, 'SGVsbG8gV29ybGQ='). | ||||
|  | ||||
| H = 'Hello World' | ||||
| == | ||||
|  | ||||
| @tbd	Stream I/O | ||||
| @tbd	White-space introduction and parsing | ||||
| @author	Jan Wielemaker | ||||
| */ | ||||
|  | ||||
| %%	base64(+Plain, -Encoded) is det. | ||||
| %%	base64(-Plain, +Encoded) is det. | ||||
| % | ||||
| %	Translates between plaintext and base64  encoded atom or string. | ||||
| %	See also base64//1. | ||||
|  | ||||
| base64(Plain, Encoded) :- | ||||
| 	nonvar(Plain), !, | ||||
| 	atom_codes(Plain, PlainCodes), | ||||
| 	phrase(base64(PlainCodes), EncCodes), | ||||
| 	atom_codes(Encoded, EncCodes). | ||||
| base64(Plain, Encoded) :- | ||||
| 	nonvar(Encoded), !, | ||||
| 	atom_codes(Encoded, EncCodes), | ||||
| 	phrase(base64(PlainCodes), EncCodes), | ||||
| 	atom_codes(Plain, PlainCodes). | ||||
| base64(_, _) :- | ||||
| 	throw(error(instantiation_error, _)). | ||||
|  | ||||
|  | ||||
| %%	base64(+PlainText)// is det. | ||||
| %%	base64(-PlainText)// is det. | ||||
| % | ||||
| %	Encode/decode list of character codes using _base64_.  See also | ||||
| %	base64/2. | ||||
|  | ||||
| base64(Input) --> | ||||
| 	{ nonvar(Input) }, !, | ||||
| 	encode(Input). | ||||
| base64(Output) --> | ||||
| 	decode(Output). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	      ENCODING		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| encode([I0, I1, I2|Rest]) --> !, | ||||
| 	[O0, O1, O2, O3], | ||||
| 	{ A is (I0<<16)+(I1<<8)+I2, | ||||
| 	  O00 is (A>>18) /\ 0x3f, | ||||
| 	  O01 is (A>>12) /\ 0x3f, | ||||
| 	  O02 is  (A>>6) /\ 0x3f, | ||||
| 	  O03 is       A /\ 0x3f, | ||||
| 	  base64_char(O00, O0), | ||||
| 	  base64_char(O01, O1), | ||||
| 	  base64_char(O02, O2), | ||||
| 	  base64_char(O03, O3) | ||||
| 	}, | ||||
| 	encode(Rest). | ||||
| encode([I0, I1]) --> !, | ||||
| 	[O0, O1, O2, 0'=], | ||||
| 	{ A is (I0<<16)+(I1<<8), | ||||
| 	  O00 is (A>>18) /\ 0x3f, | ||||
| 	  O01 is (A>>12) /\ 0x3f, | ||||
| 	  O02 is  (A>>6) /\ 0x3f, | ||||
| 	  base64_char(O00, O0), | ||||
| 	  base64_char(O01, O1), | ||||
| 	  base64_char(O02, O2) | ||||
| 	}. | ||||
| encode([I0]) --> !, | ||||
| 	[O0, O1, 0'=, 0'=], | ||||
| 	{ A is (I0<<16), | ||||
| 	  O00 is (A>>18) /\ 0x3f, | ||||
| 	  O01 is (A>>12) /\ 0x3f, | ||||
| 	  base64_char(O00, O0), | ||||
| 	  base64_char(O01, O1) | ||||
| 	}. | ||||
| encode([]) --> | ||||
| 	[]. | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	      DECODE		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| decode(Text) --> | ||||
| 	[C0, C1, C2, C3], !, | ||||
| 	{ base64_char(B0, C0), | ||||
| 	  base64_char(B1, C1) | ||||
| 	}, !, | ||||
| 	{   C3 == 0'= | ||||
| 	->  (   C2 == 0'= | ||||
| 	    ->  A is (B0<<18) + (B1<<12), | ||||
| 	        I0 is (A>>16) /\ 0xff, | ||||
| 	        Text = [I0|Rest] | ||||
| 	    ;   base64_char(B2, C2) | ||||
| 	    ->  A is (B0<<18) + (B1<<12) + (B2<<6), | ||||
| 	        I0 is (A>>16) /\ 0xff, | ||||
| 	        I1 is  (A>>8) /\ 0xff, | ||||
| 	        Text = [I0,I1|Rest] | ||||
| 	    ) | ||||
| 	;   base64_char(B2, C2), | ||||
| 	    base64_char(B3, C3) | ||||
| 	->  A is (B0<<18) + (B1<<12) + (B2<<6) + B3, | ||||
| 	    I0 is (A>>16) /\ 0xff, | ||||
| 	    I1 is  (A>>8) /\ 0xff, | ||||
| 	    I2 is      A  /\ 0xff, | ||||
| 	    Text = [I0,I1,I2|Rest] | ||||
| 	}, | ||||
| 	decode(Rest). | ||||
| decode([]) --> | ||||
| 	[]. | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *   BASIC CHARACTER ENCODING	* | ||||
| 		 *******************************/ | ||||
|  | ||||
| base64_char(00, 0'A). | ||||
| base64_char(01, 0'B). | ||||
| base64_char(02, 0'C). | ||||
| base64_char(03, 0'D). | ||||
| base64_char(04, 0'E). | ||||
| base64_char(05, 0'F). | ||||
| base64_char(06, 0'G). | ||||
| base64_char(07, 0'H). | ||||
| base64_char(08, 0'I). | ||||
| base64_char(09, 0'J). | ||||
| base64_char(10, 0'K). | ||||
| base64_char(11, 0'L). | ||||
| base64_char(12, 0'M). | ||||
| base64_char(13, 0'N). | ||||
| base64_char(14, 0'O). | ||||
| base64_char(15, 0'P). | ||||
| base64_char(16, 0'Q). | ||||
| base64_char(17, 0'R). | ||||
| base64_char(18, 0'S). | ||||
| base64_char(19, 0'T). | ||||
| base64_char(20, 0'U). | ||||
| base64_char(21, 0'V). | ||||
| base64_char(22, 0'W). | ||||
| base64_char(23, 0'X). | ||||
| base64_char(24, 0'Y). | ||||
| base64_char(25, 0'Z). | ||||
| base64_char(26, 0'a). | ||||
| base64_char(27, 0'b). | ||||
| base64_char(28, 0'c). | ||||
| base64_char(29, 0'd). | ||||
| base64_char(30, 0'e). | ||||
| base64_char(31, 0'f). | ||||
| base64_char(32, 0'g). | ||||
| base64_char(33, 0'h). | ||||
| base64_char(34, 0'i). | ||||
| base64_char(35, 0'j). | ||||
| base64_char(36, 0'k). | ||||
| base64_char(37, 0'l). | ||||
| base64_char(38, 0'm). | ||||
| base64_char(39, 0'n). | ||||
| base64_char(40, 0'o). | ||||
| base64_char(41, 0'p). | ||||
| base64_char(42, 0'q). | ||||
| base64_char(43, 0'r). | ||||
| base64_char(44, 0's). | ||||
| base64_char(45, 0't). | ||||
| base64_char(46, 0'u). | ||||
| base64_char(47, 0'v). | ||||
| base64_char(48, 0'w). | ||||
| base64_char(49, 0'x). | ||||
| base64_char(50, 0'y). | ||||
| base64_char(51, 0'z). | ||||
| base64_char(52, 0'0). | ||||
| base64_char(53, 0'1). | ||||
| base64_char(54, 0'2). | ||||
| base64_char(55, 0'3). | ||||
| base64_char(56, 0'4). | ||||
| base64_char(57, 0'5). | ||||
| base64_char(58, 0'6). | ||||
| base64_char(59, 0'7). | ||||
| base64_char(60, 0'8). | ||||
| base64_char(61, 0'9). | ||||
| base64_char(62, 0'+). | ||||
| base64_char(63, 0'/). | ||||
							
								
								
									
										177
									
								
								LGPL/broadcast.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										177
									
								
								LGPL/broadcast.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,177 @@ | ||||
| /*  $Id$ | ||||
|  | ||||
|     Part of SWI-Prolog | ||||
|  | ||||
|     Author:        Jan Wielemaker | ||||
|     E-mail:        wielemak@science.uva.nl | ||||
|     WWW:           http://www.swi-prolog.org | ||||
|     Copyright (C): 1985-2006, University of Amsterdam | ||||
|  | ||||
|     This program is free software; you can redistribute it and/or | ||||
|     modify it under the terms of the GNU General Public License | ||||
|     as published by the Free Software Foundation; either version 2 | ||||
|     of the License, or (at your option) any later version. | ||||
|  | ||||
|     This program is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public | ||||
|     License along with this library; if not, write to the Free Software | ||||
|     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  | ||||
|     As a special exception, if you link this library with other files, | ||||
|     compiled with a Free Software compiler, to produce an executable, this | ||||
|     library does not by itself cause the resulting executable to be covered | ||||
|     by the GNU General Public License. This exception does not however | ||||
|     invalidate any other reasons why the executable file might be covered by | ||||
|     the GNU General Public License. | ||||
| */ | ||||
|  | ||||
| :- module(broadcast, | ||||
| 	  [ listen/3,		% Listener x Templ x Goal | ||||
| 	    listen/2,		% Templ x Goal | ||||
| 	    unlisten/1,		% Listener | ||||
| 	    unlisten/2,		% Listener x Templ | ||||
| 	    unlisten/3,		% Listener x Templ x Goal | ||||
| 	    listening/3,	% Listener x Templ x Goal | ||||
| 	    broadcast/1,	% Templ | ||||
| 	    broadcast_request/1	% Templ | ||||
| 	  ]). | ||||
| :- meta_predicate | ||||
| 	listen(+, :), | ||||
| 	listen(+, +, :), | ||||
| 	unlisten(+, +, :). | ||||
|  | ||||
| :- dynamic | ||||
| 	listener/4. | ||||
|  | ||||
| /** <module> Event service | ||||
|  | ||||
| Generic broadcasting service. Broadcasts are   made  using the predicate | ||||
| broadcast(+Templ). All registered  `listeners'  will   have  their  goal | ||||
| called. Success or failure of this is ignored. The listener can not bind | ||||
| arguments. | ||||
|  | ||||
| This library is particulary  useful  for   disconnecting  modules  in an | ||||
| application. Modules can broadcast events  such as changes, anticipating | ||||
| other modules need to react on   such  changes. For example, settings.pl | ||||
| broadcasts changes to settings, allowing dependent   modules to react on | ||||
| changes: | ||||
|  | ||||
| == | ||||
| :- listing(setting(changed(http:workers, New)), | ||||
| 	   change_workers(New)). | ||||
|  | ||||
| change_workers(New) :- | ||||
| 	setting(http:port, Port), | ||||
| 	http_workers(Port, New). | ||||
| == | ||||
| */ | ||||
|  | ||||
| %%	listen(+Listener, +Templ, :Goal) is det. | ||||
| %%	listen(+Templ, :Goal) is det. | ||||
| % | ||||
| %	Open a channel for listening for events of the given `Templ'. | ||||
|  | ||||
| listen(Listener0, Templ, Goal) :- | ||||
| 	canonical_listener(Listener0, Listener), | ||||
| 	strip_module(Goal, Module, TheGoal), | ||||
| 	assert_listener(Templ, Listener, Module, TheGoal). | ||||
|  | ||||
| listen(Templ, Goal) :- | ||||
| 	strip_module(Goal, Module, TheGoal), | ||||
| 	assert_listener(Templ, Module, Module, TheGoal). | ||||
|  | ||||
|  | ||||
| %%	unlisten(+Listener) is det. | ||||
| %%	unlisten(+Listener, +Templ) is det. | ||||
| %%	unlisten(+Listener, +Templ, :Goal) is det. | ||||
| % | ||||
| %	Destroy a channel. All arguments may  be variables, removing the | ||||
| %	all matching listening channals. | ||||
|  | ||||
| unlisten(Listener0) :- | ||||
| 	canonical_listener(Listener0, Listener), | ||||
| 	retractall(listener(_, Listener, _, _)). | ||||
| unlisten(Listener0, Templ) :- | ||||
| 	canonical_listener(Listener0, Listener), | ||||
| 	retractall(listener(Templ, Listener, _, _)). | ||||
| unlisten(Listener0, Templ, Goal) :- | ||||
| 	canonical_listener(Listener0, Listener), | ||||
| 	(   var(Goal) | ||||
| 	->  true | ||||
| 	;   strip_module(Goal, Module, TheGoal) | ||||
| 	), | ||||
| 	retract_listener(Templ, Listener, Module, TheGoal). | ||||
|  | ||||
|  | ||||
| %%	listening(?Listener, ?Templ, ?Goal) is nondet. | ||||
| % | ||||
| %	returns currently open channels | ||||
|  | ||||
| listening(Listener0, Templ, Module:Goal) :- | ||||
| 	canonical_listener(Listener0, Listener), | ||||
| 	listener(Templ, Listener, Module, Goal). | ||||
|  | ||||
|  | ||||
| %%	broadcast(+Templ) is det. | ||||
| % | ||||
| %	Broadcast given event. | ||||
|  | ||||
| broadcast(Templ) :- | ||||
| 	(   listener(Templ, _Listener, Module, Goal), | ||||
| 	    (   Module:Goal | ||||
| 	    ->  fail | ||||
| 	    ) | ||||
| 	;   true | ||||
| 	). | ||||
|  | ||||
|  | ||||
| %%	broadcast_request(+Templ) is nonet. | ||||
| % | ||||
| %	Broadcast given event till accepted.  Succeeds then, fail if no | ||||
| %	listener accepts the call.  Bindings made by the listener goal | ||||
| %	are maintained.  May be used to make broadcast requests. | ||||
|  | ||||
| broadcast_request(Templ) :- | ||||
| 	listener(Templ, _Listener, Module, Goal), | ||||
| 	Module:Goal. | ||||
|  | ||||
|  | ||||
| %	{assert,retract}_listener(+Templ, +Listener, +Module, +Goal) | ||||
| % | ||||
| %	Implemented as sub-predicate to ensure storage in this module. | ||||
| %	Second registration is ignored.  Is this ok?  It avoids problems | ||||
| %	using multiple registration of global listen channels. | ||||
|  | ||||
| assert_listener(Templ, Listener, Module, TheGoal) :- | ||||
| 	listener(Templ, Listener, Module, TheGoal), !. | ||||
| assert_listener(Templ, Listener, Module, TheGoal) :- | ||||
| 	asserta(listener(Templ, Listener, Module, TheGoal)). | ||||
|  | ||||
| retract_listener(Templ, Listener, Module, TheGoal) :- | ||||
| 	retractall(listener(Templ, Listener, Module, TheGoal)). | ||||
|  | ||||
| %%	canonical_listener(+Raw, -Canonical) | ||||
| % | ||||
| %	Entry for later optimization. | ||||
|  | ||||
| canonical_listener(Templ, Templ). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	  GOAL EXPANSION	* | ||||
| 		 *******************************/ | ||||
|  | ||||
| :- multifile | ||||
| 	user:goal_expansion/2. | ||||
|  | ||||
| user:goal_expansion(listen(L,T,G0), listen(L,T,G)) :- | ||||
| 	expand_goal(G0, G). | ||||
| user:goal_expansion(listen(T,G0), listen(T,G)) :- | ||||
| 	expand_goal(G0, G). | ||||
| user:goal_expansion(unlisten(L,T,G0), unlisten(L,T,G)) :- | ||||
| 	expand_goal(G0, G). | ||||
|  | ||||
							
								
								
									
										401
									
								
								LGPL/quintus.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										401
									
								
								LGPL/quintus.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,401 @@ | ||||
| /*  $Id$ | ||||
|  | ||||
|     Part of SWI-Prolog | ||||
|  | ||||
|     Author:        Jan Wielemaker | ||||
|     E-mail:        J.Wielemaker@uva.nl | ||||
|     WWW:           http://www.swi-prolog.org | ||||
|     Copyright (C): 1985-2008, University of Amsterdam | ||||
|  | ||||
|     This program is free software; you can redistribute it and/or | ||||
|     modify it under the terms of the GNU General Public License | ||||
|     as published by the Free Software Foundation; either version 2 | ||||
|     of the License, or (at your option) any later version. | ||||
|  | ||||
|     This program is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU Lesser General Public | ||||
|     License along with this library; if not, write to the Free Software | ||||
|     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  | ||||
|     As a special exception, if you link this library with other files, | ||||
|     compiled with a Free Software compiler, to produce an executable, this | ||||
|     library does not by itself cause the resulting executable to be covered | ||||
|     by the GNU General Public License. This exception does not however | ||||
|     invalidate any other reasons why the executable file might be covered by | ||||
|     the GNU General Public License. | ||||
| */ | ||||
|  | ||||
| :- module(quintus, | ||||
| 	[ % unix/1, | ||||
| %	  file_exists/1, | ||||
|  | ||||
| 	  abs/2, | ||||
| 	  sin/2, | ||||
| 	  cos/2, | ||||
| 	  tan/2, | ||||
| 	  log/2, | ||||
| 	  log10/2, | ||||
| 	  pow/3, | ||||
| 	  ceiling/2, | ||||
| 	  floor/2, | ||||
| 	  round/2, | ||||
| 	  acos/2, | ||||
| 	  asin/2, | ||||
| 	  atan/2, | ||||
| 	  atan2/3, | ||||
| 	  sign/2, | ||||
| 	  sqrt/2, | ||||
|  | ||||
| 	  genarg/3, | ||||
|  | ||||
| 	  (mode)/1, | ||||
| 	  (public)/1, | ||||
| 	  no_style_check/1, | ||||
| 	  otherwise/0, | ||||
| 	  simple/1, | ||||
| %	  statistics/2,			% Please access as quintus:statistics/2 | ||||
| 	  prolog_flag/2, | ||||
|  | ||||
| 	  date/1,			% -date(Year, Month, Day) | ||||
|  | ||||
| 	  current_stream/3,		% ?File, ?Mode, ?Stream | ||||
| 	  stream_position/3,		% +Stream, -Old, +New | ||||
| 	  skip_line/0, | ||||
| 	  skip_line/1,			% +Stream | ||||
|  | ||||
| 	  compile/1,			% +File(s) | ||||
|  | ||||
| 	  atom_char/2, | ||||
| 	  midstring/3,			% ABC, B, AC | ||||
| 	  midstring/4,			% ABC, B, AC, LenA | ||||
| 	  midstring/5,			% ABC, B, AC, LenA, LenB | ||||
| 	  midstring/6,			% ABC, B, AC, LenA, LenB, LenC | ||||
|  | ||||
| 	  raise_exception/1,		% +Exception | ||||
| 	  on_exception/3		% +Ball, :Goal, :Recover | ||||
| 	]). | ||||
| :- use_module(library(lists), [member/2]). | ||||
|  | ||||
| /** <module> Quintus compatibility | ||||
|  | ||||
| This  module  defines  several  predicates    from  the  Quintus  Prolog | ||||
| libraries. Note that our library structure is totally different. If this | ||||
| library were complete, Prolog  code  could   be  ported  by removing the | ||||
| use_module/1 declarations, relying on the SWI-Prolog autoloader. | ||||
|  | ||||
| Bluffers guide to porting: | ||||
|  | ||||
| 	* Remove =|use_module(library(...))|= | ||||
| 	* Run =|?- list_undefined.|= | ||||
| 	* Fix problems | ||||
|  | ||||
| Of course, this library is incomplete ... | ||||
| */ | ||||
|  | ||||
| 		/******************************** | ||||
| 		*      SYSTEM INTERACTION       * | ||||
| 		*********************************/ | ||||
|  | ||||
| % %%	unix(+Action) | ||||
| % %	interface to  Unix. | ||||
|  | ||||
| % unix(system(Command)) :- | ||||
| %         shell(Command). | ||||
| % unix(shell(Command)) :- | ||||
| %         shell(Command). | ||||
| % unix(shell) :- | ||||
| %         shell. | ||||
| % unix(access(File, 0)) :- | ||||
| %         access_file(File, read). | ||||
| % unix(cd) :- | ||||
| % 	expand_file_name(~, [Home]), | ||||
| % 	working_directory(_, Home). | ||||
| % unix(cd(Dir)) :- | ||||
| % 	working_directory(_, Dir). | ||||
| % unix(args(L)) :- | ||||
| % 	current_prolog_flag(argv, L). | ||||
| % unix(argv(L)) :- | ||||
| % 	current_prolog_flag(argv, S), | ||||
| % 	maplist(to_prolog, S, L). | ||||
|  | ||||
| % to_prolog(S, A) :- | ||||
| % 	name(S, L), | ||||
| % 	name(A, L). | ||||
|  | ||||
|  | ||||
| 		/******************************** | ||||
| 		*        META PREDICATES        * | ||||
| 		*********************************/ | ||||
|  | ||||
| %%	otherwise | ||||
| % | ||||
| %	For (A -> B ; otherwise -> C) | ||||
|  | ||||
| % otherwise. | ||||
|  | ||||
|  | ||||
| 		/******************************** | ||||
| 		*          ARITHMETIC           * | ||||
| 		*********************************/ | ||||
|  | ||||
| %%	abs(+Number, -Absolute) | ||||
| %	Unify `Absolute' with the absolute value of `Number'. | ||||
|  | ||||
| abs(Number, Absolute) :- | ||||
| 	Absolute is abs(Number). | ||||
|  | ||||
| %%	sin(+Angle, -Sine) is det. | ||||
| %%	cos(+Angle, -Cosine) is det. | ||||
| %%	tan(+Angle, -Tangent) is det. | ||||
| %%	log(+X, -NatLog) is det. | ||||
| %%	log10(+X, -Log) is det. | ||||
| % | ||||
| %	Math library predicates. SWI-Prolog (and   ISO) support these as | ||||
| %	functions under is/2, etc. | ||||
|  | ||||
| sin(A, V) :-	  V is sin(A). | ||||
| cos(A, V) :-	  V is cos(A). | ||||
| tan(A, V) :-	  V is tan(A). | ||||
| log(A, V) :-	  V is log(A). | ||||
| log10(X, V) :-	  V is log10(X). | ||||
| pow(X,Y,V) :-	  V is X**Y. | ||||
| ceiling(X, V) :-  V is ceil(X). | ||||
| floor(X, V) :-	  V is floor(X). | ||||
| round(X, V) :-	  V is round(X). | ||||
| sqrt(X, V) :-	  V is sqrt(X). | ||||
| acos(X, V) :-	  V is acos(X). | ||||
| asin(X, V) :-	  V is asin(X). | ||||
| atan(X, V) :-	  V is atan(X). | ||||
| atan2(Y, X, V) :- V is atan(Y, X). | ||||
| sign(X, V) :-	  V is sign(X). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	TERM MANIPULATION	* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	genarg(?Index, +Term, ?Arg) is nondet. | ||||
| % | ||||
| %	Generalised version of ISO arg/3.  SWI-Prolog's arg/3 is already | ||||
| %	genarg/3. | ||||
|  | ||||
| genarg(N, T, A) :-			% SWI-Prolog arg/3 is generic | ||||
| 	arg(N, T, A). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	      FLAGS		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	prolog_flag(?Flag, ?Value) is nondet. | ||||
| % | ||||
| %	Same as ISO current_prolog_flag/2.  Maps =version=. | ||||
| % | ||||
| %	@bug	Should map relevant Quintus flag identifiers. | ||||
|  | ||||
| % prolog_flag(version, Version) :- !, | ||||
| % 	current_prolog_flag(version_data, swi(Major, Minor, Patch, _)), | ||||
| % 	current_prolog_flag(arch, Arch), | ||||
| % 	current_prolog_flag(compiled_at, Compiled), | ||||
| % 	atomic_list_concat(['SWI-Prolog ', | ||||
| % 		     Major, '.', Minor, '.', Patch, | ||||
| % 		     ' (', Arch, '): ', Compiled], Version). | ||||
| % prolog_flag(Flag, Value) :- | ||||
| % 	current_prolog_flag(Flag, Value). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	    STATISTICS		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %	Here used to be a definition of Quintus statistics/2 in traditional | ||||
| %	SWI-Prolog statistics/2.  The current built-in emulates Quintus | ||||
| %	almost completely. | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	     DATE/TIME		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	date(-Date) is det. | ||||
| % | ||||
| %	Get current date as date(Y,M,D) | ||||
|  | ||||
| date(Date) :- | ||||
| 	get_time(T), | ||||
| 	stamp_date_time(T, DaTime, local), | ||||
| 	date_time_value(date, DaTime, Date). | ||||
|  | ||||
|  | ||||
| 		/******************************** | ||||
| 		*          STYLE CHECK          * | ||||
| 		*********************************/ | ||||
|  | ||||
| %%	no_style_check(Style) is det. | ||||
| % | ||||
| %	Same as SWI-Prolog =|style_check(-Style)|=.   The Quintus option | ||||
| %	=single_var= is mapped to =singleton=. | ||||
| % | ||||
| %	@see style_check/1. | ||||
|  | ||||
| q_style_option(single_var, singleton) :- !. | ||||
| q_style_option(Option, Option). | ||||
|  | ||||
| % no_style_check(QOption) :- | ||||
| % 	q_style_option(QOption, SWIOption), | ||||
| % 	style_check(-SWIOption). | ||||
|  | ||||
|  | ||||
| 		/******************************** | ||||
| 		*         DIRECTIVES            * | ||||
| 		*********************************/ | ||||
|  | ||||
| % :- op(1150, fx, [(mode), (public)]). | ||||
|  | ||||
| % mode(_). | ||||
| % public(_). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	      TYPES		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	simple(@Term) is semidet. | ||||
| % | ||||
| %	Term is atomic or a variable. | ||||
|  | ||||
| % simple(X) :- | ||||
| % 	(   atomic(X) | ||||
| % 	->  true | ||||
| % 	;   var(X) | ||||
| % 	). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	      STREAMS		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	current_stream(?Object, ?Mode, ?Stream) | ||||
| % | ||||
| %	SICStus/Quintus and backward compatible predicate.  New code should | ||||
| %	be using the ISO compatible stream_property/2. | ||||
|  | ||||
| % current_stream(Object, Mode, Stream) :- | ||||
| % 	stream_property(Stream, mode(FullMode)), | ||||
| % 	stream_mode(FullMode, Mode), | ||||
| % 	(   stream_property(Stream, file_name(Object0)) | ||||
| % 	->  true | ||||
| % 	;   stream_property(Stream, file_no(Object0)) | ||||
| % 	->  true | ||||
| % 	;   Object0 = [] | ||||
| % 	), | ||||
| % 	Object = Object0. | ||||
|  | ||||
| % stream_mode(read,   read). | ||||
| % stream_mode(write,  write). | ||||
| % stream_mode(append, write). | ||||
| % stream_mode(update, write). | ||||
|  | ||||
| % %%	stream_position(+Stream, -Old, +New) | ||||
|  | ||||
| % stream_position(Stream, Old, New) :- | ||||
| % 	stream_property(Stream, position(Old)), | ||||
| % 	set_stream_position(Stream, New). | ||||
|  | ||||
|  | ||||
| %%	skip_line is det. | ||||
| %%	skip_line(Stream) is det. | ||||
| % | ||||
| %	Skip  the  rest  of  the  current  line  (on  Stream).  Same  as | ||||
| %	=|skip(0'\n)|=. | ||||
|  | ||||
| skip_line :- | ||||
| 	skip(10). | ||||
| skip_line(Stream) :- | ||||
| 	skip(Stream, 10). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	   COMPILATION		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	compile(+Files) is det. | ||||
| % | ||||
| %	Compile   files.   SWI-Prolog   doesn't    distinguish   between | ||||
| %	compilation and consult. | ||||
| % | ||||
| %	@see load_files/2. | ||||
|  | ||||
| % :- meta_predicate | ||||
| % 	compile(:). | ||||
|  | ||||
| % compile(Files) :- | ||||
| % 	consult(Files). | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	   ATOM-HANDLING	* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	atom_char(+Char, -Code) is det. | ||||
| %%	atom_char(-Char, +Code) is det. | ||||
| % | ||||
| %	Same as ISO char_code/2. | ||||
|  | ||||
| atom_char(Char, Code) :- | ||||
| 	char_code(Char, Code). | ||||
|  | ||||
| %%	midstring(?ABC, ?B, ?AC) is nondet. | ||||
| %%	midstring(?ABC, ?B, ?AC, LenA) is nondet. | ||||
| %%	midstring(?ABC, ?B, ?AC, LenA, LenB) is nondet. | ||||
| %%	midstring(?ABC, ?B, ?AC, LenA, LenB, LenC) is nondet. | ||||
| % | ||||
| %	Too difficult to explain.  See the Quintus docs.  As far as I | ||||
| %	understand them the code below emulates this function just fine. | ||||
|  | ||||
| midstring(ABC, B, AC) :- | ||||
| 	midstring(ABC, B, AC, _, _, _). | ||||
| midstring(ABC, B, AC, LenA) :- | ||||
| 	midstring(ABC, B, AC, LenA, _, _). | ||||
| midstring(ABC, B, AC, LenA, LenB) :- | ||||
| 	midstring(ABC, B, AC, LenA, LenB, _). | ||||
| midstring(ABC, B, AC, LenA, LenB, LenC) :-	% -ABC, +B, +AC | ||||
| 	var(ABC), !, | ||||
| 	atom_length(AC, LenAC), | ||||
| 	(   nonvar(LenA) ; nonvar(LenC) | ||||
| 	->  plus(LenA, LenC, LenAC) | ||||
| 	;   true | ||||
| 	), | ||||
| 	sub_atom(AC, 0, LenA, _, A), | ||||
| 	LenC is LenAC - LenA, | ||||
| 	sub_atom(AC, _, LenC, 0, C), | ||||
| 	atom_length(B, LenB), | ||||
| 	atomic_list_concat([A,B,C], ABC). | ||||
| midstring(ABC, B, AC, LenA, LenB, LenC) :- | ||||
| 	sub_atom(ABC, LenA, LenB, LenC, B), | ||||
| 	sub_atom(ABC, 0, LenA, _, A), | ||||
| 	sub_atom(ABC, _, LenC, 0, C), | ||||
| 	atom_concat(A, C, AC). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	     EXCEPTIONS		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	raise_exception(+Term) | ||||
| % | ||||
| %	Quintus compatible exception handling | ||||
|  | ||||
| % raise_exception(Term) :- | ||||
| % 	throw(Term). | ||||
|  | ||||
| %%	on_exception(+Template, :Goal, :Recover) | ||||
|  | ||||
| :- meta_predicate | ||||
| 	on_exception(+, 0, 0). | ||||
|  | ||||
| % on_exception(Except, Goal, Recover) :- | ||||
| % 	catch(Goal, Except, Recover). | ||||
							
								
								
									
										629
									
								
								LGPL/settings.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										629
									
								
								LGPL/settings.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,629 @@ | ||||
| /*  $Id$ | ||||
|  | ||||
|     Part of SWI-Prolog | ||||
|  | ||||
|     Author:        Jan Wielemaker | ||||
|     E-mail:        wielemak@science.uva.nl | ||||
|     WWW:           http://www.swi-prolog.org | ||||
|     Copyright (C): 1985-2007, University of Amsterdam | ||||
|  | ||||
|     This program is free software; you can redistribute it and/or | ||||
|     modify it under the terms of the GNU General Public License | ||||
|     as published by the Free Software Foundation; either version 2 | ||||
|     of the License, or (at your option) any later version. | ||||
|  | ||||
|     This program is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public | ||||
|     License along with this library; if not, write to the Free Software | ||||
|     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  | ||||
|     As a special exception, if you link this library with other files, | ||||
|     compiled with a Free Software compiler, to produce an executable, this | ||||
|     library does not by itself cause the resulting executable to be covered | ||||
|     by the GNU General Public License. This exception does not however | ||||
|     invalidate any other reasons why the executable file might be covered by | ||||
|     the GNU General Public License. | ||||
| */ | ||||
|  | ||||
| :- module(settings, | ||||
| 	  [ setting/4,			% :Name, +Type, +Default, +Comment | ||||
| 	    setting/2,			% :Name, ?Value | ||||
| 	    set_setting/2,		% :Name, +Value | ||||
| 	    set_setting_default/2,	% :Name, +Value | ||||
| 	    restore_setting/1,		% :Name | ||||
| 	    load_settings/1,		% +File | ||||
| 	    load_settings/2,		% +File, +Options | ||||
| 	    save_settings/0, | ||||
| 	    save_settings/1,		% +File | ||||
| 	    current_setting/1,		% Module:Name | ||||
| 	    setting_property/2,		% ?Setting, ?Property | ||||
| 	    list_settings/0, | ||||
|  | ||||
| 	    convert_setting_text/3	% +Type, +Text, -Value | ||||
| 	  ]). | ||||
| :- use_module(library(error)). | ||||
| :- use_module(library(broadcast)). | ||||
| :- use_module(library(debug)). | ||||
| :- use_module(library(option)). | ||||
|  | ||||
| /** <module> Setting management | ||||
|  | ||||
| This library allows management  of   configuration  settings  for Prolog | ||||
| applications. Applications define settings  in   one  or  multiple files | ||||
| using the directive setting/4 as illustrated below: | ||||
|  | ||||
| == | ||||
| :- use_module(library(setting)). | ||||
|  | ||||
| :- setting(version, atom,   '1.0', 'Current version'). | ||||
| :- setting(timeout, number,    20, 'Timeout in seconds'). | ||||
| == | ||||
|  | ||||
| The directive is subject to   term_expansion/2,  which guarantees proper | ||||
| synchronisation of the  database  if   source-files  are  reloaded. This | ||||
| implies it is *not* possible to call setting/4 as a predicate. | ||||
|  | ||||
| Settings are local to a  module.  This   implies  they  are defined in a | ||||
| two-level namespace. Managing settings  per   module  greatly simplifies | ||||
| assembling large applications from multiple   modules that configuration | ||||
| through  settings.  This  settings  management  library  ensures  proper | ||||
| access, loading and saving of settings. | ||||
|  | ||||
| @see	library(config) distributed with XPCE provides an alternative | ||||
| 	aimed at graphical applications. | ||||
| @author	Jan Wielemaker | ||||
| */ | ||||
|  | ||||
| :- dynamic | ||||
| 	st_value/3,			% Name, Module, Value | ||||
| 	st_default/3,			% Name, Module, Value | ||||
| 	local_file/1.			% Path | ||||
|  | ||||
| :- multifile | ||||
| 	current_setting/6.		% Name, Module, Type, Default, Comment, Source | ||||
|  | ||||
| :- meta_predicate | ||||
| 	setting(:, +, +, +), | ||||
| 	setting(:, ?), | ||||
| 	set_setting(:, +), | ||||
| 	set_setting_default(:, +), | ||||
| 	current_setting(:), | ||||
| 	restore_setting(:). | ||||
|  | ||||
| curr_setting(Name, Module, Type, Default, Comment) :- | ||||
| 	current_setting(Name, Module, Type, Default0, Comment, _Src), | ||||
| 	(   st_default(Name, Module, Default1) | ||||
| 	->  Default = Default1 | ||||
| 	;   Default = Default0 | ||||
| 	). | ||||
|  | ||||
| %%	setting(Name, Type, Default, Comment) is det. | ||||
| % | ||||
| %	Define a setting. Name denotes the name of the setting, Type its | ||||
| %	type. Default is the value before  it is modified. Default refer | ||||
| %	to environment variables  and  use   arithmetic  expressions  as | ||||
| %	defined by eval_default/4. | ||||
| % | ||||
| %	@param Name	Name of the setting (an atom) | ||||
| %	@param Type	Type for setting.  One of =any= or a type defined | ||||
| %			by must_be/2. | ||||
| %	@param Default  Default value for the setting. | ||||
| %	@param Comment	Atom containing a (short) descriptive note. | ||||
|  | ||||
|  | ||||
| setting(Name, Type, Default, Comment) :- | ||||
| 	throw(error(context_error(nodirective, | ||||
| 				  setting(Name, Type, Default, Comment)), | ||||
| 		    _)). | ||||
|  | ||||
| :- multifile | ||||
| 	system:term_expansion/2. | ||||
|  | ||||
| system:term_expansion((:- setting(QName, Type, Default, Comment)), | ||||
| 		    Expanded) :- | ||||
| 	prolog_load_context(module, M0), | ||||
| 	strip_module(M0:QName, Module, Name), | ||||
| 	must_be(atom, Name), | ||||
| 	to_atom(Comment, CommentAtom), | ||||
| 	eval_default(Default, Module, Type, Value), | ||||
| 	check_type(Type, Value), | ||||
| 	(   current_setting(Name, Module, _, _, _, OldLoc) | ||||
| 	->  format(string(Message), | ||||
| 		   'Already defined at: ~w', [OldLoc]), | ||||
| 	    throw(error(permission_error(redefine, setting, Module:Name), | ||||
| 			context(Message, _))) | ||||
| 	;   source_location(File, Line) | ||||
| 	->  Expanded = settings:current_setting(Name, Module, Type, Default, | ||||
| 						CommentAtom, File:Line) | ||||
| 	). | ||||
|  | ||||
| to_atom(Atom, Atom) :- | ||||
| 	atom(Atom), !. | ||||
| to_atom(String, Atom) :- | ||||
| 	format(atom(Atom), '~s', String). | ||||
|  | ||||
| %%	setting(:Name, ?Value) is nondet. | ||||
| % | ||||
| %	True if Name is a currently defined setting with Value. | ||||
| % | ||||
| %	@error	existence_error(setting, Name) | ||||
|  | ||||
| setting(QName, Value) :- | ||||
| 	strip_module(QName, Module, Name), | ||||
| 	(   ground(Name) | ||||
| 	->  (   st_value(Name, Module, Value0) | ||||
| 	    ->  Value = Value0 | ||||
| 	    ;   curr_setting(Name, Module, Type, Default, _) | ||||
| 	    ->	eval_default(Default, Module, Type, Value) | ||||
| 	    ;	existence_error(setting, Module:Name) | ||||
| 	    ) | ||||
| 	;   current_setting(Name, Module, _, _, _, _), | ||||
| 	    setting(Module:Name, Value) | ||||
| 	). | ||||
|  | ||||
|  | ||||
| :- dynamic | ||||
| 	setting_cache/3. | ||||
| :- volatile | ||||
| 	setting_cache/3. | ||||
|  | ||||
| %%	clear_setting_cache is det. | ||||
| % | ||||
| %	Clear the cache for evaluation of default values. | ||||
|  | ||||
| clear_setting_cache :- | ||||
| 	retractall(setting_cache(_,_,_)). | ||||
|  | ||||
| %%	eval_default(+Default, +Module, +Type, -Value) is det. | ||||
| % | ||||
| %	Convert the settings default value. The notation allows for some | ||||
| %	`function-style' notations to make the library more generic: | ||||
| % | ||||
| %		* env(Name) | ||||
| %		Get value from the given environment variable. The value | ||||
| %		is handed to convert_setting_text/3 to convert the | ||||
| %		textual representation into a Prolog term.  Raises an | ||||
| %		existence_error of the variable is not defined. | ||||
| % | ||||
| %		* env(Name, Default) | ||||
| %		As env(Name), but uses the value Default if the variable | ||||
| %		is not defined. | ||||
| % | ||||
| %		* setting(Name) | ||||
| %		Ask the value of another setting. | ||||
| % | ||||
| %		* Expression | ||||
| %		If Type is numeric, evaluate the expression.  env(Var) | ||||
| %		evaluates to the value of an environment variable. | ||||
| %		If Type is =atom=, concatenate A+B+....  Elements of the | ||||
| %		expression can be env(Name). | ||||
|  | ||||
| :- multifile | ||||
| 	eval_default/3.			% +Default, +Type, -Value | ||||
|  | ||||
| eval_default(Default, _, Type, Value) :- | ||||
| 	eval_default(Default, Type, Val), !, | ||||
| 	Value = Val. | ||||
| eval_default(Default, _, _, Value) :- | ||||
| 	atomic(Default), !, | ||||
| 	Value = Default. | ||||
| eval_default(Default, _, Type, Value) :- | ||||
| 	setting_cache(Default, Type, Val), !, | ||||
| 	Value = Val. | ||||
| eval_default(env(Name), _, Type, Value) :- !, | ||||
| 	(   getenv(Name, TextValue) | ||||
| 	->  convert_setting_text(Type, TextValue, Val), | ||||
| 	    assert(setting_cache(env(Name), Type, Val)), | ||||
| 	    Value = Val | ||||
| 	;   existence_error(environment_variable, Name) | ||||
| 	). | ||||
| eval_default(env(Name, Default), _, Type, Value) :- !, | ||||
| 	(   getenv(Name, TextValue) | ||||
| 	->  convert_setting_text(Type, TextValue, Val) | ||||
| 	;   Value = Default | ||||
| 	), | ||||
| 	assert(setting_cache(env(Name), Type, Val)), | ||||
| 	Value = Val. | ||||
| eval_default(setting(Name), Module, Type, Value) :- !, | ||||
| 	strip_module(Module:Name, M, N), | ||||
| 	setting(M:N, Value), | ||||
| 	must_be(Type, Value). | ||||
| eval_default(Expr, _, Type, Value) :- | ||||
| 	numeric_type(Type, Basic), !, | ||||
| 	Val0 is Expr, | ||||
| 	(   Basic == float | ||||
| 	->  Val is float(Val0) | ||||
| 	;   Basic = integer | ||||
| 	->  Val is round(Val0) | ||||
| 	;   Val = Val0 | ||||
| 	), | ||||
| 	assert(setting_cache(Expr, Type, Val)), | ||||
| 	Value = Val. | ||||
| eval_default(A+B, Module, atom, Value) :- !, | ||||
| 	phrase(expr_to_list(A+B, Module), L), | ||||
| 	atomic_list_concat(L, Val), | ||||
| 	assert(setting_cache(A+B, atom, Val)), | ||||
| 	Value = Val. | ||||
| eval_default(List, Module, list(Type), Value) :- !, | ||||
| 	eval_list_default(List, Module, Type, Val), | ||||
| 	assert(setting_cache(List, list(Type), Val)), | ||||
| 	Value = Val. | ||||
| eval_default(Default, _, _, Default). | ||||
|  | ||||
|  | ||||
| %%	eval_list_default(+List, +Module, +ElementType, -DefaultList) | ||||
| % | ||||
| %	Evaluate the default for a list of values. | ||||
|  | ||||
| eval_list_default([], _, _, []). | ||||
| eval_list_default([H0|T0], Module, Type, [H|T]) :- | ||||
| 	eval_default(H0, Module, Type, H), | ||||
| 	eval_list_default(T0, Module, Type, T). | ||||
|  | ||||
| %%	expr_to_list(+Expression, +Module)// is det. | ||||
| % | ||||
| %	Process the components to create an  atom. Atom concatenation is | ||||
| %	expressed as A+B. Components may refer to envrionment variables. | ||||
|  | ||||
| expr_to_list(A+B, Module) --> !, | ||||
| 	expr_to_list(A, Module), | ||||
| 	expr_to_list(B, Module). | ||||
| expr_to_list(env(Name), _) --> !, | ||||
| 	(   { getenv(Name, Text) } | ||||
| 	->  [Text] | ||||
| 	;   { existence_error(environment_variable, Name) } | ||||
| 	). | ||||
| expr_to_list(env(Name, Default), _) --> !, | ||||
| 	(   { getenv(Name, Text) } | ||||
| 	->  [Text] | ||||
| 	;   [Default] | ||||
| 	). | ||||
| expr_to_list(setting(Name), Module) --> !, | ||||
| 	{ strip_module(Module:Name, M, N), | ||||
| 	  setting(M:N, Value) | ||||
| 	}, | ||||
| 	[ Value ]. | ||||
| expr_to_list(A, _) --> | ||||
| 	[A]. | ||||
|  | ||||
| %%	env(+Name:atom, -Value:number) is det. | ||||
| %%	env(+Name:atom, +Default:number, -Value:number) is det | ||||
| % | ||||
| %	Evaluate  environment  variables   on    behalf   of  arithmetic | ||||
| %	expressions. | ||||
|  | ||||
| :- arithmetic_function(env/1). | ||||
| :- arithmetic_function(env/2). | ||||
|  | ||||
| env(Name, Value) :- | ||||
| 	(   getenv(Name, Text) | ||||
| 	->  convert_setting_text(number, Text, Value) | ||||
| 	;   existence_error(environment_variable, Name) | ||||
| 	). | ||||
| env(Name, Default, Value) :- | ||||
| 	(   getenv(Name, Text) | ||||
| 	->  convert_setting_text(number, Text, Value) | ||||
| 	;   Value = Default | ||||
| 	). | ||||
|  | ||||
|  | ||||
| %%	numeric_type(+Type, -BaseType) | ||||
| % | ||||
| %	True if Type is a numeric type   and  BaseType is the associated | ||||
| %	basic Prolog type. BaseType is  one   of  =integer=,  =float= or | ||||
| %	=number=. | ||||
|  | ||||
| numeric_type(integer, integer). | ||||
| numeric_type(nonneg, integer). | ||||
| numeric_type(float, float). | ||||
| numeric_type(between(L,_), Type) :- | ||||
| 	( integer(L) -> Type = integer ; Type = float ). | ||||
|  | ||||
|  | ||||
| %%	set_setting(:Name, +Value) is det. | ||||
| % | ||||
| %	Change a setting. Performs existence   and type-checking for the | ||||
| %	setting. If the effective value  of   the  setting is changed it | ||||
| %	broadcasts the event below. | ||||
| % | ||||
| %		settings(changed(Module:Name, Old, New)) | ||||
| % | ||||
| %	@error	existence_error(setting, Name) | ||||
| %	@error  type_error(Type, Value) | ||||
|  | ||||
| set_setting(QName, Value) :- | ||||
| 	strip_module(QName, Module, Name), | ||||
| 	must_be(atom, Name), | ||||
| 	(   curr_setting(Name, Module, Type, Default0, _Comment), | ||||
| 	    eval_default(Default0, Module, Type, Default) | ||||
| 	->  (   Value == Default | ||||
| 	    ->	retract_setting(Module:Name) | ||||
| 	    ;	st_value(Name, Module, Value) | ||||
| 	    ->	true | ||||
| 	    ;	check_type(Type, Value) | ||||
| 	    ->	setting(Module:Name, Old), | ||||
| 	        retract_setting(Module:Name), | ||||
| 	        assert_setting(Module:Name, Value), | ||||
| 		broadcast(settings(changed(Module:Name, Old, Value))), | ||||
| 		clear_setting_cache	% might influence dependent settings. | ||||
| 	    ) | ||||
| 	;   existence_error(setting, Name) | ||||
| 	). | ||||
|  | ||||
| retract_setting(Module:Name) :- | ||||
| 	retractall(st_value(Name, Module, _)). | ||||
|  | ||||
| assert_setting(Module:Name, Value) :- | ||||
| 	assert(st_value(Name, Module, Value)). | ||||
|  | ||||
| %%	restore_setting(:Name) is det. | ||||
| % | ||||
| %	Restore the value of setting Name   to  its default. Broadcast a | ||||
| %	change like set_setting/2 if  the  current   value  is  not  the | ||||
| %	default. | ||||
|  | ||||
| restore_setting(QName) :- | ||||
| 	strip_module(QName, Module, Name), | ||||
| 	must_be(atom, Name), | ||||
| 	(   st_value(Name, Module, Old) | ||||
| 	->  retract_setting(Module:Name), | ||||
| 	    setting(Module:Name, Value), | ||||
| 	    (	Old \== Value | ||||
| 	    ->	broadcast(settings(changed(Module:Name, Old, Value))) | ||||
| 	    ;	true | ||||
| 	    ) | ||||
| 	;   true | ||||
| 	). | ||||
|  | ||||
| %%	set_setting_default(:Name, +Default) is det. | ||||
| % | ||||
| %	Change the default for a setting.  The   effect  is  the same as | ||||
| %	set_setting/2, but the new value is  considered the default when | ||||
| %	saving and restoring  a  setting.  It   is  intended  to  change | ||||
| %	application defaults in a particular context. | ||||
|  | ||||
| set_setting_default(QName, Default) :- | ||||
| 	strip_module(QName, Module, Name), | ||||
| 	must_be(atom, Name), | ||||
| 	(   current_setting(Name, Module, Type, Default0, _Comment, _Src) | ||||
| 	->  retractall(settings:st_default(Name, Module, _)), | ||||
| 	    retract_setting(Module:Name), | ||||
| 	    (   Default == Default0 | ||||
| 	    ->	true | ||||
| 	    ;	assert(settings:st_default(Name, Module, Default)) | ||||
| 	    ), | ||||
| 	    eval_default(Default, Module, Type, Value), | ||||
| 	    set_setting(Module:Name, Value) | ||||
| 	;   existence_error(setting, Module:Name) | ||||
| 	). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	       TYPES		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	check_type(+Type, +Term) | ||||
| % | ||||
| %	Type  checking  for  settings.  Currently  simply  forwarded  to | ||||
| %	must_be/2. | ||||
|  | ||||
| check_type(Type, Term) :- | ||||
| 	must_be(Type, Term). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	       FILE		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	load_settings(File) is det. | ||||
| %%	load_settings(File, +Options) is det. | ||||
| % | ||||
| %	Load local settings from File. Succeeds  if File does not exist, | ||||
| %	setting the default save-file to File.  Options are: | ||||
| % | ||||
| %	  * undefined(+Action) | ||||
| %	  Define how to handle settings that are not defined.  When | ||||
| %	  =error=, an error is printed and the setting is ignored. | ||||
| %	  when =load=, the setting is loaded anyway, waiting for a | ||||
| %	  definition. | ||||
|  | ||||
| load_settings(File) :- | ||||
| 	load_settings(File, []). | ||||
|  | ||||
| load_settings(File, Options) :- | ||||
| 	absolute_file_name(File, Path, | ||||
| 			   [ access(read), | ||||
| 			     file_errors(fail) | ||||
| 			   ]), !, | ||||
| 	assert(local_file(Path)), | ||||
| 	open(Path, read, In, [encoding(utf8)]), | ||||
| 	read_setting(In, T0), | ||||
| 	call_cleanup(load_settings(T0, In, Options), close(In)), | ||||
| 	clear_setting_cache. | ||||
| load_settings(File, _) :- | ||||
| 	absolute_file_name(File, Path, | ||||
| 			   [ access(write), | ||||
| 			     file_errors(fail) | ||||
| 			   ]), !, | ||||
| 	assert(local_file(Path)). | ||||
| load_settings(_, _). | ||||
|  | ||||
| load_settings(end_of_file, _, _) :- !. | ||||
| load_settings(Setting, In, Options) :- | ||||
| 	catch(store_setting(Setting, Options), E, | ||||
| 	      print_message(warning, E)), | ||||
| 	read_setting(In, Next), | ||||
| 	load_settings(Next, In, Options). | ||||
|  | ||||
| read_setting(In, Term) :- | ||||
| 	read_term(In, Term, | ||||
| 		  [ errors(dec10) | ||||
| 		  ]). | ||||
|  | ||||
| %%	store_setting(Term, +Options) | ||||
| % | ||||
| %	Store setting loaded from file in the Prolog database. | ||||
|  | ||||
| store_setting(setting(Module:Name, Value), _) :- | ||||
| 	curr_setting(Name, Module, Type, Default0, _Commentm), !, | ||||
| 	eval_default(Default0, Module, Type, Default), | ||||
| 	(   Value == Default | ||||
| 	->  true | ||||
| 	;   check_type(Type, Value) | ||||
| 	->  retractall(st_value(Name, Module, _)), | ||||
| 	    assert(st_value(Name, Module, Value)), | ||||
| 	    broadcast(settings(changed(Module:Name, Default, Value))) | ||||
| 	). | ||||
| store_setting(setting(Module:Name, Value), Options) :- !, | ||||
| 	(   option(undefined(load), Options, load) | ||||
| 	->  retractall(st_value(Name, Module, _)), | ||||
| 	    assert(st_value(Name, Module, Value)) | ||||
| 	;   existence_error(setting, Module:Name) | ||||
| 	). | ||||
| store_setting(Term, _) :- | ||||
| 	type_error(setting, Term). | ||||
|  | ||||
| %%	save_settings is det. | ||||
| %%	save_settings(+File) is det. | ||||
| % | ||||
| %	Save modified settings to File. | ||||
|  | ||||
| save_settings :- | ||||
| 	local_file(File), !, | ||||
| 	save_settings(File). | ||||
|  | ||||
| save_settings(File) :- | ||||
| 	absolute_file_name(File, Path, | ||||
| 			   [ access(write) | ||||
| 			   ]), !, | ||||
| 	open(Path, write, Out, | ||||
| 	     [ encoding(utf8), | ||||
| 	       bom(true) | ||||
| 	     ]), | ||||
| 	write_setting_header(Out), | ||||
| 	forall(current_setting(Name, Module, _, _, _, _), | ||||
| 	       save_setting(Out, Module:Name)), | ||||
| 	close(Out). | ||||
|  | ||||
|  | ||||
| write_setting_header(Out) :- | ||||
| 	get_time(Now), | ||||
| 	format_time(string(Date), '%+', Now), | ||||
| 	format(Out, '/*  Saved settings~n', []), | ||||
| 	format(Out, '    Date: ~w~n', [Date]), | ||||
| 	format(Out, '*/~n~n', []). | ||||
|  | ||||
| save_setting(Out, Module:Name) :- | ||||
| 	curr_setting(Name, Module, Type, Default, Comment), | ||||
| 	(   st_value(Name, Module, Value), | ||||
| 	    \+ ( eval_default(Default, Module, Type, DefValue), | ||||
| 		 debug(setting, '~w <-> ~w~n', [DefValue, Value]), | ||||
| 	         DefValue =@= Value | ||||
| 	       ) | ||||
| 	->  format(Out, '~n%	~w~n', [Comment]), | ||||
| 	    format(Out, 'setting(~q:~q, ~q).~n', [Module, Name, Value]) | ||||
| 	;   true | ||||
| 	). | ||||
|  | ||||
| %%	current_setting(?Setting) is nondet. | ||||
| % | ||||
| %	True if Setting is a currently defined setting | ||||
|  | ||||
| current_setting(Setting) :- | ||||
| 	ground(Setting), !, | ||||
| 	strip_module(Setting, Module, Name), | ||||
| 	current_setting(Name, Module, _, _, _, _). | ||||
| current_setting(Module:Name) :- | ||||
| 	current_setting(Name, Module, _, _, _, _). | ||||
|  | ||||
| %%	setting_property(+Setting, +Property) is det. | ||||
| %%	setting_property(?Setting, ?Property) is nondet. | ||||
| % | ||||
| %	Query currently defined settings.  Property is one of | ||||
| % | ||||
| %		* comment(-Atom) | ||||
| %		* type(-Type) | ||||
| %		Type of the setting. | ||||
| %		* default(-Default) | ||||
| %		Default value.  If this is an expression, it is | ||||
| %		evaluated. | ||||
|  | ||||
| setting_property(Setting, Property) :- | ||||
| 	ground(Setting), !, | ||||
| 	Setting = Module:Name, | ||||
| 	curr_setting(Name, Module, Type, Default, Comment), !, | ||||
| 	setting_property(Property, Module, Type, Default, Comment). | ||||
| setting_property(Setting, Property) :- | ||||
| 	Setting = Module:Name, | ||||
| 	curr_setting(Name, Module, Type, Default, Comment), | ||||
| 	setting_property(Property, Module, Type, Default, Comment). | ||||
|  | ||||
| setting_property(type(Type),       _, Type, _,        _). | ||||
| setting_property(default(Default), M, Type, Default0, _) :- | ||||
| 	eval_default(Default0, M, Type, Default). | ||||
| setting_property(comment(Comment), _, _,    _,        Comment). | ||||
|  | ||||
| %%	list_settings | ||||
| % | ||||
| %	List settings to =current_output=. | ||||
|  | ||||
| list_settings :- | ||||
| 	format('~`=t~72|~n'), | ||||
| 	format('~w~t~20| ~w~w~t~40| ~w~n', ['Name', 'Value (*=modified)', '', 'Comment']), | ||||
| 	format('~`=t~72|~n'), | ||||
| 	forall(current_setting(Module:Setting), | ||||
| 	       list_setting(Module:Setting)). | ||||
|  | ||||
| list_setting(Module:Name) :- | ||||
| 	curr_setting(Name, Module, Type, Default0, Comment), | ||||
| 	eval_default(Default0, Module, Type, Default), | ||||
| 	setting(Module:Name, Value), | ||||
| 	(   Value \== Default | ||||
| 	->  Modified = (*) | ||||
| 	;   Modified = '' | ||||
| 	), | ||||
|         format('~w~t~20| ~q~w~t~40| ~w~n', [Module:Name, Value, Modified, Comment]). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	      TYPES		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	convert_setting_text(+Type, +Text, -Value) | ||||
| % | ||||
| %	Converts from textual form to  Prolog   Value.  Used  to convert | ||||
| %	values obtained from the environment.  Public to provide support | ||||
| %	in user-interfaces to this library. | ||||
| % | ||||
| %	@error	type_error(Type, Value) | ||||
|  | ||||
| :- multifile | ||||
| 	convert_text/3.			% +Type, +Text, -Value | ||||
|  | ||||
| convert_setting_text(Type, Text, Value) :- | ||||
| 	convert_text(Type, Text, Value), !. | ||||
| convert_setting_text(atom, Value, Value) :- !, | ||||
| 	must_be(atom, Value). | ||||
| convert_setting_text(boolean, Value, Value) :- !, | ||||
| 	must_be(boolean, Value). | ||||
| convert_setting_text(integer, Atom, Number) :- !, | ||||
| 	term_to_atom(Term, Atom), | ||||
| 	Number is round(Term). | ||||
| convert_setting_text(float, Atom, Number) :- !, | ||||
| 	term_to_atom(Term, Atom), | ||||
| 	Number is float(Term). | ||||
| convert_setting_text(between(L,U), Atom, Number) :- !, | ||||
| 	(   integer(L) | ||||
| 	->  convert_setting_text(integer, Atom, Number) | ||||
| 	;   convert_setting_text(float, Atom, Number) | ||||
| 	), | ||||
| 	must_be(between(L,U), Number). | ||||
| convert_setting_text(Type, Atom, Term) :- | ||||
| 	term_to_atom(Term, Atom), | ||||
| 	must_be(Type, Term). | ||||
|  | ||||
|  | ||||
							
								
								
									
										1048
									
								
								LGPL/url.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1048
									
								
								LGPL/url.pl
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										134
									
								
								LGPL/utf8.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										134
									
								
								LGPL/utf8.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,134 @@ | ||||
| /*  $Id$ | ||||
|  | ||||
|     Part of SWI-Prolog | ||||
|  | ||||
|     Author:        Jan Wielemaker | ||||
|     E-mail:        wielemak@science.uva.nl | ||||
|     WWW:           http://www.swi-prolog.org | ||||
|     Copyright (C): 1985-2005, University of Amsterdam | ||||
|  | ||||
|     This program is free software; you can redistribute it and/or | ||||
|     modify it under the terms of the GNU General Public License | ||||
|     as published by the Free Software Foundation; either version 2 | ||||
|     of the License, or (at your option) any later version. | ||||
|  | ||||
|     This program is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public | ||||
|     License along with this library; if not, write to the Free Software | ||||
|     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  | ||||
|     As a special exception, if you link this library with other files, | ||||
|     compiled with a Free Software compiler, to produce an executable, this | ||||
|     library does not by itself cause the resulting executable to be covered | ||||
|     by the GNU General Public License. This exception does not however | ||||
|     invalidate any other reasons why the executable file might be covered by | ||||
|     the GNU General Public License. | ||||
| */ | ||||
|  | ||||
| :- module(utf8, | ||||
| 	  [ utf8_codes//1		% ?String | ||||
| 	  ]). | ||||
|  | ||||
| %%	utf8_codes(?Codes)// is det. | ||||
| % | ||||
| %	DCG translating between  a  Unicode   code-list  and  its  UTF-8 | ||||
| %	encoded  byte-string.  The  DCG  works   two  ways.  Encoding  a | ||||
| %	code-list to a UTF-8 byte string is achieved using | ||||
| % | ||||
| %		phrase(utf8_codes(Codes), UTF8) | ||||
| % | ||||
| %	The  algorithm  is  a  close  copy    of  the  C-algorithm  used | ||||
| %	internally and defined in src/pl-utf8.c | ||||
| % | ||||
| %	NOTE: in many  cases  you  can   avoid  this  library  and leave | ||||
| %	encoding and decoding to I/O streams. If   only part of the data | ||||
| %	is to be encoded the  encoding  of   a  stream  can  be switched | ||||
| %	temporary using set_stream(Stream, encoding(utf8)) | ||||
|  | ||||
| utf8_codes([H|T]) --> | ||||
| 	utf8_code(H), !, | ||||
| 	utf8_codes(T). | ||||
| utf8_codes([]) --> | ||||
| 	[]. | ||||
|  | ||||
| utf8_code(C) --> | ||||
| 	[C0], | ||||
| 	{ nonvar(C0) }, !,		% decoding version | ||||
| 	(   {C0 < 0x80} | ||||
| 	->  {C = C0} | ||||
| 	;   {C0/\0xe0 =:= 0xc0} | ||||
| 	->  utf8_cont(C1, 0), | ||||
| 	    {C is (C0/\0x1f)<<6\/C1} | ||||
| 	;   {C0/\0xf0 =:= 0xe0} | ||||
| 	->  utf8_cont(C1, 6), | ||||
| 	    utf8_cont(C2, 0), | ||||
| 	    {C is ((C0/\0xf)<<12)\/C1\/C2} | ||||
| 	;   {C0/\0xf8 =:= 0xf0} | ||||
| 	->  utf8_cont(C1, 12), | ||||
| 	    utf8_cont(C2, 6), | ||||
| 	    utf8_cont(C3, 0), | ||||
| 	    {C is ((C0/\0x7)<<18)\/C1\/C2\/C3} | ||||
| 	;   {C0/\0xfc =:= 0xf8} | ||||
| 	->  utf8_cont(C1, 18), | ||||
| 	    utf8_cont(C2, 12), | ||||
| 	    utf8_cont(C3, 6), | ||||
| 	    utf8_cont(C4, 0), | ||||
| 	    {C is ((C0/\0x3)<<24)\/C1\/C2\/C3\/C4} | ||||
| 	;   {C0/\0xfe =:= 0xfc} | ||||
| 	->  utf8_cont(C1, 24), | ||||
| 	    utf8_cont(C2, 18), | ||||
| 	    utf8_cont(C3, 12), | ||||
| 	    utf8_cont(C4, 6), | ||||
| 	    utf8_cont(C5, 0), | ||||
| 	    {C is ((C0/\0x1)<<30)\/C1\/C2\/C3\/C4\/C5} | ||||
| 	). | ||||
| utf8_code(C) --> | ||||
| 	{ nonvar(C) }, !,		% encoding version | ||||
| 	(   { C < 0x80 } | ||||
| 	->  [C] | ||||
| 	;   { C < 0x800 } | ||||
| 	->  { C0 is 0xc0\/((C>>6)/\0x1f), | ||||
| 	      C1 is 0x80\/(C/\0x3f) | ||||
| 	    }, | ||||
| 	    [C0,C1] | ||||
| 	;   { C < 0x10000 } | ||||
| 	->  { C0 is 0xe0\/((C>>12)/\0x0f), | ||||
| 	      C1 is 0x80\/((C>>6)/\0x3f), | ||||
| 	      C2 is 0x80\/(C/\0x3f) | ||||
| 	    }, | ||||
| 	    [C0,C1,C2] | ||||
| 	;   { C < 0x200000 } | ||||
| 	->  { C0 is 0xf0\/((C>>18)/\0x07), | ||||
| 	      C1 is 0x80\/((C>>12)/\0x3f), | ||||
| 	      C2 is 0x80\/((C>>6)/\0x3f), | ||||
| 	      C3 is 0x80\/(C/\0x3f) | ||||
| 	    }, | ||||
| 	    [C0,C1,C2,C3] | ||||
| 	;   { C < 0x4000000 } | ||||
| 	->  { C0 is 0xf8\/((C>>24)/\0x03), | ||||
| 	      C1 is 0x80\/((C>>18)/\0x3f), | ||||
| 	      C2 is 0x80\/((C>>12)/\0x3f), | ||||
| 	      C3 is 0x80\/((C>>6)/\0x3f), | ||||
| 	      C4 is 0x80\/(C/\0x3f) | ||||
| 	    }, | ||||
| 	    [C0,C1,C2,C3,C4] | ||||
| 	;   { C < 0x80000000 } | ||||
| 	->  { C0 is 0xfc\/((C>>30)/\0x01), | ||||
| 	      C1 is 0x80\/((C>>24)/\0x3f), | ||||
| 	      C2 is 0x80\/((C>>18)/\0x3f), | ||||
| 	      C3 is 0x80\/((C>>12)/\0x3f), | ||||
| 	      C4 is 0x80\/((C>>6)/\0x3f), | ||||
| 	      C5 is 0x80\/(C/\0x3f) | ||||
| 	    }, | ||||
| 	    [C0,C1,C2,C3,C4,C5] | ||||
| 	). | ||||
|  | ||||
| utf8_cont(Val, Shift) --> | ||||
| 	[C], | ||||
| 	{ C/\0xc0 =:= 0x80, | ||||
| 	  Val is (C/\0x3f)<<Shift | ||||
| 	}. | ||||
		Reference in New Issue
	
	Block a user