2003-08-30 12:47:54 -04:00
|
|
|
/* load.c
|
|
|
|
*
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright 1990, 1991, 1992, 1993, 1994, 1995, Oliver Laumann, Berlin
|
|
|
|
* Copyright 2002, 2003 Sam Hocevar <sam@zoy.org>, Paris
|
|
|
|
*
|
|
|
|
* This software was derived from Elk 1.2, which was Copyright 1987, 1988,
|
|
|
|
* 1989, Nixdorf Computer AG and TELES GmbH, Berlin (Elk 1.2 has been written
|
|
|
|
* by Oliver Laumann for TELES Telematic Services, Berlin, in a joint project
|
|
|
|
* between TELES and Nixdorf Microprocessor Engineering, Berlin).
|
|
|
|
*
|
|
|
|
* Oliver Laumann, TELES GmbH, Nixdorf Computer AG and Sam Hocevar, as co-
|
|
|
|
* owners or individual owners of copyright in this software, grant to any
|
|
|
|
* person or company a worldwide, royalty free, license to
|
|
|
|
*
|
|
|
|
* i) copy this software,
|
|
|
|
* ii) prepare derivative works based on this software,
|
|
|
|
* iii) distribute copies of this software or derivative works,
|
|
|
|
* iv) perform this software, or
|
|
|
|
* v) display this software,
|
|
|
|
*
|
|
|
|
* provided that this notice is not removed and that neither Oliver Laumann
|
|
|
|
* nor Teles nor Nixdorf are deemed to have made any representations as to
|
|
|
|
* the suitability of this software for any purpose nor are held responsible
|
|
|
|
* for any defects of this software.
|
|
|
|
*
|
|
|
|
* THERE IS ABSOLUTELY NO WARRANTY FOR THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2003-09-06 07:25:29 -04:00
|
|
|
#include "config.h"
|
|
|
|
|
2003-09-07 05:55:56 -04:00
|
|
|
#include <string.h>
|
|
|
|
|
2003-08-19 15:19:38 -04:00
|
|
|
#include "kernel.h"
|
|
|
|
|
|
|
|
Object V_Load_Path, V_Load_Noisilyp, V_Load_Libraries;
|
|
|
|
|
|
|
|
char *Loader_Input; /* tmp file name used by load.xx.c */
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
extern void Switch_Environment (Object);
|
2003-09-07 05:55:56 -04:00
|
|
|
#ifdef CAN_LOAD_LIB
|
|
|
|
extern void Load_Library (Object libs);
|
2003-08-19 15:19:38 -04:00
|
|
|
#endif
|
2003-09-07 05:55:56 -04:00
|
|
|
void Load_Source (Object);
|
2003-08-19 15:19:38 -04:00
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
void Init_Load () {
|
2003-08-19 15:19:38 -04:00
|
|
|
Define_Variable (&V_Load_Path, "load-path",
|
2003-09-02 04:12:11 -04:00
|
|
|
Cons (Make_String (".", 1),
|
|
|
|
Cons (Make_String (SCM_DIR, sizeof (SCM_DIR) - 1),
|
|
|
|
Cons (Make_String (OBJ_DIR, sizeof (OBJ_DIR) - 1), Null))));
|
2003-08-19 15:19:38 -04:00
|
|
|
Define_Variable (&V_Load_Noisilyp, "load-noisily?", False);
|
2003-09-06 21:33:17 -04:00
|
|
|
Define_Variable (&V_Load_Libraries, "load-libraries", Make_String ("", 0));
|
2003-08-19 15:19:38 -04:00
|
|
|
}
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
void Init_Loadpath (char *s) { /* No GC possible here */
|
2003-08-19 15:19:38 -04:00
|
|
|
register char *p;
|
|
|
|
Object path;
|
2003-08-19 15:24:23 -04:00
|
|
|
|
2003-08-19 15:19:38 -04:00
|
|
|
path = Null;
|
|
|
|
if (s[0] == '\0')
|
2003-09-02 04:12:11 -04:00
|
|
|
return;
|
2003-08-19 15:19:38 -04:00
|
|
|
while (1) {
|
2003-09-02 04:12:11 -04:00
|
|
|
for (p = s; *p && *p != ':'; p++)
|
|
|
|
;
|
|
|
|
path = Cons (Make_String (s, p-s), path);
|
|
|
|
if (*p == '\0')
|
|
|
|
break;
|
|
|
|
s = ++p;
|
2003-08-19 15:19:38 -04:00
|
|
|
}
|
|
|
|
Var_Set (V_Load_Path, P_Reverse (path));
|
|
|
|
}
|
|
|
|
|
2003-08-19 15:25:03 -04:00
|
|
|
int Has_Suffix (Object name, char const *suffix) {
|
2003-08-19 15:19:38 -04:00
|
|
|
register char *p;
|
2003-08-19 15:25:03 -04:00
|
|
|
register int len = strlen(suffix);
|
2003-08-19 15:19:38 -04:00
|
|
|
register struct S_String *str;
|
|
|
|
|
|
|
|
if (TYPE(name) == T_Symbol)
|
2003-09-02 04:12:11 -04:00
|
|
|
name = SYMBOL(name)->name;
|
2003-08-19 15:19:38 -04:00
|
|
|
str = STRING(name);
|
2003-08-19 15:25:03 -04:00
|
|
|
p = str->data + str->size - len;
|
|
|
|
return len <= str->size && !strncasecmp(p, suffix, len);
|
2003-08-19 15:19:38 -04:00
|
|
|
}
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
void Check_Loadarg (Object x) {
|
2003-08-19 15:19:38 -04:00
|
|
|
Object tail;
|
2003-08-19 15:24:23 -04:00
|
|
|
register int t = TYPE(x);
|
2003-08-19 15:19:38 -04:00
|
|
|
|
|
|
|
if (t == T_Symbol || t == T_String)
|
2003-09-02 04:12:11 -04:00
|
|
|
return;
|
2003-08-19 15:19:38 -04:00
|
|
|
if (t != T_Pair)
|
2003-09-02 04:12:11 -04:00
|
|
|
Wrong_Type_Combination (x, "string, symbol, or list");
|
2003-08-19 15:19:38 -04:00
|
|
|
for (tail = x; !Nullp (tail); tail = Cdr (tail)) {
|
2003-09-02 04:12:11 -04:00
|
|
|
Object f;
|
2003-08-19 15:24:23 -04:00
|
|
|
|
2003-09-02 04:12:11 -04:00
|
|
|
f = Car (tail);
|
|
|
|
if (TYPE(f) != T_Symbol && TYPE(f) != T_String)
|
|
|
|
Wrong_Type_Combination (f, "string or symbol");
|
2003-09-07 05:55:56 -04:00
|
|
|
if (!Has_Suffix (f, ".so"))
|
2003-09-02 04:12:11 -04:00
|
|
|
Primitive_Error ("~s: not an object file", f);
|
2003-08-19 15:19:38 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
Object General_Load (Object what, Object env) {
|
2003-08-19 15:19:38 -04:00
|
|
|
Object oldenv;
|
|
|
|
GC_Node;
|
|
|
|
|
|
|
|
Check_Type (env, T_Environment);
|
|
|
|
oldenv = The_Environment;
|
|
|
|
GC_Link (oldenv);
|
|
|
|
Switch_Environment (env);
|
|
|
|
Check_Loadarg (what);
|
2003-08-19 15:25:03 -04:00
|
|
|
if (TYPE(what) == T_Pair) {
|
2003-09-06 21:33:17 -04:00
|
|
|
if (Has_Suffix (Car (what), ".so"))
|
2003-08-19 15:25:03 -04:00
|
|
|
#ifdef CAN_LOAD_LIB
|
2003-09-02 04:12:11 -04:00
|
|
|
Load_Library (what)
|
2003-08-19 15:25:03 -04:00
|
|
|
#endif
|
2003-09-02 04:12:11 -04:00
|
|
|
;
|
2003-08-19 15:25:03 -04:00
|
|
|
}
|
|
|
|
else if (Has_Suffix (what, ".so"))
|
|
|
|
#ifdef CAN_LOAD_LIB
|
2003-09-02 04:12:11 -04:00
|
|
|
Load_Library (Cons (what, Null))
|
2003-08-19 15:19:38 -04:00
|
|
|
#endif
|
2003-09-02 04:12:11 -04:00
|
|
|
;
|
2003-08-19 15:19:38 -04:00
|
|
|
else
|
2003-09-02 04:12:11 -04:00
|
|
|
Load_Source (what);
|
2003-08-19 15:19:38 -04:00
|
|
|
Switch_Environment (oldenv);
|
|
|
|
GC_Unlink;
|
|
|
|
return Void;
|
|
|
|
}
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
Object P_Load (int argc, Object *argv) {
|
2003-08-19 15:19:38 -04:00
|
|
|
return General_Load (argv[0], argc == 1 ? The_Environment : argv[1]);
|
|
|
|
}
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
void Load_Source_Port (Object port) {
|
2003-08-19 15:19:38 -04:00
|
|
|
Object val;
|
|
|
|
GC_Node;
|
|
|
|
TC_Prolog;
|
|
|
|
|
|
|
|
GC_Link (port);
|
|
|
|
while (1) {
|
2003-09-02 04:12:11 -04:00
|
|
|
val = General_Read (port, 1);
|
|
|
|
if (TYPE(val) == T_End_Of_File)
|
|
|
|
break;
|
|
|
|
TC_Disable;
|
|
|
|
val = Eval (val);
|
|
|
|
TC_Enable;
|
|
|
|
if (Var_Is_True (V_Load_Noisilyp)) {
|
|
|
|
Print (val);
|
|
|
|
(void)P_Newline (0, (Object *)0);
|
|
|
|
}
|
2003-08-19 15:19:38 -04:00
|
|
|
}
|
|
|
|
GC_Unlink;
|
|
|
|
}
|
|
|
|
|
2003-08-19 15:24:23 -04:00
|
|
|
void Load_Source (Object name) {
|
2003-08-19 15:19:38 -04:00
|
|
|
Object port;
|
|
|
|
GC_Node;
|
|
|
|
|
|
|
|
port = General_Open_File (name, P_INPUT, Var_Get (V_Load_Path));
|
|
|
|
GC_Link (port);
|
|
|
|
Load_Source_Port (port);
|
|
|
|
(void)P_Close_Input_Port (port);
|
|
|
|
GC_Unlink;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Interface to P_Load() for use by applications.
|
|
|
|
*/
|
2003-08-19 15:24:23 -04:00
|
|
|
void Load_File (char *name) {
|
2003-08-19 15:19:38 -04:00
|
|
|
Object arg;
|
|
|
|
|
|
|
|
arg = Make_String(name, strlen(name));
|
|
|
|
(void)P_Load(1, &arg);
|
|
|
|
}
|