mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-06-11 22:31:12 +02:00
make primitive-load-path load compiled files if available
* libguile/load.h: Update scm_search_path prototype. * libguile/load.c: Include vm.h for load-compiled/vm. Not sure if this is bad wrt modularity. (scm_c_string_has_an_ext): New private helper. (scm_search_path): Add an extra optional arg, `require_exts'; if true, require that the returned file name have one of the given extensions. Changes the C API, but not the scheme API. (scm_sys_search_load_path): Adapt to scm_search_path API change. (primitive-load-path): Here is the craziness: load a compiled file if found and newer than the corresponding (or not) source file. (scm_init_load): Define %load-compiled-extensions as the list of extensions denoting compiled files; defaults to '(".go").
This commit is contained in:
parent
90b0be2028
commit
22f4ee4882
3 changed files with 90 additions and 8 deletions
|
@ -44,6 +44,8 @@
|
||||||
#include "libguile/load.h"
|
#include "libguile/load.h"
|
||||||
#include "libguile/fluids.h"
|
#include "libguile/fluids.h"
|
||||||
|
|
||||||
|
#include "libguile/vm.h" /* for load-compiled/vm */
|
||||||
|
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
@ -172,6 +174,9 @@ static SCM *scm_loc_load_path;
|
||||||
/* List of extensions we try adding to the filenames. */
|
/* List of extensions we try adding to the filenames. */
|
||||||
static SCM *scm_loc_load_extensions;
|
static SCM *scm_loc_load_extensions;
|
||||||
|
|
||||||
|
/* Like %load-extensions, but for compiled files. */
|
||||||
|
static SCM *scm_loc_load_compiled_extensions;
|
||||||
|
|
||||||
|
|
||||||
SCM_DEFINE (scm_parse_path, "parse-path", 1, 1, 0,
|
SCM_DEFINE (scm_parse_path, "parse-path", 1, 1, 0,
|
||||||
(SCM path, SCM tail),
|
(SCM path, SCM tail),
|
||||||
|
@ -291,14 +296,33 @@ stringbuf_cat (struct stringbuf *buf, char *str)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
scm_c_string_has_an_ext (char *str, size_t len, SCM extensions)
|
||||||
|
{
|
||||||
|
for (; !scm_is_null (extensions); extensions = SCM_CDR (extensions))
|
||||||
|
{
|
||||||
|
char *ext;
|
||||||
|
size_t extlen;
|
||||||
|
int match;
|
||||||
|
ext = scm_to_locale_string (SCM_CAR (extensions));
|
||||||
|
extlen = strlen (ext);
|
||||||
|
match = (len > extlen && str[len - extlen - 1] == '.'
|
||||||
|
&& strncmp (str + (len - extlen), ext, extlen) == 0);
|
||||||
|
free (ext);
|
||||||
|
if (match)
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* Search PATH for a directory containing a file named FILENAME.
|
/* Search PATH for a directory containing a file named FILENAME.
|
||||||
The file must be readable, and not a directory.
|
The file must be readable, and not a directory.
|
||||||
If we find one, return its full filename; otherwise, return #f.
|
If we find one, return its full filename; otherwise, return #f.
|
||||||
If FILENAME is absolute, return it unchanged.
|
If FILENAME is absolute, return it unchanged.
|
||||||
If given, EXTENSIONS is a list of strings; for each directory
|
If given, EXTENSIONS is a list of strings; for each directory
|
||||||
in PATH, we search for FILENAME concatenated with each EXTENSION. */
|
in PATH, we search for FILENAME concatenated with each EXTENSION. */
|
||||||
SCM_DEFINE (scm_search_path, "search-path", 2, 1, 0,
|
SCM_DEFINE (scm_search_path, "search-path", 2, 2, 0,
|
||||||
(SCM path, SCM filename, SCM extensions),
|
(SCM path, SCM filename, SCM extensions, SCM require_exts),
|
||||||
"Search @var{path} for a directory containing a file named\n"
|
"Search @var{path} for a directory containing a file named\n"
|
||||||
"@var{filename}. The file must be readable, and not a directory.\n"
|
"@var{filename}. The file must be readable, and not a directory.\n"
|
||||||
"If we find one, return its full filename; otherwise, return\n"
|
"If we find one, return its full filename; otherwise, return\n"
|
||||||
|
@ -316,6 +340,9 @@ SCM_DEFINE (scm_search_path, "search-path", 2, 1, 0,
|
||||||
if (SCM_UNBNDP (extensions))
|
if (SCM_UNBNDP (extensions))
|
||||||
extensions = SCM_EOL;
|
extensions = SCM_EOL;
|
||||||
|
|
||||||
|
if (SCM_UNBNDP (require_exts))
|
||||||
|
require_exts = SCM_BOOL_F;
|
||||||
|
|
||||||
scm_dynwind_begin (0);
|
scm_dynwind_begin (0);
|
||||||
|
|
||||||
filename_chars = scm_to_locale_string (filename);
|
filename_chars = scm_to_locale_string (filename);
|
||||||
|
@ -334,8 +361,14 @@ SCM_DEFINE (scm_search_path, "search-path", 2, 1, 0,
|
||||||
if (filename_len >= 1 && filename_chars[0] == '/')
|
if (filename_len >= 1 && filename_chars[0] == '/')
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
|
SCM res = filename;
|
||||||
|
if (scm_is_true (require_exts) &&
|
||||||
|
!scm_c_string_has_an_ext (filename_chars, filename_len,
|
||||||
|
extensions))
|
||||||
|
res = SCM_BOOL_F;
|
||||||
|
|
||||||
scm_dynwind_end ();
|
scm_dynwind_end ();
|
||||||
return filename;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If FILENAME has an extension, don't try to add EXTENSIONS to it. */
|
/* If FILENAME has an extension, don't try to add EXTENSIONS to it. */
|
||||||
|
@ -348,6 +381,15 @@ SCM_DEFINE (scm_search_path, "search-path", 2, 1, 0,
|
||||||
{
|
{
|
||||||
if (*endp == '.')
|
if (*endp == '.')
|
||||||
{
|
{
|
||||||
|
if (scm_is_true (require_exts) &&
|
||||||
|
!scm_c_string_has_an_ext (filename_chars, filename_len,
|
||||||
|
extensions))
|
||||||
|
{
|
||||||
|
/* This filename has an extension, but not one of the right
|
||||||
|
ones... */
|
||||||
|
scm_dynwind_end ();
|
||||||
|
return SCM_BOOL_F;
|
||||||
|
}
|
||||||
/* This filename already has an extension, so cancel the
|
/* This filename already has an extension, so cancel the
|
||||||
list of extensions. */
|
list of extensions. */
|
||||||
extensions = SCM_EOL;
|
extensions = SCM_EOL;
|
||||||
|
@ -453,7 +495,7 @@ SCM_DEFINE (scm_sys_search_load_path, "%search-load-path", 1, 0, 0,
|
||||||
SCM_MISC_ERROR ("%load-path is not a proper list", SCM_EOL);
|
SCM_MISC_ERROR ("%load-path is not a proper list", SCM_EOL);
|
||||||
if (scm_ilength (exts) < 0)
|
if (scm_ilength (exts) < 0)
|
||||||
SCM_MISC_ERROR ("%load-extension list is not a proper list", SCM_EOL);
|
SCM_MISC_ERROR ("%load-extension list is not a proper list", SCM_EOL);
|
||||||
return scm_search_path (path, filename, exts);
|
return scm_search_path (path, filename, exts, SCM_UNDEFINED);
|
||||||
}
|
}
|
||||||
#undef FUNC_NAME
|
#undef FUNC_NAME
|
||||||
|
|
||||||
|
@ -466,15 +508,51 @@ SCM_DEFINE (scm_primitive_load_path, "primitive-load-path", 1, 0, 0,
|
||||||
"an error is signalled.")
|
"an error is signalled.")
|
||||||
#define FUNC_NAME s_scm_primitive_load_path
|
#define FUNC_NAME s_scm_primitive_load_path
|
||||||
{
|
{
|
||||||
SCM full_filename;
|
SCM full_filename, compiled_filename;
|
||||||
|
|
||||||
full_filename = scm_sys_search_load_path (filename);
|
full_filename = scm_sys_search_load_path (filename);
|
||||||
|
compiled_filename = scm_search_path (*scm_loc_load_path,
|
||||||
|
filename,
|
||||||
|
*scm_loc_load_compiled_extensions,
|
||||||
|
SCM_BOOL_T);
|
||||||
|
|
||||||
if (scm_is_false (full_filename))
|
if (scm_is_false (full_filename) && scm_is_false (compiled_filename))
|
||||||
SCM_MISC_ERROR ("Unable to find file ~S in load path",
|
SCM_MISC_ERROR ("Unable to find file ~S in load path",
|
||||||
scm_list_1 (filename));
|
scm_list_1 (filename));
|
||||||
|
|
||||||
return scm_primitive_load (full_filename);
|
if (scm_is_false (compiled_filename))
|
||||||
|
return scm_primitive_load (full_filename);
|
||||||
|
|
||||||
|
if (scm_is_false (full_filename))
|
||||||
|
return scm_load_compiled_with_vm (compiled_filename);
|
||||||
|
|
||||||
|
{
|
||||||
|
char *source, *compiled;
|
||||||
|
struct stat stat_source, stat_compiled;
|
||||||
|
|
||||||
|
source = scm_to_locale_string (full_filename);
|
||||||
|
compiled = scm_to_locale_string (compiled_filename);
|
||||||
|
|
||||||
|
if (stat (source, &stat_source) == 0
|
||||||
|
&& stat (compiled, &stat_compiled) == 0
|
||||||
|
&& stat_source.st_mtime <= stat_compiled.st_mtime)
|
||||||
|
{
|
||||||
|
free (source);
|
||||||
|
free (compiled);
|
||||||
|
return scm_load_compiled_with_vm (compiled_filename);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
scm_puts (";;; note: source file ", scm_current_error_port ());
|
||||||
|
scm_puts (source, scm_current_error_port ());
|
||||||
|
scm_puts (" newer than compiled ", scm_current_error_port ());
|
||||||
|
scm_puts (compiled, scm_current_error_port ());
|
||||||
|
scm_puts ("\n", scm_current_error_port ());
|
||||||
|
free (source);
|
||||||
|
free (compiled);
|
||||||
|
return scm_primitive_load (full_filename);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#undef FUNC_NAME
|
#undef FUNC_NAME
|
||||||
|
|
||||||
|
@ -514,6 +592,9 @@ scm_init_load ()
|
||||||
= SCM_VARIABLE_LOC (scm_c_define ("%load-extensions",
|
= SCM_VARIABLE_LOC (scm_c_define ("%load-extensions",
|
||||||
scm_list_2 (scm_from_locale_string (".scm"),
|
scm_list_2 (scm_from_locale_string (".scm"),
|
||||||
scm_nullstr)));
|
scm_nullstr)));
|
||||||
|
scm_loc_load_compiled_extensions
|
||||||
|
= SCM_VARIABLE_LOC (scm_c_define ("%load-compiled-extensions",
|
||||||
|
scm_list_1 (scm_from_locale_string (".go"))));
|
||||||
scm_loc_load_hook = SCM_VARIABLE_LOC (scm_c_define ("%load-hook", SCM_BOOL_F));
|
scm_loc_load_hook = SCM_VARIABLE_LOC (scm_c_define ("%load-hook", SCM_BOOL_F));
|
||||||
|
|
||||||
the_reader = scm_make_fluid ();
|
the_reader = scm_make_fluid ();
|
||||||
|
|
|
@ -31,7 +31,7 @@ SCM_API SCM scm_c_primitive_load (const char *filename);
|
||||||
SCM_API SCM scm_sys_package_data_dir (void);
|
SCM_API SCM scm_sys_package_data_dir (void);
|
||||||
SCM_API SCM scm_sys_library_dir (void);
|
SCM_API SCM scm_sys_library_dir (void);
|
||||||
SCM_API SCM scm_sys_site_dir (void);
|
SCM_API SCM scm_sys_site_dir (void);
|
||||||
SCM_API SCM scm_search_path (SCM path, SCM filename, SCM exts);
|
SCM_API SCM scm_search_path (SCM path, SCM filename, SCM exts, SCM require_exts);
|
||||||
SCM_API SCM scm_sys_search_load_path (SCM filename);
|
SCM_API SCM scm_sys_search_load_path (SCM filename);
|
||||||
SCM_API SCM scm_primitive_load_path (SCM filename);
|
SCM_API SCM scm_primitive_load_path (SCM filename);
|
||||||
SCM_API SCM scm_c_primitive_load_path (const char *filename);
|
SCM_API SCM scm_c_primitive_load_path (const char *filename);
|
||||||
|
|
|
@ -43,6 +43,7 @@
|
||||||
#define _SCM_VM_H_
|
#define _SCM_VM_H_
|
||||||
|
|
||||||
#include <libguile.h>
|
#include <libguile.h>
|
||||||
|
#include <libguile/programs.h>
|
||||||
|
|
||||||
#define SCM_VM_BOOT_HOOK 0
|
#define SCM_VM_BOOT_HOOK 0
|
||||||
#define SCM_VM_HALT_HOOK 1
|
#define SCM_VM_HALT_HOOK 1
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue