428
|
1 /* emodules.c - Support routines for dynamic module loading
|
|
2 (C) Copyright 1998, 1999 J. Kean Johnston. All rights reserved.
|
|
3
|
|
4 This file is part of XEmacs.
|
|
5
|
|
6 XEmacs is free software; you can redistribute it and/or modify it
|
|
7 under the terms of the GNU General Public License as published by the
|
|
8 Free Software Foundation; either version 2, or (at your option) any
|
|
9 later version.
|
|
10
|
|
11 XEmacs is distributed in the hope that it will be useful, but WITHOUT
|
|
12 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
14 for more details.
|
|
15
|
|
16 You should have received a copy of the GNU General Public License
|
|
17 along with XEmacs; see the file COPYING. If not, write to
|
|
18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
19 Boston, MA 02111-1307, USA. */
|
|
20
|
|
21 #include "emodules.h"
|
|
22 #include "sysdll.h"
|
2078
|
23 #ifdef HAVE_LTDL
|
|
24 #include <ltdl.h>
|
|
25 #endif
|
428
|
26
|
1750
|
27 /* Load path */
|
|
28 static Lisp_Object Vmodule_load_path;
|
|
29
|
|
30 /* Module lFile extensions */
|
|
31 static Lisp_Object Vmodule_extensions;
|
|
32
|
428
|
33 #ifdef HAVE_SHLIB
|
|
34
|
|
35 /* CE-Emacs version number */
|
|
36 Lisp_Object Vmodule_version;
|
|
37
|
|
38 /* Do we do our work quietly? */
|
|
39 int load_modules_quietly;
|
|
40
|
996
|
41 /* Set this while unloading a module. This should NOT be made set by users,
|
|
42 as it allows the unbinding of symbol-value-forward variables. */
|
|
43 int unloading_module;
|
|
44
|
564
|
45 Lisp_Object Qdll_error;
|
996
|
46 Lisp_Object Qmodule, Qunload_module, module_tag;
|
564
|
47
|
428
|
48 typedef struct _emodules_list
|
|
49 {
|
996
|
50 int used; /* Is this slot used? */
|
1706
|
51 CIbyte *soname; /* Name of the shared object loaded (full path) */
|
|
52 CIbyte *modname; /* The name of the module */
|
|
53 CIbyte *modver; /* The module version string */
|
|
54 CIbyte *modtitle; /* How the module announces itself */
|
996
|
55 void (*unload)(void); /* Module cleanup function to run before unloading */
|
|
56 dll_handle dlhandle; /* Dynamic lib handle */
|
428
|
57 } emodules_list;
|
|
58
|
|
59 static int emodules_depth;
|
|
60 static dll_handle dlhandle;
|
|
61 static emodules_list *modules;
|
|
62 static int modnum;
|
|
63
|
1706
|
64 static int find_make_module (const CIbyte *mod, const CIbyte *name,
|
|
65 const CIbyte *ver, int make_or_find);
|
428
|
66 static Lisp_Object module_load_unwind (Lisp_Object);
|
|
67 static void attempt_module_delete (int mod);
|
|
68
|
|
69 DEFUN ("load-module", Fload_module, 1, 3, "FLoad dynamic module: ", /*
|
|
70 Load in a C Emacs Extension module named FILE.
|
|
71 The optional NAME and VERSION are used to identify specific modules.
|
|
72
|
996
|
73 DO NOT USE THIS FUNCTION in your programs. Use `require' instead.
|
|
74
|
428
|
75 This function is similar in intent to `load' except that it loads in
|
|
76 pre-compiled C or C++ code, using dynamic shared objects. If NAME is
|
|
77 specified, then the module is only loaded if its internal name matches
|
|
78 the NAME specified. If VERSION is specified, then the module is only
|
|
79 loaded if it matches that VERSION. This function will check to make
|
|
80 sure that the same module is not loaded twice. Modules are searched
|
|
81 for in the same way as Lisp files, except that the valid file
|
1632
|
82 extensions are `.so', `.dll', `.ell', or `.dylib'.
|
428
|
83
|
|
84 All symbols in the shared module must be completely resolved in order
|
|
85 for this function to be successful. Any modules which the specified
|
|
86 FILE depends on will be automatically loaded. You can determine which
|
|
87 modules have been loaded as dynamic shared objects by examining the
|
|
88 return value of the function `list-modules'.
|
|
89
|
996
|
90 It is possible, although unwise, to unload modules using `unload-feature'.
|
442
|
91 The preferred mechanism for unloading or reloading modules is to quit
|
428
|
92 XEmacs, and then reload those new or changed modules that are required.
|
|
93
|
|
94 Messages informing you of the progress of the load are displayed unless
|
|
95 the variable `load-modules-quietly' is non-NIL.
|
|
96 */
|
444
|
97 (file, name, version))
|
428
|
98 {
|
1706
|
99 CIbyte *mod, *mname, *mver;
|
428
|
100 int speccount = specpdl_depth();
|
|
101
|
|
102 CHECK_STRING(file);
|
|
103
|
1706
|
104 mod = (CIbyte *) XSTRING_DATA (file);
|
428
|
105
|
|
106 if (NILP (name))
|
|
107 mname = "";
|
|
108 else
|
1706
|
109 mname = (CIbyte *) XSTRING_DATA (name);
|
428
|
110
|
|
111 if (NILP (version))
|
|
112 mver = "";
|
|
113 else
|
1706
|
114 mver = (CIbyte *) XSTRING_DATA (version);
|
428
|
115
|
|
116 dlhandle = 0;
|
|
117 record_unwind_protect (module_load_unwind, make_int(modnum));
|
|
118 emodules_load (mod, mname, mver);
|
771
|
119 unbind_to (speccount);
|
428
|
120
|
|
121 return Qt;
|
|
122 }
|
|
123
|
996
|
124 DEFUN ("unload-module", Funload_module, 1, 3, 0, /*
|
|
125 Unload a module previously loaded with load-module.
|
428
|
126
|
996
|
127 DO NOT USE THIS FUNCTION in your programs. Use `unload-feature' instead.
|
428
|
128
|
|
129 As with load-module, this function requires at least the module FILE, and
|
|
130 optionally the module NAME and VERSION to unload. It may not be possible
|
|
131 for the module to be unloaded from memory, as there may be Lisp objects
|
442
|
132 referring to variables inside the module code. However, once you have
|
428
|
133 requested a module to be unloaded, it will be unloaded from memory as
|
|
134 soon as the last reference to symbols within the module is destroyed.
|
|
135 */
|
444
|
136 (file, name, version))
|
428
|
137 {
|
|
138 int x;
|
1706
|
139 CIbyte *mod, *mname, *mver;
|
996
|
140 Lisp_Object foundname = Qnil;
|
|
141 struct gcpro gcpro1;
|
428
|
142
|
|
143 CHECK_STRING(file);
|
|
144
|
996
|
145 GCPRO1 (foundname);
|
|
146 if (locate_file (Vmodule_load_path, file, Vmodule_extensions, &foundname, 0)
|
|
147 < 0)
|
|
148 return Qt;
|
1706
|
149 mod = (CIbyte *) XSTRING_DATA (foundname);
|
996
|
150 UNGCPRO;
|
428
|
151
|
|
152 if (NILP (name))
|
|
153 mname = "";
|
|
154 else
|
1706
|
155 mname = (CIbyte *) XSTRING_DATA (name);
|
428
|
156
|
|
157 if (NILP (version))
|
|
158 mver = "";
|
|
159 else
|
1706
|
160 mver = (CIbyte *) XSTRING_DATA (version);
|
428
|
161
|
|
162 x = find_make_module (mod, mname, mver, 1);
|
|
163 if (x != -1)
|
996
|
164 {
|
|
165 if (modules[x].unload != NULL)
|
|
166 modules[x].unload ();
|
|
167 attempt_module_delete (x);
|
|
168 }
|
428
|
169 return Qt;
|
|
170 }
|
|
171
|
|
172 DEFUN ("list-modules", Flist_modules, 0, 0, "", /*
|
|
173 Produce a list of loaded dynamic modules.
|
|
174
|
|
175 This function will return a list of all the loaded dynamic modules.
|
|
176 Each element in the list is a list in the form (SONAME NAME VER DESC),
|
|
177 where SONAME is the name of the shared object that was loaded, NAME
|
|
178 is the internal module name, VER is the version of the module, and DESC
|
|
179 is how the module describes itself.
|
|
180
|
|
181 This function returns a list, so you will need to assign the return value
|
|
182 to a variable and then examine the variable with `describe-variable'.
|
|
183 For example:
|
|
184
|
|
185 (setq mylist (list-modules))
|
|
186 (describe-variable 'mylist)
|
|
187
|
|
188
|
|
189 NOTE: It is possible for the same module to be loaded more than once,
|
|
190 at different versions. However, you should never see the same module,
|
|
191 with the same name and version, loaded more than once. If you do, this
|
|
192 is a bug, and you are encouraged to report it.
|
|
193 */
|
|
194 ())
|
|
195 {
|
|
196 Lisp_Object mlist = Qnil;
|
|
197 int i;
|
|
198
|
|
199 for (i = 0; i < modnum; i++)
|
|
200 {
|
|
201 if (modules[i].used == 1)
|
|
202 mlist = Fcons (list4 (build_string (modules[i].soname),
|
|
203 build_string (modules[i].modname),
|
|
204 build_string (modules[i].modver),
|
|
205 build_string (modules[i].modtitle)), mlist);
|
|
206 }
|
|
207
|
|
208 return mlist;
|
|
209 }
|
|
210
|
|
211 static int
|
1706
|
212 find_make_module (const CIbyte *mod, const CIbyte *name, const CIbyte *ver,
|
|
213 int mof)
|
428
|
214 {
|
|
215 int i, fs = -1;
|
|
216
|
|
217 for (i = 0; i < modnum; i++)
|
|
218 {
|
|
219 if (fs == -1 && modules[i].used == 0)
|
|
220 fs = i;
|
|
221 if (strcmp (modules[i].soname, mod) == 0)
|
|
222 {
|
|
223 if (name && name[0] && strcmp (modules[i].modname, name))
|
|
224 continue;
|
|
225 if (ver && ver[0] && strcmp (modules[i].modver, ver))
|
|
226 continue;
|
|
227 return i; /* Found a match */
|
|
228 }
|
|
229 }
|
|
230
|
|
231 if (mof)
|
|
232 return fs;
|
|
233
|
|
234 if (fs != -1)
|
|
235 return fs; /* First free slot */
|
|
236
|
|
237 /*
|
442
|
238 * We only get here if we haven't found a free slot and the module was
|
428
|
239 * not previously loaded.
|
|
240 */
|
1706
|
241 if (modules == NULL)
|
2367
|
242 modules = xnew (emodules_list);
|
428
|
243 modnum++;
|
2367
|
244 XREALLOC_ARRAY (modules, emodules_list, modnum);
|
428
|
245
|
|
246 fs = modnum - 1;
|
2367
|
247 memset (&modules[fs], 0, sizeof (emodules_list));
|
428
|
248 return fs;
|
|
249 }
|
|
250
|
|
251 static void
|
|
252 attempt_module_delete (int mod)
|
|
253 {
|
|
254 if (dll_close (modules[mod].dlhandle) == 0)
|
|
255 {
|
1726
|
256 xfree (modules[mod].soname, CIbyte *);
|
|
257 xfree (modules[mod].modname, CIbyte *);
|
|
258 xfree (modules[mod].modver, CIbyte *);
|
|
259 xfree (modules[mod].modtitle, CIbyte *);
|
428
|
260 modules[mod].dlhandle = 0;
|
|
261 modules[mod].used = 0;
|
|
262 }
|
|
263 else if (modules[mod].used > 1)
|
|
264 modules[mod].used = 1; /* We couldn't delete it - it stays */
|
|
265 }
|
|
266
|
|
267 static Lisp_Object
|
|
268 module_load_unwind (Lisp_Object upto)
|
|
269 {
|
|
270 int x,l=0;
|
|
271
|
|
272 /*
|
|
273 * First close off the current handle if it is open.
|
|
274 */
|
|
275 if (dlhandle != 0)
|
|
276 dll_close (dlhandle);
|
|
277 dlhandle = 0;
|
|
278
|
|
279 if (CONSP (upto))
|
|
280 {
|
|
281 if (INTP (XCAR (upto)))
|
|
282 l = XINT (XCAR (upto));
|
853
|
283 free_cons (upto);
|
428
|
284 }
|
|
285 else
|
|
286 l = XINT (upto);
|
|
287
|
|
288 /*
|
|
289 * Here we need to go through and dlclose() (IN REVERSE ORDER!) any
|
|
290 * modules that were loaded as part of this load chain. We only mark
|
|
291 * the slots as closed if the dlclose() succeeds.
|
|
292 */
|
|
293 for (x = modnum-1; x >= l; x--)
|
|
294 {
|
|
295 if (modules[x].used > 1)
|
|
296 attempt_module_delete (x);
|
|
297 }
|
|
298 emodules_depth = 0;
|
|
299
|
|
300 return Qnil;
|
|
301 }
|
|
302
|
|
303 /*
|
|
304 * Do the actual grunt-work of loading in a module. We first try and
|
|
305 * dlopen() the module. If that fails, we have an error and we bail
|
|
306 * out immediately. If the dlopen() succeeds, we need to check for the
|
442
|
307 * existence of certain special symbols.
|
428
|
308 *
|
|
309 * All modules will have complete access to the variables and functions
|
|
310 * defined within XEmacs itself. It is up to the module to declare any
|
|
311 * variables or functions it uses, however. Modules will also have access
|
|
312 * to other functions and variables in other loaded modules, unless they
|
|
313 * are defined as STATIC.
|
|
314 *
|
|
315 * We need to be very careful with how we load modules. If we encounter an
|
|
316 * error along the way, we need to back out completely to the point at
|
442
|
317 * which the user started. Since we can be called recursively, we need to
|
428
|
318 * take care with marking modules as loaded. When we first start loading
|
|
319 * modules, we set the counter to zero. As we enter the function each time,
|
442
|
320 * we increment the counter, and before we leave we decrement it. When
|
428
|
321 * we get back down to 0, we know we are at the end of the chain and we
|
|
322 * can mark all the modules in the list as loaded.
|
|
323 *
|
|
324 * When we signal an error, we need to be sure to unwind all modules loaded
|
|
325 * thus far (but only for this module chain). It is assumed that if any
|
|
326 * modules in a chain fail, then they all do. This is logical, considering
|
442
|
327 * that the only time we recurse is when we have dependent modules. So in
|
428
|
328 * the error handler we take great care to close off the module chain before
|
|
329 * we call "error" and let the Fmodule_load unwind_protect() function handle
|
|
330 * the cleaning up.
|
|
331 */
|
|
332 void
|
1706
|
333 emodules_load (const CIbyte *module, const CIbyte *modname,
|
|
334 const CIbyte *modver)
|
428
|
335 {
|
2367
|
336 /* !!#### Needs serious work */
|
996
|
337 Lisp_Object old_load_list;
|
428
|
338 Lisp_Object filename;
|
996
|
339 Lisp_Object foundname, lisp_modname;
|
|
340 int x, mpx;
|
1706
|
341 CIbyte *soname;
|
|
342 const CIbyte **f;
|
442
|
343 const long *ellcc_rev;
|
1706
|
344 CIbyte *mver, *mname, *mtitle, *symname;
|
428
|
345 void (*modload)(void) = 0;
|
|
346 void (*modsyms)(void) = 0;
|
|
347 void (*modvars)(void) = 0;
|
|
348 void (*moddocs)(void) = 0;
|
996
|
349 void (*modunld)(void) = 0;
|
428
|
350 emodules_list *mp;
|
996
|
351 struct gcpro gcpro1, gcpro2, gcpro3, gcpro4;
|
428
|
352
|
|
353 filename = Qnil;
|
|
354 foundname = Qnil;
|
|
355
|
|
356 emodules_depth++;
|
|
357 dlhandle = 0;
|
|
358
|
1706
|
359 if (module == NULL || module[0] == '\0')
|
563
|
360 invalid_argument ("Empty module name", Qunbound);
|
428
|
361
|
996
|
362 GCPRO4(filename, foundname, old_load_list, lisp_modname);
|
|
363 filename = build_string (module);
|
|
364 if (locate_file (Vmodule_load_path, filename, Vmodule_extensions,
|
|
365 &foundname, 0) < 0)
|
563
|
366 signal_error (Qdll_error, "Cannot open dynamic module", filename);
|
428
|
367
|
1706
|
368 LISP_STRING_TO_EXTERNAL (foundname, soname, Qfile_name);
|
996
|
369 lisp_modname = call1 (Qfile_name_sans_extension,
|
|
370 Ffile_name_nondirectory (foundname));
|
428
|
371
|
1706
|
372 dlhandle = dll_open (foundname);
|
|
373 if (dlhandle == NULL)
|
563
|
374 {
|
1811
|
375 signal_error (Qdll_error, "Opening dynamic module", dll_error ());
|
563
|
376 }
|
428
|
377
|
442
|
378 ellcc_rev = (const long *)dll_variable (dlhandle, "emodule_compiler");
|
1706
|
379 if (ellcc_rev == NULL || *ellcc_rev <= 0L)
|
563
|
380 signal_error (Qdll_error, "Invalid dynamic module: Missing symbol `emodule_compiler'", Qunbound);
|
428
|
381 if (*ellcc_rev > EMODULES_REVISION)
|
563
|
382 signal_ferror (Qdll_error, "Invalid dynamic module: Unsupported version `%ld(%ld)'", *ellcc_rev, EMODULES_REVISION);
|
428
|
383
|
1706
|
384 f = (const CIbyte **) dll_variable (dlhandle, "emodule_name");
|
|
385 if (f == NULL || *f == NULL)
|
563
|
386 signal_error (Qdll_error, "Invalid dynamic module: Missing symbol `emodule_name'", Qunbound);
|
428
|
387
|
1706
|
388 mname = (CIbyte *) ALLOCA (strlen (*f) + 1);
|
428
|
389 strcpy (mname, *f);
|
|
390 if (mname[0] == '\0')
|
563
|
391 signal_error (Qdll_error, "Invalid dynamic module: Empty value for `emodule_name'", Qunbound);
|
428
|
392
|
1706
|
393 f = (const CIbyte **) dll_variable (dlhandle, "emodule_version");
|
|
394 if (f == NULL || *f == NULL)
|
563
|
395 signal_error (Qdll_error, "Missing symbol `emodule_version': Invalid dynamic module", Qunbound);
|
428
|
396
|
1706
|
397 mver = (CIbyte *) ALLOCA (strlen (*f) + 1);
|
428
|
398 strcpy (mver, *f);
|
|
399
|
1706
|
400 f = (const CIbyte **) dll_variable (dlhandle, "emodule_title");
|
|
401 if (f == NULL || *f == NULL)
|
563
|
402 signal_error (Qdll_error, "Invalid dynamic module: Missing symbol `emodule_title'", Qunbound);
|
428
|
403
|
1706
|
404 mtitle = (CIbyte *) ALLOCA (strlen (*f) + 1);
|
428
|
405 strcpy (mtitle, *f);
|
|
406
|
1706
|
407 symname = (CIbyte *) ALLOCA (strlen (mname) + 15);
|
428
|
408
|
|
409 strcpy (symname, "modules_of_");
|
|
410 strcat (symname, mname);
|
|
411 modload = (void (*)(void))dll_function (dlhandle, symname);
|
|
412 /*
|
442
|
413 * modload is optional. If the module doesn't require other modules it can
|
428
|
414 * be left out.
|
|
415 */
|
|
416
|
|
417 strcpy (symname, "syms_of_");
|
|
418 strcat (symname, mname);
|
|
419 modsyms = (void (*)(void))dll_function (dlhandle, symname);
|
1706
|
420 if (modsyms == NULL)
|
563
|
421 {
|
|
422 missing_symbol:
|
|
423 signal_error (Qdll_error, "Invalid dynamic module: Missing symbol",
|
|
424 build_string (symname));
|
|
425 }
|
428
|
426
|
|
427 strcpy (symname, "vars_of_");
|
|
428 strcat (symname, mname);
|
|
429 modvars = (void (*)(void))dll_function (dlhandle, symname);
|
1706
|
430 if (modvars == NULL)
|
563
|
431 goto missing_symbol;
|
428
|
432
|
|
433 strcpy (symname, "docs_of_");
|
|
434 strcat (symname, mname);
|
|
435 moddocs = (void (*)(void))dll_function (dlhandle, symname);
|
1706
|
436 if (moddocs == NULL)
|
563
|
437 goto missing_symbol;
|
428
|
438
|
996
|
439 /* Now look for the optional unload function. */
|
|
440 strcpy (symname, "unload_");
|
|
441 strcat (symname, mname);
|
|
442 modunld = (void (*)(void))dll_function (dlhandle, symname);
|
|
443
|
428
|
444 if (modname && modname[0] && strcmp (modname, mname))
|
563
|
445 signal_error (Qdll_error, "Module name mismatch", Qunbound);
|
428
|
446
|
|
447 if (modver && modver[0] && strcmp (modver, mver))
|
563
|
448 signal_error (Qdll_error, "Module version mismatch", Qunbound);
|
428
|
449
|
|
450 /*
|
|
451 * Attempt to make a new slot for this module. If this really is the
|
|
452 * first time we are loading this module, the used member will be 0.
|
|
453 * If that is non-zero, we know that we have a previously loaded module
|
442
|
454 * of the same name and version, and we don't need to go any further.
|
428
|
455 */
|
|
456 mpx = find_make_module (soname, mname, mver, 0);
|
|
457 mp = &modules[mpx];
|
|
458 if (mp->used > 0)
|
|
459 {
|
|
460 emodules_depth--;
|
|
461 dll_close (dlhandle);
|
806
|
462 dlhandle = 0; /* Zero this out before module_load_unwind runs */
|
428
|
463 return;
|
|
464 }
|
|
465
|
|
466 if (!load_modules_quietly)
|
|
467 message ("Loading %s v%s (%s)", mname, mver, mtitle);
|
|
468
|
|
469 /*
|
|
470 * We have passed the basic initialization, and can now add this
|
|
471 * module to the list of modules.
|
|
472 */
|
|
473 mp->used = emodules_depth + 1;
|
|
474 mp->soname = xstrdup (soname);
|
|
475 mp->modname = xstrdup (mname);
|
|
476 mp->modver = xstrdup (mver);
|
|
477 mp->modtitle = xstrdup (mtitle);
|
|
478 mp->dlhandle = dlhandle;
|
996
|
479 mp->unload = modunld;
|
428
|
480 dlhandle = 0;
|
|
481
|
996
|
482 old_load_list = Vcurrent_load_list;
|
|
483 Vcurrent_load_list = Qnil;
|
|
484 LOADHIST_ATTACH (lisp_modname);
|
|
485 LOADHIST_ATTACH (module_tag);
|
|
486
|
428
|
487 /*
|
|
488 * Now we need to call the module init function and perform the various
|
|
489 * startup tasks.
|
|
490 */
|
|
491 if (modload != 0)
|
|
492 (*modload)();
|
|
493
|
|
494 /*
|
|
495 * Now we can get the module to initialize its symbols, and then its
|
|
496 * variables, and lastly the documentation strings.
|
|
497 */
|
|
498 (*modsyms)();
|
|
499 (*modvars)();
|
|
500 (*moddocs)();
|
|
501
|
|
502 if (!load_modules_quietly)
|
|
503 message ("Loaded module %s v%s (%s)", mname, mver, mtitle);
|
|
504
|
996
|
505 Vload_history = Fcons (Fnreverse (Vcurrent_load_list), Vload_history);
|
|
506 Vcurrent_load_list = old_load_list;
|
|
507 UNGCPRO;
|
428
|
508
|
|
509 emodules_depth--;
|
|
510 if (emodules_depth == 0)
|
|
511 {
|
|
512 /*
|
|
513 * We have reached the end of the load chain. We now go through the
|
|
514 * list of loaded modules and mark all the valid modules as just
|
|
515 * that.
|
|
516 */
|
|
517 for (x = 0; x < modnum; x++)
|
|
518 if (modules[x].used > 1)
|
|
519 modules[x].used = 1;
|
|
520 }
|
|
521 }
|
|
522
|
|
523 void
|
442
|
524 emodules_doc_subr(const char *symname, const char *doc)
|
428
|
525 {
|
|
526 Bytecount len = strlen (symname);
|
867
|
527 Lisp_Object sym = oblookup (Vobarray, (const Ibyte *)symname, len);
|
440
|
528 Lisp_Subr *subr;
|
428
|
529
|
1632
|
530 /* Skip autoload cookies */
|
|
531 if (SYMBOLP (sym) && SUBRP (XSYMBOL (sym)->function))
|
428
|
532 {
|
1632
|
533 subr = XSUBR (XSYMBOL (sym)->function);
|
428
|
534 subr->doc = xstrdup (doc);
|
|
535 }
|
|
536 /*
|
|
537 * FIXME: I wish there was some way to avoid the xstrdup(). Is it
|
|
538 * possible to just set a pointer to the string, or somehow create a
|
|
539 * symbol whose value we can point to the constant string? Can someone
|
|
540 * look into this?
|
|
541 */
|
|
542 }
|
|
543
|
|
544 void
|
442
|
545 emodules_doc_sym (const char *symname, const char *doc)
|
428
|
546 {
|
|
547 Bytecount len = strlen (symname);
|
867
|
548 Lisp_Object sym = oblookup (Vobarray, (const Ibyte *)symname, len);
|
428
|
549 Lisp_Object docstr;
|
|
550 struct gcpro gcpro1;
|
|
551
|
|
552 if (SYMBOLP(sym))
|
|
553 {
|
|
554 docstr = build_string (doc);
|
|
555 GCPRO1(docstr);
|
|
556 Fput (sym, Qvariable_documentation, docstr);
|
|
557 UNGCPRO;
|
|
558 }
|
|
559 }
|
|
560
|
|
561
|
|
562 void
|
|
563 syms_of_module (void)
|
|
564 {
|
564
|
565 DEFERROR_STANDARD (Qdll_error, Qerror);
|
996
|
566 DEFSYMBOL (Qmodule);
|
|
567 DEFSYMBOL (Qunload_module);
|
428
|
568 DEFSUBR(Fload_module);
|
|
569 DEFSUBR(Flist_modules);
|
|
570 DEFSUBR(Funload_module);
|
996
|
571 module_tag = Fcons (Qmodule, Qnil);
|
|
572 staticpro (&module_tag);
|
|
573 Fput (Qunload_module, Qdisabled, Qt);
|
428
|
574 }
|
|
575
|
|
576 void
|
|
577 reinit_vars_of_module (void)
|
|
578 {
|
|
579 emodules_depth = 0;
|
1706
|
580 modules = NULL;
|
428
|
581 modnum = 0;
|
|
582 }
|
|
583
|
1750
|
584 #endif /* HAVE_SHLIB */
|
|
585
|
428
|
586 void
|
|
587 vars_of_module (void)
|
|
588 {
|
1750
|
589 #ifdef HAVE_SHLIB
|
|
590 Fprovide (intern ("modules"));
|
|
591
|
2078
|
592 #ifdef HAVE_LTDL
|
|
593 lt_dlinit ();
|
|
594 lt_dlmalloc = (lt_ptr (*) (size_t)) xmalloc;
|
|
595 lt_dlrealloc = (lt_ptr (*) (lt_ptr, size_t)) xrealloc;
|
|
596 lt_dlfree = (void (*) (lt_ptr)) xfree_1;
|
|
597 #endif
|
|
598
|
428
|
599 DEFVAR_LISP ("module-version", &Vmodule_version /*
|
|
600 Emacs dynamic loading mechanism version, as a string.
|
|
601
|
|
602 This string is in the form XX.YY.ppp, where XX is the major version
|
|
603 number, YY is the minor version number, and ppp is the patch level.
|
442
|
604 This variable can be used to distinguish between different versions of
|
428
|
605 the dynamic loading technology used in Emacs, if required. It is not
|
|
606 a given that this value will be the same as the Emacs version number.
|
|
607 */ );
|
|
608 Vmodule_version = build_string (EMODULES_VERSION);
|
|
609
|
|
610 DEFVAR_BOOL ("load-modules-quietly", &load_modules_quietly /*
|
|
611 *Set to t if module loading is to be silent.
|
|
612
|
|
613 Normally, when loading dynamic modules, Emacs will inform you of its
|
|
614 progress, and will display the module name and version if the module
|
|
615 is loaded correctly. Setting this variable to `t' will suppress these
|
|
616 messages. This would normally only be done if `load-module' was being
|
|
617 called by a Lisp function.
|
|
618 */);
|
1733
|
619 load_modules_quietly = 0;
|
428
|
620
|
1750
|
621 DEFVAR_BOOL ("unloading-module", &unloading_module /*
|
|
622 Used internally by `unload-feature'. Do not set this variable.
|
|
623 Danger, danger, Will Robinson!
|
|
624 */);
|
|
625 unloading_module = 0;
|
|
626
|
|
627 #endif /* HAVE_SHLIB */
|
|
628
|
428
|
629 DEFVAR_LISP ("module-load-path", &Vmodule_load_path /*
|
|
630 *List of directories to search for dynamic modules to load.
|
|
631 Each element is a string (directory name) or nil (try default directory).
|
|
632
|
|
633 Note that elements of this list *may not* begin with "~", so you must
|
442
|
634 call `expand-file-name' on them before adding them to this list.
|
428
|
635
|
|
636 Initialized based on EMACSMODULEPATH environment variable, if any, otherwise
|
|
637 to default specified the file `paths.h' when XEmacs was built. If there
|
|
638 were no paths specified in `paths.h', then XEmacs chooses a default
|
|
639 value for this variable by looking around in the file-system near the
|
|
640 directory in which the XEmacs executable resides.
|
|
641
|
|
642 Due to the nature of dynamic modules, the path names should almost always
|
442
|
643 refer to architecture-dependent directories. It is unwise to attempt to
|
|
644 store dynamic modules in a heterogenous environment. Some environments
|
428
|
645 are similar enough to each other that XEmacs will be unable to determine
|
|
646 the correctness of a dynamic module, which can have unpredictable results
|
|
647 when a dynamic module is loaded.
|
|
648 */);
|
1733
|
649 Vmodule_load_path = Qnil;
|
428
|
650
|
1733
|
651 DEFVAR_LISP ("module-extensions", &Vmodule_extensions /*
|
|
652 *List of filename extensions to use when searching for dynamic modules.
|
|
653 */);
|
|
654 Vmodule_extensions = list5 (build_string (".ell"),
|
996
|
655 build_string (".so"),
|
1381
|
656 build_string (".dll"),
|
1733
|
657 build_string (".dylib"),
|
|
658 build_string (""));
|
428
|
659 }
|