0
0
mirror of https://github.com/python/cpython.git synced 2024-11-25 01:20:47 +01:00
cpython/Python/import.c

207 lines
3.5 KiB
C
Raw Normal View History

1990-10-14 13:07:46 +01:00
/* Module definition and import implementation */
1990-12-20 16:06:42 +01:00
#include "allobjects.h"
1990-10-14 13:07:46 +01:00
#include "node.h"
#include "token.h"
#include "graminit.h"
#include "import.h"
#include "errcode.h"
#include "sysmodule.h"
1990-12-20 16:06:42 +01:00
#include "pythonrun.h"
1990-10-14 13:07:46 +01:00
1990-12-20 16:06:42 +01:00
/* Define pathname separator used in file names */
1990-10-14 13:07:46 +01:00
#ifdef THINK_C
#define SEP ':'
#endif
#ifndef SEP
#define SEP '/'
#endif
1990-12-20 16:06:42 +01:00
static object *modules;
/* Initialization */
1990-10-14 13:07:46 +01:00
void
initimport()
{
1990-12-20 16:06:42 +01:00
if ((modules = newdictobject()) == NULL)
fatal("no mem for dictionary of modules");
1990-10-14 13:07:46 +01:00
}
object *
1990-12-20 16:06:42 +01:00
get_modules()
{
return modules;
}
object *
add_module(name)
1990-10-14 13:07:46 +01:00
char *name;
{
object *m;
1990-12-20 16:06:42 +01:00
if ((m = dictlookup(modules, name)) != NULL && is_moduleobject(m))
return m;
1990-10-14 13:07:46 +01:00
m = newmoduleobject(name);
if (m == NULL)
return NULL;
1990-12-20 16:06:42 +01:00
if (dictinsert(modules, name, m) != 0) {
1990-10-14 13:07:46 +01:00
DECREF(m);
return NULL;
}
1990-12-20 16:06:42 +01:00
DECREF(m); /* Yes, it still exists, in modules! */
1990-10-14 13:07:46 +01:00
return m;
}
static FILE *
1990-12-20 16:06:42 +01:00
open_module(name, suffix, namebuf)
1990-10-14 13:07:46 +01:00
char *name;
char *suffix;
1990-12-20 16:06:42 +01:00
char *namebuf; /* XXX No buffer overflow checks! */
1990-10-14 13:07:46 +01:00
{
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;
1990-12-20 16:06:42 +01:00
strcpy(namebuf+len, name);
strcat(namebuf, suffix);
1990-10-14 13:07:46 +01:00
fp = fopen(namebuf, "r");
if (fp != NULL)
break;
}
}
return fp;
}
static object *
1990-12-20 16:06:42 +01:00
get_module(m, name, m_ret)
/*module*/object *m;
1990-10-14 13:07:46 +01:00
char *name;
1990-12-20 16:06:42 +01:00
object **m_ret;
1990-10-14 13:07:46 +01:00
{
1990-12-20 16:06:42 +01:00
object *d;
1990-10-14 13:07:46 +01:00
FILE *fp;
1990-10-26 15:58:58 +01:00
node *n;
1990-10-14 13:07:46 +01:00
int err;
1990-12-20 16:06:42 +01:00
char namebuf[256];
1990-10-14 13:07:46 +01:00
1990-12-20 16:06:42 +01:00
fp = open_module(name, ".py", namebuf);
1990-10-14 13:07:46 +01:00
if (fp == NULL) {
1990-12-20 16:06:42 +01:00
if (m == NULL)
err_setstr(NameError, name);
else
err_setstr(RuntimeError, "no module source file");
1990-10-14 13:07:46 +01:00
return NULL;
}
1990-12-20 16:06:42 +01:00
err = parse_file(fp, namebuf, file_input, &n);
1990-10-14 13:07:46 +01:00
fclose(fp);
if (err != E_DONE) {
1990-12-20 16:06:42 +01:00
err_input(err);
1990-10-14 13:07:46 +01:00
return NULL;
}
if (m == NULL) {
1990-12-20 16:06:42 +01:00
m = add_module(name);
if (m == NULL) {
freetree(n);
return NULL;
}
*m_ret = m;
1990-10-14 13:07:46 +01:00
}
1990-12-20 16:06:42 +01:00
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);
1990-10-14 13:07:46 +01:00
return m;
}
object *
1990-12-20 16:06:42 +01:00
import_module(name)
1990-10-14 13:07:46 +01:00
char *name;
{
object *m;
1990-12-20 16:06:42 +01:00
if ((m = dictlookup(modules, name)) == NULL)
m = load_module(name);
1990-10-14 13:07:46 +01:00
return m;
}
1990-10-26 15:58:58 +01:00
object *
1990-12-20 16:06:42 +01:00
reload_module(m)
1990-10-26 15:58:58 +01:00
object *m;
{
if (m == NULL || !is_moduleobject(m)) {
1990-12-20 16:06:42 +01:00
err_setstr(TypeError, "reload() argument must be module");
1990-10-26 15:58:58 +01:00
return NULL;
}
1990-12-20 16:06:42 +01:00
/* 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);
1990-10-26 15:58:58 +01:00
}
1990-12-20 16:06:42 +01:00
}
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);
1990-10-26 15:58:58 +01:00
}
1990-12-20 16:06:42 +01:00
DECREF(modules);
1990-10-26 15:58:58 +01:00
}