207 lines
3.5 KiB
C
207 lines
3.5 KiB
C
/* Module definition and import implementation */
|
|
|
|
#include "allobjects.h"
|
|
|
|
#include "node.h"
|
|
#include "token.h"
|
|
#include "graminit.h"
|
|
#include "import.h"
|
|
#include "errcode.h"
|
|
#include "sysmodule.h"
|
|
#include "pythonrun.h"
|
|
|
|
/* Define pathname separator used in file names */
|
|
|
|
#ifdef THINK_C
|
|
#define SEP ':'
|
|
#endif
|
|
|
|
#ifndef SEP
|
|
#define SEP '/'
|
|
#endif
|
|
|
|
static object *modules;
|
|
|
|
/* Initialization */
|
|
|
|
void
|
|
initimport()
|
|
{
|
|
if ((modules = newdictobject()) == NULL)
|
|
fatal("no mem for dictionary of modules");
|
|
}
|
|
|
|
object *
|
|
get_modules()
|
|
{
|
|
return modules;
|
|
}
|
|
|
|
object *
|
|
add_module(name)
|
|
char *name;
|
|
{
|
|
object *m;
|
|
if ((m = dictlookup(modules, name)) != NULL && is_moduleobject(m))
|
|
return m;
|
|
m = newmoduleobject(name);
|
|
if (m == NULL)
|
|
return NULL;
|
|
if (dictinsert(modules, name, m) != 0) {
|
|
DECREF(m);
|
|
return NULL;
|
|
}
|
|
DECREF(m); /* Yes, it still exists, in modules! */
|
|
return m;
|
|
}
|
|
|
|
static FILE *
|
|
open_module(name, suffix, namebuf)
|
|
char *name;
|
|
char *suffix;
|
|
char *namebuf; /* XXX No buffer overflow checks! */
|
|
{
|
|
object *path;
|
|
FILE *fp;
|
|
|
|
path = sysget("path");
|
|
if (path == NULL || !is_listobject(path)) {
|
|
strcpy(namebuf, name);
|
|
strcat(namebuf, suffix);
|
|
fp = fopen(namebuf, "r");
|
|
}
|
|
else {
|
|
int npath = getlistsize(path);
|
|
int i;
|
|
fp = NULL;
|
|
for (i = 0; i < npath; i++) {
|
|
object *v = getlistitem(path, i);
|
|
int len;
|
|
if (!is_stringobject(v))
|
|
continue;
|
|
strcpy(namebuf, getstringvalue(v));
|
|
len = getstringsize(v);
|
|
if (len > 0 && namebuf[len-1] != SEP)
|
|
namebuf[len++] = SEP;
|
|
strcpy(namebuf+len, name);
|
|
strcat(namebuf, suffix);
|
|
fp = fopen(namebuf, "r");
|
|
if (fp != NULL)
|
|
break;
|
|
}
|
|
}
|
|
return fp;
|
|
}
|
|
|
|
static object *
|
|
get_module(m, name, m_ret)
|
|
/*module*/object *m;
|
|
char *name;
|
|
object **m_ret;
|
|
{
|
|
object *d;
|
|
FILE *fp;
|
|
node *n;
|
|
int err;
|
|
char namebuf[256];
|
|
|
|
fp = open_module(name, ".py", namebuf);
|
|
if (fp == NULL) {
|
|
if (m == NULL)
|
|
err_setstr(NameError, name);
|
|
else
|
|
err_setstr(RuntimeError, "no module source file");
|
|
return NULL;
|
|
}
|
|
err = parse_file(fp, namebuf, file_input, &n);
|
|
fclose(fp);
|
|
if (err != E_DONE) {
|
|
err_input(err);
|
|
return NULL;
|
|
}
|
|
if (m == NULL) {
|
|
m = add_module(name);
|
|
if (m == NULL) {
|
|
freetree(n);
|
|
return NULL;
|
|
}
|
|
*m_ret = m;
|
|
}
|
|
d = getmoduledict(m);
|
|
return run_node(n, namebuf, d, d);
|
|
}
|
|
|
|
static object *
|
|
load_module(name)
|
|
char *name;
|
|
{
|
|
object *m, *v;
|
|
v = get_module((object *)NULL, name, &m);
|
|
if (v == NULL)
|
|
return NULL;
|
|
DECREF(v);
|
|
return m;
|
|
}
|
|
|
|
object *
|
|
import_module(name)
|
|
char *name;
|
|
{
|
|
object *m;
|
|
if ((m = dictlookup(modules, name)) == NULL)
|
|
m = load_module(name);
|
|
return m;
|
|
}
|
|
|
|
object *
|
|
reload_module(m)
|
|
object *m;
|
|
{
|
|
if (m == NULL || !is_moduleobject(m)) {
|
|
err_setstr(TypeError, "reload() argument must be module");
|
|
return NULL;
|
|
}
|
|
/* XXX Ought to check for builtin modules -- can't reload these... */
|
|
return get_module(m, getmodulename(m), (object **)NULL);
|
|
}
|
|
|
|
static void
|
|
cleardict(d)
|
|
object *d;
|
|
{
|
|
int i;
|
|
for (i = getdictsize(d); --i >= 0; ) {
|
|
char *k;
|
|
k = getdictkey(d, i);
|
|
if (k != NULL)
|
|
(void) dictremove(d, k);
|
|
}
|
|
}
|
|
|
|
void
|
|
doneimport()
|
|
{
|
|
if (modules != NULL) {
|
|
int i;
|
|
/* Explicitly erase all modules; this is the safest way
|
|
to get rid of at least *some* circular dependencies */
|
|
for (i = getdictsize(modules); --i >= 0; ) {
|
|
char *k;
|
|
k = getdictkey(modules, i);
|
|
if (k != NULL) {
|
|
object *m;
|
|
m = dictlookup(modules, k);
|
|
if (m != NULL && is_moduleobject(m)) {
|
|
object *d;
|
|
d = getmoduledict(m);
|
|
if (d != NULL && is_dictobject(d)) {
|
|
cleardict(d);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
cleardict(modules);
|
|
}
|
|
DECREF(modules);
|
|
}
|