diff options
Diffstat (limited to 'Python/bltinmodule.c')
| -rw-r--r-- | Python/bltinmodule.c | 535 | 
1 files changed, 535 insertions, 0 deletions
diff --git a/Python/bltinmodule.c b/Python/bltinmodule.c new file mode 100644 index 0000000000..dc88581c6b --- /dev/null +++ b/Python/bltinmodule.c @@ -0,0 +1,535 @@ +/* Built-in functions */ + +#include "allobjects.h" + +#include "node.h" +#include "graminit.h" +#include "errcode.h" +#include "sysmodule.h" +#include "builtinmodule.h" +#include "import.h" +#include "pythonrun.h" +#include "compile.h" /* For ceval.h */ +#include "ceval.h" +#include "modsupport.h" + +static object * +builtin_abs(self, v) +	object *self; +	object *v; +{ +	/* XXX This should be a method in the as_number struct in the type */ +	if (v == NULL) { +		/* */ +	} +	else if (is_intobject(v)) { +		long x = getintvalue(v); +		if (x < 0) +			x = -x; +		return newintobject(x); +	} +	else if (is_floatobject(v)) { +		double x = getfloatvalue(v); +		if (x < 0) +			x = -x; +		return newfloatobject(x); +	} +	err_setstr(TypeError, "abs() argument must be float or int"); +	return NULL; +} + +static object * +builtin_chr(self, v) +	object *self; +	object *v; +{ +	long x; +	char s[1]; +	if (v == NULL || !is_intobject(v)) { +		err_setstr(TypeError, "chr() must have int argument"); +		return NULL; +	} +	x = getintvalue(v); +	if (x < 0 || x >= 256) { +		err_setstr(RuntimeError, "chr() arg not in range(256)"); +		return NULL; +	} +	s[0] = x; +	return newsizedstringobject(s, 1); +} + +static object * +builtin_dir(self, v) +	object *self; +	object *v; +{ +	object *d; +	if (v == NULL) { +		d = getlocals(); +	} +	else { +		if (!is_moduleobject(v)) { +			err_setstr(TypeError, +				"dir() argument, must be module or absent"); +			return NULL; +		} +		d = getmoduledict(v); +	} +	v = getdictkeys(d); +	if (sortlist(v) != 0) { +		DECREF(v); +		v = NULL; +	} +	return v; +} + +static object * +builtin_divmod(self, v) +	object *self; +	object *v; +{ +	object *x, *y; +	long xi, yi, xdivy, xmody; +	if (v == NULL || !is_tupleobject(v) || gettuplesize(v) != 2) { +		err_setstr(TypeError, "divmod() requires 2 int arguments"); +		return NULL; +	} +	x = gettupleitem(v, 0); +	y = gettupleitem(v, 1); +	if (!is_intobject(x) || !is_intobject(y)) { +		err_setstr(TypeError, "divmod() requires 2 int arguments"); +		return NULL; +	} +	xi = getintvalue(x); +	yi = getintvalue(y); +	if (yi == 0) { +		err_setstr(TypeError, "divmod() division by zero"); +		return NULL; +	} +	if (yi < 0) { +		xdivy = -xi / -yi; +	} +	else { +		xdivy = xi / yi; +	} +	xmody = xi - xdivy*yi; +	if (xmody < 0 && yi > 0 || xmody > 0 && yi < 0) { +		xmody += yi; +		xdivy -= 1; +	} +	v = newtupleobject(2); +	x = newintobject(xdivy); +	y = newintobject(xmody); +	if (v == NULL || x == NULL || y == NULL || +		settupleitem(v, 0, x) != 0 || +		settupleitem(v, 1, y) != 0) { +		XDECREF(v); +		XDECREF(x); +		XDECREF(y); +		return NULL; +	} +	return v; +} + +static object * +exec_eval(v, start) +	object *v; +	int start; +{ +	object *str = NULL, *globals = NULL, *locals = NULL; +	int n; +	if (v != NULL) { +		if (is_stringobject(v)) +			str = v; +		else if (is_tupleobject(v) && +				((n = gettuplesize(v)) == 2 || n == 3)) { +			str = gettupleitem(v, 0); +			globals = gettupleitem(v, 1); +			if (n == 3) +				locals = gettupleitem(v, 2); +		} +	} +	if (str == NULL || !is_stringobject(str) || +			globals != NULL && !is_dictobject(globals) || +			locals != NULL && !is_dictobject(locals)) { +		err_setstr(TypeError, +		    "exec/eval arguments must be string[,dict[,dict]]"); +		return NULL; +	} +	return run_string(getstringvalue(str), start, globals, locals); +} + +static object * +builtin_eval(self, v) +	object *self; +	object *v; +{ +	return exec_eval(v, eval_input); +} + +static object * +builtin_exec(self, v) +	object *self; +	object *v; +{ +	return exec_eval(v, file_input); +} + +static object * +builtin_float(self, v) +	object *self; +	object *v; +{ +	if (v == NULL) { +		/* */ +	} +	else if (is_floatobject(v)) { +		INCREF(v); +		return v; +	} +	else if (is_intobject(v)) { +		long x = getintvalue(v); +		return newfloatobject((double)x); +	} +	err_setstr(TypeError, "float() argument must be float or int"); +	return NULL; +} + +static object * +builtin_input(self, v) +	object *self; +	object *v; +{ +	FILE *in = sysgetfile("stdin", stdin); +	FILE *out = sysgetfile("stdout", stdout); +	node *n; +	int err; +	object *m, *d; +	flushline(); +	if (v != NULL) +		printobject(v, out, PRINT_RAW); +	m = add_module("__main__"); +	d = getmoduledict(m); +	return run_file(in, "<stdin>", expr_input, d, d); +} + +static object * +builtin_int(self, v) +	object *self; +	object *v; +{ +	if (v == NULL) { +		/* */ +	} +	else if (is_intobject(v)) { +		INCREF(v); +		return v; +	} +	else if (is_floatobject(v)) { +		double x = getfloatvalue(v); +		return newintobject((long)x); +	} +	err_setstr(TypeError, "int() argument must be float or int"); +	return NULL; +} + +static object * +builtin_len(self, v) +	object *self; +	object *v; +{ +	long len; +	typeobject *tp; +	if (v == NULL) { +		err_setstr(TypeError, "len() without argument"); +		return NULL; +	} +	tp = v->ob_type; +	if (tp->tp_as_sequence != NULL) { +		len = (*tp->tp_as_sequence->sq_length)(v); +	} +	else if (tp->tp_as_mapping != NULL) { +		len = (*tp->tp_as_mapping->mp_length)(v); +	} +	else { +		err_setstr(TypeError, "len() of unsized object"); +		return NULL; +	} +	return newintobject(len); +} + +static object * +min_max(v, sign) +	object *v; +	int sign; +{ +	int i, n, cmp; +	object *w, *x; +	sequence_methods *sq; +	if (v == NULL) { +		err_setstr(TypeError, "min() or max() without argument"); +		return NULL; +	} +	sq = v->ob_type->tp_as_sequence; +	if (sq == NULL) { +		err_setstr(TypeError, "min() or max() of non-sequence"); +		return NULL; +	} +	n = (*sq->sq_length)(v); +	if (n == 0) { +		err_setstr(RuntimeError, "min() or max() of empty sequence"); +		return NULL; +	} +	w = (*sq->sq_item)(v, 0); /* Implies INCREF */ +	for (i = 1; i < n; i++) { +		x = (*sq->sq_item)(v, i); /* Implies INCREF */ +		cmp = cmpobject(x, w); +		if (cmp * sign > 0) { +			DECREF(w); +			w = x; +		} +		else +			DECREF(x); +	} +	return w; +} + +static object * +builtin_min(self, v) +	object *self; +	object *v; +{ +	return min_max(v, -1); +} + +static object * +builtin_max(self, v) +	object *self; +	object *v; +{ +	return min_max(v, 1); +} + +static object * +builtin_open(self, v) +	object *self; +	object *v; +{ +	object *name, *mode; +	if (v == NULL || !is_tupleobject(v) || gettuplesize(v) != 2 || +		!is_stringobject(name = gettupleitem(v, 0)) || +		!is_stringobject(mode = gettupleitem(v, 1))) { +		err_setstr(TypeError, "open() requires 2 string arguments"); +		return NULL; +	} +	v = newfileobject(getstringvalue(name), getstringvalue(mode)); +	return v; +} + +static object * +builtin_ord(self, v) +	object *self; +	object *v; +{ +	if (v == NULL || !is_stringobject(v)) { +		err_setstr(TypeError, "ord() must have string argument"); +		return NULL; +	} +	if (getstringsize(v) != 1) { +		err_setstr(RuntimeError, "ord() arg must have length 1"); +		return NULL; +	} +	return newintobject((long)(getstringvalue(v)[0] & 0xff)); +} + +static object * +builtin_range(self, v) +	object *self; +	object *v; +{ +	static char *errmsg = "range() requires 1-3 int arguments"; +	int i, n; +	long ilow, ihigh, istep; +	if (v != NULL && is_intobject(v)) { +		ilow = 0; ihigh = getintvalue(v); istep = 1; +	} +	else if (v == NULL || !is_tupleobject(v)) { +		err_setstr(TypeError, errmsg); +		return NULL; +	} +	else { +		n = gettuplesize(v); +		if (n < 1 || n > 3) { +			err_setstr(TypeError, errmsg); +			return NULL; +		} +		for (i = 0; i < n; i++) { +			if (!is_intobject(gettupleitem(v, i))) { +				err_setstr(TypeError, errmsg); +				return NULL; +			} +		} +		if (n == 3) { +			istep = getintvalue(gettupleitem(v, 2)); +			--n; +		} +		else +			istep = 1; +		ihigh = getintvalue(gettupleitem(v, --n)); +		if (n > 0) +			ilow = getintvalue(gettupleitem(v, 0)); +		else +			ilow = 0; +	} +	if (istep == 0) { +		err_setstr(RuntimeError, "zero step for range()"); +		return NULL; +	} +	/* XXX ought to check overflow of subtraction */ +	if (istep > 0) +		n = (ihigh - ilow + istep - 1) / istep; +	else +		n = (ihigh - ilow + istep + 1) / istep; +	if (n < 0) +		n = 0; +	v = newlistobject(n); +	if (v == NULL) +		return NULL; +	for (i = 0; i < n; i++) { +		object *w = newintobject(ilow); +		if (w == NULL) { +			DECREF(v); +			return NULL; +		} +		setlistitem(v, i, w); +		ilow += istep; +	} +	return v; +} + +static object * +builtin_raw_input(self, v) +	object *self; +	object *v; +{ +	FILE *in = sysgetfile("stdin", stdin); +	FILE *out = sysgetfile("stdout", stdout); +	char *p; +	int err; +	int n = 1000; +	flushline(); +	if (v != NULL) +		printobject(v, out, PRINT_RAW); +	v = newsizedstringobject((char *)NULL, n); +	if (v != NULL) { +		if ((err = fgets_intr(getstringvalue(v), n+1, in)) != E_OK) { +			err_input(err); +			DECREF(v); +			return NULL; +		} +		else { +			n = strlen(getstringvalue(v)); +			if (n > 0 && getstringvalue(v)[n-1] == '\n') +				n--; +			resizestring(&v, n); +		} +	} +	return v; +} + +static object * +builtin_reload(self, v) +	object *self; +	object *v; +{ +	return reload_module(v); +} + +static object * +builtin_type(self, v) +	object *self; +	object *v; +{ +	if (v == NULL) { +		err_setstr(TypeError, "type() requres an argument"); +		return NULL; +	} +	v = (object *)v->ob_type; +	INCREF(v); +	return v; +} + +static struct methodlist builtin_methods[] = { +	{"abs", builtin_abs}, +	{"chr", builtin_chr}, +	{"dir", builtin_dir}, +	{"divmod", builtin_divmod}, +	{"eval", builtin_eval}, +	{"exec", builtin_exec}, +	{"float", builtin_float}, +	{"input", builtin_input}, +	{"int", builtin_int}, +	{"len", builtin_len}, +	{"min", builtin_min}, +	{"max", builtin_max}, +	{"open", builtin_open}, /* XXX move to OS module */ +	{"ord", builtin_ord}, +	{"range", builtin_range}, +	{"raw_input", builtin_raw_input}, +	{"reload", builtin_reload}, +	{"type", builtin_type}, +	{NULL, NULL}, +}; + +static object *builtin_dict; + +object * +getbuiltin(name) +	char *name; +{ +	return dictlookup(builtin_dict, name); +} + +/* Predefined exceptions */ + +object *RuntimeError; +object *EOFError; +object *TypeError; +object *MemoryError; +object *NameError; +object *SystemError; +object *KeyboardInterrupt; + +static object * +newstdexception(name, message) +	char *name, *message; +{ +	object *v = newstringobject(message); +	if (v == NULL || dictinsert(builtin_dict, name, v) != 0) +		fatal("no mem for new standard exception"); +	return v; +} + +static void +initerrors() +{ +	RuntimeError = newstdexception("RuntimeError", "run-time error"); +	EOFError = newstdexception("EOFError", "end-of-file read"); +	TypeError = newstdexception("TypeError", "type error"); +	MemoryError = newstdexception("MemoryError", "out of memory"); +	NameError = newstdexception("NameError", "undefined name"); +	SystemError = newstdexception("SystemError", "system error"); +	KeyboardInterrupt = +		newstdexception("KeyboardInterrupt", "keyboard interrupt"); +} + +void +initbuiltin() +{ +	object *m; +	m = initmodule("builtin", builtin_methods); +	builtin_dict = getmoduledict(m); +	INCREF(builtin_dict); +	initerrors(); +	(void) dictinsert(builtin_dict, "None", None); +}  | 
