Commit adb69fcd authored by Neal Norwitz's avatar Neal Norwitz

Merge from ast-arena. This reduces the code in Python/ast.c by ~300 lines,

simplifies a lot of error handling code, and fixes many memory leaks.
parent 23a69589
...@@ -328,81 +328,79 @@ struct _alias { ...@@ -328,81 +328,79 @@ struct _alias {
}; };
mod_ty Module(asdl_seq * body); mod_ty Module(asdl_seq * body, PyArena *arena);
mod_ty Interactive(asdl_seq * body); mod_ty Interactive(asdl_seq * body, PyArena *arena);
mod_ty Expression(expr_ty body); mod_ty Expression(expr_ty body, PyArena *arena);
mod_ty Suite(asdl_seq * body); mod_ty Suite(asdl_seq * body, PyArena *arena);
stmt_ty FunctionDef(identifier name, arguments_ty args, asdl_seq * body, stmt_ty FunctionDef(identifier name, arguments_ty args, asdl_seq * body,
asdl_seq * decorators, int lineno); asdl_seq * decorators, int lineno, PyArena *arena);
stmt_ty ClassDef(identifier name, asdl_seq * bases, asdl_seq * body, int stmt_ty ClassDef(identifier name, asdl_seq * bases, asdl_seq * body, int
lineno); lineno, PyArena *arena);
stmt_ty Return(expr_ty value, int lineno); stmt_ty Return(expr_ty value, int lineno, PyArena *arena);
stmt_ty Delete(asdl_seq * targets, int lineno); stmt_ty Delete(asdl_seq * targets, int lineno, PyArena *arena);
stmt_ty Assign(asdl_seq * targets, expr_ty value, int lineno); stmt_ty Assign(asdl_seq * targets, expr_ty value, int lineno, PyArena *arena);
stmt_ty AugAssign(expr_ty target, operator_ty op, expr_ty value, int lineno); stmt_ty AugAssign(expr_ty target, operator_ty op, expr_ty value, int lineno,
stmt_ty Print(expr_ty dest, asdl_seq * values, bool nl, int lineno); PyArena *arena);
stmt_ty Print(expr_ty dest, asdl_seq * values, bool nl, int lineno, PyArena
*arena);
stmt_ty For(expr_ty target, expr_ty iter, asdl_seq * body, asdl_seq * orelse, stmt_ty For(expr_ty target, expr_ty iter, asdl_seq * body, asdl_seq * orelse,
int lineno); int lineno, PyArena *arena);
stmt_ty While(expr_ty test, asdl_seq * body, asdl_seq * orelse, int lineno); stmt_ty While(expr_ty test, asdl_seq * body, asdl_seq * orelse, int lineno,
stmt_ty If(expr_ty test, asdl_seq * body, asdl_seq * orelse, int lineno); PyArena *arena);
stmt_ty Raise(expr_ty type, expr_ty inst, expr_ty tback, int lineno); stmt_ty If(expr_ty test, asdl_seq * body, asdl_seq * orelse, int lineno,
PyArena *arena);
stmt_ty Raise(expr_ty type, expr_ty inst, expr_ty tback, int lineno, PyArena
*arena);
stmt_ty TryExcept(asdl_seq * body, asdl_seq * handlers, asdl_seq * orelse, int stmt_ty TryExcept(asdl_seq * body, asdl_seq * handlers, asdl_seq * orelse, int
lineno); lineno, PyArena *arena);
stmt_ty TryFinally(asdl_seq * body, asdl_seq * finalbody, int lineno); stmt_ty TryFinally(asdl_seq * body, asdl_seq * finalbody, int lineno, PyArena
stmt_ty Assert(expr_ty test, expr_ty msg, int lineno); *arena);
stmt_ty Import(asdl_seq * names, int lineno); stmt_ty Assert(expr_ty test, expr_ty msg, int lineno, PyArena *arena);
stmt_ty ImportFrom(identifier module, asdl_seq * names, int lineno); stmt_ty Import(asdl_seq * names, int lineno, PyArena *arena);
stmt_ty Exec(expr_ty body, expr_ty globals, expr_ty locals, int lineno); stmt_ty ImportFrom(identifier module, asdl_seq * names, int lineno, PyArena
stmt_ty Global(asdl_seq * names, int lineno); *arena);
stmt_ty Expr(expr_ty value, int lineno); stmt_ty Exec(expr_ty body, expr_ty globals, expr_ty locals, int lineno, PyArena
stmt_ty Pass(int lineno); *arena);
stmt_ty Break(int lineno); stmt_ty Global(asdl_seq * names, int lineno, PyArena *arena);
stmt_ty Continue(int lineno); stmt_ty Expr(expr_ty value, int lineno, PyArena *arena);
expr_ty BoolOp(boolop_ty op, asdl_seq * values, int lineno); stmt_ty Pass(int lineno, PyArena *arena);
expr_ty BinOp(expr_ty left, operator_ty op, expr_ty right, int lineno); stmt_ty Break(int lineno, PyArena *arena);
expr_ty UnaryOp(unaryop_ty op, expr_ty operand, int lineno); stmt_ty Continue(int lineno, PyArena *arena);
expr_ty Lambda(arguments_ty args, expr_ty body, int lineno); expr_ty BoolOp(boolop_ty op, asdl_seq * values, int lineno, PyArena *arena);
expr_ty Dict(asdl_seq * keys, asdl_seq * values, int lineno); expr_ty BinOp(expr_ty left, operator_ty op, expr_ty right, int lineno, PyArena
expr_ty ListComp(expr_ty elt, asdl_seq * generators, int lineno); *arena);
expr_ty GeneratorExp(expr_ty elt, asdl_seq * generators, int lineno); expr_ty UnaryOp(unaryop_ty op, expr_ty operand, int lineno, PyArena *arena);
expr_ty Yield(expr_ty value, int lineno); expr_ty Lambda(arguments_ty args, expr_ty body, int lineno, PyArena *arena);
expr_ty Dict(asdl_seq * keys, asdl_seq * values, int lineno, PyArena *arena);
expr_ty ListComp(expr_ty elt, asdl_seq * generators, int lineno, PyArena
*arena);
expr_ty GeneratorExp(expr_ty elt, asdl_seq * generators, int lineno, PyArena
*arena);
expr_ty Yield(expr_ty value, int lineno, PyArena *arena);
expr_ty Compare(expr_ty left, asdl_seq * ops, asdl_seq * comparators, int expr_ty Compare(expr_ty left, asdl_seq * ops, asdl_seq * comparators, int
lineno); lineno, PyArena *arena);
expr_ty Call(expr_ty func, asdl_seq * args, asdl_seq * keywords, expr_ty expr_ty Call(expr_ty func, asdl_seq * args, asdl_seq * keywords, expr_ty
starargs, expr_ty kwargs, int lineno); starargs, expr_ty kwargs, int lineno, PyArena *arena);
expr_ty Repr(expr_ty value, int lineno); expr_ty Repr(expr_ty value, int lineno, PyArena *arena);
expr_ty Num(object n, int lineno); expr_ty Num(object n, int lineno, PyArena *arena);
expr_ty Str(string s, int lineno); expr_ty Str(string s, int lineno, PyArena *arena);
expr_ty Attribute(expr_ty value, identifier attr, expr_context_ty ctx, int expr_ty Attribute(expr_ty value, identifier attr, expr_context_ty ctx, int
lineno); lineno, PyArena *arena);
expr_ty Subscript(expr_ty value, slice_ty slice, expr_context_ty ctx, int expr_ty Subscript(expr_ty value, slice_ty slice, expr_context_ty ctx, int
lineno); lineno, PyArena *arena);
expr_ty Name(identifier id, expr_context_ty ctx, int lineno); expr_ty Name(identifier id, expr_context_ty ctx, int lineno, PyArena *arena);
expr_ty List(asdl_seq * elts, expr_context_ty ctx, int lineno); expr_ty List(asdl_seq * elts, expr_context_ty ctx, int lineno, PyArena *arena);
expr_ty Tuple(asdl_seq * elts, expr_context_ty ctx, int lineno); expr_ty Tuple(asdl_seq * elts, expr_context_ty ctx, int lineno, PyArena *arena);
slice_ty Ellipsis(void); slice_ty Ellipsis(PyArena *arena);
slice_ty Slice(expr_ty lower, expr_ty upper, expr_ty step); slice_ty Slice(expr_ty lower, expr_ty upper, expr_ty step, PyArena *arena);
slice_ty ExtSlice(asdl_seq * dims); slice_ty ExtSlice(asdl_seq * dims, PyArena *arena);
slice_ty Index(expr_ty value); slice_ty Index(expr_ty value, PyArena *arena);
comprehension_ty comprehension(expr_ty target, expr_ty iter, asdl_seq * ifs); comprehension_ty comprehension(expr_ty target, expr_ty iter, asdl_seq * ifs,
excepthandler_ty excepthandler(expr_ty type, expr_ty name, asdl_seq * body); PyArena *arena);
excepthandler_ty excepthandler(expr_ty type, expr_ty name, asdl_seq * body,
PyArena *arena);
arguments_ty arguments(asdl_seq * args, identifier vararg, identifier kwarg, arguments_ty arguments(asdl_seq * args, identifier vararg, identifier kwarg,
asdl_seq * defaults); asdl_seq * defaults, PyArena *arena);
keyword_ty keyword(identifier arg, expr_ty value); keyword_ty keyword(identifier arg, expr_ty value, PyArena *arena);
alias_ty alias(identifier name, identifier asname); alias_ty alias(identifier name, identifier asname, PyArena *arena);
void free_mod(mod_ty);
void free_stmt(stmt_ty);
void free_expr(expr_ty);
void free_expr_context(expr_context_ty);
void free_slice(slice_ty);
void free_boolop(boolop_ty);
void free_operator(operator_ty);
void free_unaryop(unaryop_ty);
void free_cmpop(cmpop_ty);
void free_comprehension(comprehension_ty);
void free_excepthandler(excepthandler_ty);
void free_arguments(arguments_ty);
void free_keyword(keyword_ty);
void free_alias(alias_ty);
...@@ -113,6 +113,7 @@ ...@@ -113,6 +113,7 @@
#include "pystate.h" #include "pystate.h"
#include "pyarena.h"
#include "modsupport.h" #include "modsupport.h"
#include "pythonrun.h" #include "pythonrun.h"
#include "ceval.h" #include "ceval.h"
......
...@@ -23,7 +23,7 @@ typedef struct { ...@@ -23,7 +23,7 @@ typedef struct {
void *elements[1]; void *elements[1];
} asdl_seq; } asdl_seq;
asdl_seq *asdl_seq_new(int size); asdl_seq *asdl_seq_new(int size, PyArena *arena);
void asdl_seq_free(asdl_seq *); void asdl_seq_free(asdl_seq *);
#ifdef Py_DEBUG #ifdef Py_DEBUG
......
...@@ -5,7 +5,7 @@ extern "C" { ...@@ -5,7 +5,7 @@ extern "C" {
#endif #endif
PyAPI_FUNC(mod_ty) PyAST_FromNode(const node *, PyCompilerFlags *flags, PyAPI_FUNC(mod_ty) PyAST_FromNode(const node *, PyCompilerFlags *flags,
const char *); const char *, PyArena *);
#ifdef __cplusplus #ifdef __cplusplus
} }
......
...@@ -25,7 +25,7 @@ typedef struct { ...@@ -25,7 +25,7 @@ typedef struct {
struct _mod; /* Declare the existence of this type */ struct _mod; /* Declare the existence of this type */
PyAPI_FUNC(PyCodeObject *) PyAST_Compile(struct _mod *, const char *, PyAPI_FUNC(PyCodeObject *) PyAST_Compile(struct _mod *, const char *,
PyCompilerFlags *); PyCompilerFlags *, PyArena *);
PyAPI_FUNC(PyFutureFeatures *) PyFuture_FromAST(struct _mod *, const char *); PyAPI_FUNC(PyFutureFeatures *) PyFuture_FromAST(struct _mod *, const char *);
#define ERR_LATE_FUTURE \ #define ERR_LATE_FUTURE \
......
/* An arena-like memory interface for the compiler.
*/
#ifndef Py_PYARENA_H
#define Py_PYARENA_H
#ifdef __cplusplus
extern "C" {
#endif
typedef struct _arena PyArena;
/* PyArena_New() and PyArena_Free() create a new arena and free it,
respectively. Once an arena has been created, it can be used
to allocate memory. Once it is freed, all the memory it allocated
is freed and none of its pointers are valid.
PyArena_New() returns an arena pointer. On error, it
returns a negative number and sets an exception.
*/
PyAPI_FUNC(PyArena *) PyArena_New(void);
PyAPI_FUNC(void) PyArena_Free(PyArena *);
PyAPI_FUNC(void *) PyArena_Malloc(PyArena *, size_t);
/* The next two routines aren't proper arena allocation routines.
They exist to experiment with the arena API without making wholesale
changes to the implementation.
The two functions register pointers with the arena id. These
are externally allocated pointers that will be freed when the
arena is freed. One takes a pointer allocated with malloc. The
other takes a PyObject that is DECREFed when the arena is freed.
*/
PyAPI_FUNC(int) PyArena_AddMallocPointer(PyArena *, void *);
PyAPI_FUNC(int) PyArena_AddPyObject(PyArena *, PyObject *);
#ifdef __cplusplus
}
#endif
#endif /* !Py_PYARENA_H */
...@@ -37,10 +37,12 @@ PyAPI_FUNC(int) PyRun_InteractiveOneFlags(FILE *, const char *, PyCompilerFlags ...@@ -37,10 +37,12 @@ PyAPI_FUNC(int) PyRun_InteractiveOneFlags(FILE *, const char *, PyCompilerFlags
PyAPI_FUNC(int) PyRun_InteractiveLoopFlags(FILE *, const char *, PyCompilerFlags *); PyAPI_FUNC(int) PyRun_InteractiveLoopFlags(FILE *, const char *, PyCompilerFlags *);
PyAPI_FUNC(struct _mod *) PyParser_ASTFromString(const char *, const char *, PyAPI_FUNC(struct _mod *) PyParser_ASTFromString(const char *, const char *,
int, PyCompilerFlags *flags); int, PyCompilerFlags *flags,
PyArena *);
PyAPI_FUNC(struct _mod *) PyParser_ASTFromFile(FILE *, const char *, int, PyAPI_FUNC(struct _mod *) PyParser_ASTFromFile(FILE *, const char *, int,
char *, char *, char *, char *,
PyCompilerFlags *, int *); PyCompilerFlags *, int *,
PyArena *);
#define PyParser_SimpleParseString(S, B) \ #define PyParser_SimpleParseString(S, B) \
PyParser_SimpleParseStringFlags(S, B, 0) PyParser_SimpleParseStringFlags(S, B, 0)
#define PyParser_SimpleParseFile(FP, S, B) \ #define PyParser_SimpleParseFile(FP, S, B) \
......
...@@ -254,6 +254,7 @@ PYTHON_OBJS= \ ...@@ -254,6 +254,7 @@ PYTHON_OBJS= \
Python/modsupport.o \ Python/modsupport.o \
Python/mystrtoul.o \ Python/mystrtoul.o \
Python/mysnprintf.o \ Python/mysnprintf.o \
Python/pyarena.o \
Python/pyfpe.o \ Python/pyfpe.o \
Python/pystate.o \ Python/pystate.o \
Python/pythonrun.o \ Python/pythonrun.o \
...@@ -520,6 +521,7 @@ PYTHON_HEADERS= \ ...@@ -520,6 +521,7 @@ PYTHON_HEADERS= \
Include/object.h \ Include/object.h \
Include/objimpl.h \ Include/objimpl.h \
Include/patchlevel.h \ Include/patchlevel.h \
Include/pyarena.h \
Include/pydebug.h \ Include/pydebug.h \
Include/pyerrors.h \ Include/pyerrors.h \
Include/pyfpe.h \ Include/pyfpe.h \
......
...@@ -249,8 +249,9 @@ class PrototypeVisitor(EmitVisitor): ...@@ -249,8 +249,9 @@ class PrototypeVisitor(EmitVisitor):
if args: if args:
argstr = ", ".join(["%s %s" % (atype, aname) argstr = ", ".join(["%s %s" % (atype, aname)
for atype, aname, opt in args]) for atype, aname, opt in args])
argstr += ", PyArena *arena"
else: else:
argstr = "void" argstr = "PyArena *arena"
self.emit("%s %s(%s);" % (ctype, name, argstr), 0) self.emit("%s %s(%s);" % (ctype, name, argstr), 0)
def visitProduct(self, prod, name): def visitProduct(self, prod, name):
...@@ -265,6 +266,10 @@ class FunctionVisitor(PrototypeVisitor): ...@@ -265,6 +266,10 @@ class FunctionVisitor(PrototypeVisitor):
self.emit(s, depth, reflow) self.emit(s, depth, reflow)
argstr = ", ".join(["%s %s" % (atype, aname) argstr = ", ".join(["%s %s" % (atype, aname)
for atype, aname, opt in args + attrs]) for atype, aname, opt in args + attrs])
if argstr:
argstr += ", PyArena *arena"
else:
argstr = "PyArena *arena"
self.emit("%s" % ctype, 0) self.emit("%s" % ctype, 0)
emit("%s(%s)" % (name, argstr)) emit("%s(%s)" % (name, argstr))
emit("{") emit("{")
...@@ -280,7 +285,7 @@ class FunctionVisitor(PrototypeVisitor): ...@@ -280,7 +285,7 @@ class FunctionVisitor(PrototypeVisitor):
emit('return NULL;', 2) emit('return NULL;', 2)
emit('}', 1) emit('}', 1)
emit("p = (%s)malloc(sizeof(*p));" % ctype, 1) emit("p = (%s)PyArena_Malloc(arena, sizeof(*p));" % ctype, 1);
emit("if (!p) {", 1) emit("if (!p) {", 1)
emit("PyErr_NoMemory();", 2) emit("PyErr_NoMemory();", 2)
emit("return NULL;", 2) emit("return NULL;", 2)
...@@ -655,7 +660,7 @@ def main(srcfile): ...@@ -655,7 +660,7 @@ def main(srcfile):
c = ChainOfVisitors(TypeDefVisitor(f), c = ChainOfVisitors(TypeDefVisitor(f),
StructVisitor(f), StructVisitor(f),
PrototypeVisitor(f), PrototypeVisitor(f),
FreePrototypeVisitor(f), ## FreePrototypeVisitor(f),
) )
c.visit(mod) c.visit(mod)
f.close() f.close()
...@@ -671,8 +676,8 @@ def main(srcfile): ...@@ -671,8 +676,8 @@ def main(srcfile):
print >> f print >> f
v = ChainOfVisitors(MarshalPrototypeVisitor(f), v = ChainOfVisitors(MarshalPrototypeVisitor(f),
FunctionVisitor(f), FunctionVisitor(f),
FreeUtilVisitor(f), ## FreeUtilVisitor(f),
FreeVisitor(f), ## FreeVisitor(f),
MarshalUtilVisitor(f), MarshalUtilVisitor(f),
MarshalFunctionVisitor(f), MarshalFunctionVisitor(f),
) )
......
This diff is collapsed.
...@@ -2,17 +2,18 @@ ...@@ -2,17 +2,18 @@
#include "asdl.h" #include "asdl.h"
asdl_seq * asdl_seq *
asdl_seq_new(int size) asdl_seq_new(int size, PyArena *arena)
{ {
asdl_seq *seq = NULL; asdl_seq *seq = NULL;
size_t n = sizeof(asdl_seq) + size_t n = sizeof(asdl_seq) +
(size ? (sizeof(void *) * (size - 1)) : 0); (size ? (sizeof(void *) * (size - 1)) : 0);
seq = (asdl_seq *)PyObject_Malloc(n); seq = (asdl_seq *)malloc(n);
if (!seq) { if (!seq) {
PyErr_NoMemory(); PyErr_NoMemory();
return NULL; return NULL;
} }
PyArena_AddMallocPointer(arena, (void *)seq);
memset(seq, 0, n); memset(seq, 0, n);
seq->size = size; seq->size = size;
return seq; return seq;
...@@ -21,6 +22,4 @@ asdl_seq_new(int size) ...@@ -21,6 +22,4 @@ asdl_seq_new(int size)
void void
asdl_seq_free(asdl_seq *seq) asdl_seq_free(asdl_seq *seq)
{ {
PyObject_Free(seq);
} }
This diff is collapsed.
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include "Python-ast.h" #include "Python-ast.h"
#include "node.h" #include "node.h"
#include "pyarena.h"
#include "ast.h" #include "ast.h"
#include "code.h" #include "code.h"
#include "compile.h" #include "compile.h"
...@@ -148,6 +149,7 @@ struct compiler { ...@@ -148,6 +149,7 @@ struct compiler {
struct compiler_unit *u; /* compiler state for current block */ struct compiler_unit *u; /* compiler state for current block */
PyObject *c_stack; /* Python list holding compiler_unit ptrs */ PyObject *c_stack; /* Python list holding compiler_unit ptrs */
char *c_encoding; /* source encoding (a borrowed reference) */ char *c_encoding; /* source encoding (a borrowed reference) */
PyArena *c_arena; /* pointer to memory allocation arena */
}; };
struct assembler { struct assembler {
...@@ -243,7 +245,8 @@ compiler_init(struct compiler *c) ...@@ -243,7 +245,8 @@ compiler_init(struct compiler *c)
} }
PyCodeObject * PyCodeObject *
PyAST_Compile(mod_ty mod, const char *filename, PyCompilerFlags *flags) PyAST_Compile(mod_ty mod, const char *filename, PyCompilerFlags *flags,
PyArena *arena)
{ {
struct compiler c; struct compiler c;
PyCodeObject *co = NULL; PyCodeObject *co = NULL;
...@@ -259,6 +262,7 @@ PyAST_Compile(mod_ty mod, const char *filename, PyCompilerFlags *flags) ...@@ -259,6 +262,7 @@ PyAST_Compile(mod_ty mod, const char *filename, PyCompilerFlags *flags)
if (!compiler_init(&c)) if (!compiler_init(&c))
goto error; goto error;
c.c_filename = filename; c.c_filename = filename;
c.c_arena = arena;
c.c_future = PyFuture_FromAST(mod, filename); c.c_future = PyFuture_FromAST(mod, filename);
if (c.c_future == NULL) if (c.c_future == NULL)
goto error; goto error;
...@@ -292,12 +296,13 @@ PyAST_Compile(mod_ty mod, const char *filename, PyCompilerFlags *flags) ...@@ -292,12 +296,13 @@ PyAST_Compile(mod_ty mod, const char *filename, PyCompilerFlags *flags)
PyCodeObject * PyCodeObject *
PyNode_Compile(struct _node *n, const char *filename) PyNode_Compile(struct _node *n, const char *filename)
{ {
PyCodeObject *co; PyCodeObject *co = NULL;
mod_ty mod = PyAST_FromNode(n, NULL, filename); PyArena *arena;
if (!mod) arena = PyArena_New();
return NULL; mod_ty mod = PyAST_FromNode(n, NULL, filename, arena);
co = PyAST_Compile(mod, filename, NULL); if (mod)
free_mod(mod); co = PyAST_Compile(mod, filename, NULL, arena);
PyArena_Free(arena);
return co; return co;
} }
...@@ -3404,7 +3409,7 @@ compiler_augassign(struct compiler *c, stmt_ty s) ...@@ -3404,7 +3409,7 @@ compiler_augassign(struct compiler *c, stmt_ty s)
switch (e->kind) { switch (e->kind) {
case Attribute_kind: case Attribute_kind:
auge = Attribute(e->v.Attribute.value, e->v.Attribute.attr, auge = Attribute(e->v.Attribute.value, e->v.Attribute.attr,
AugLoad, e->lineno); AugLoad, e->lineno, c->c_arena);
if (auge == NULL) if (auge == NULL)
return 0; return 0;
VISIT(c, expr, auge); VISIT(c, expr, auge);
...@@ -3412,11 +3417,10 @@ compiler_augassign(struct compiler *c, stmt_ty s) ...@@ -3412,11 +3417,10 @@ compiler_augassign(struct compiler *c, stmt_ty s)
ADDOP(c, inplace_binop(c, s->v.AugAssign.op)); ADDOP(c, inplace_binop(c, s->v.AugAssign.op));
auge->v.Attribute.ctx = AugStore; auge->v.Attribute.ctx = AugStore;
VISIT(c, expr, auge); VISIT(c, expr, auge);
free(auge);
break; break;
case Subscript_kind: case Subscript_kind:
auge = Subscript(e->v.Subscript.value, e->v.Subscript.slice, auge = Subscript(e->v.Subscript.value, e->v.Subscript.slice,
AugLoad, e->lineno); AugLoad, e->lineno, c->c_arena);
if (auge == NULL) if (auge == NULL)
return 0; return 0;
VISIT(c, expr, auge); VISIT(c, expr, auge);
...@@ -3424,7 +3428,6 @@ compiler_augassign(struct compiler *c, stmt_ty s) ...@@ -3424,7 +3428,6 @@ compiler_augassign(struct compiler *c, stmt_ty s)
ADDOP(c, inplace_binop(c, s->v.AugAssign.op)); ADDOP(c, inplace_binop(c, s->v.AugAssign.op));
auge->v.Subscript.ctx = AugStore; auge->v.Subscript.ctx = AugStore;
VISIT(c, expr, auge); VISIT(c, expr, auge);
free(auge);
break; break;
case Name_kind: case Name_kind:
VISIT(c, expr, s->v.AugAssign.target); VISIT(c, expr, s->v.AugAssign.target);
......
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
#include "Python.h" #include "Python.h"
#include "Python-ast.h" #include "Python-ast.h"
#include "pyarena.h"
#include "pythonrun.h" #include "pythonrun.h"
#include "errcode.h" #include "errcode.h"
#include "marshal.h" #include "marshal.h"
...@@ -773,13 +774,14 @@ parse_source_module(const char *pathname, FILE *fp) ...@@ -773,13 +774,14 @@ parse_source_module(const char *pathname, FILE *fp)
{ {
PyCodeObject *co = NULL; PyCodeObject *co = NULL;
mod_ty mod; mod_ty mod;
PyArena *arena = PyArena_New();
mod = PyParser_ASTFromFile(fp, pathname, Py_file_input, 0, 0, 0, mod = PyParser_ASTFromFile(fp, pathname, Py_file_input, 0, 0, 0,
NULL); NULL, arena);
if (mod) { if (mod) {
co = PyAST_Compile(mod, pathname, NULL); co = PyAST_Compile(mod, pathname, NULL, arena);
free_mod(mod);
} }
PyArena_Free(arena);
return co; return co;
} }
......
#include "Python.h"
#include "pyarena.h"
/* An arena list is a linked list that can store either pointers or
PyObjects. The type is clear from context.
*/
typedef struct _arena_list {
struct _arena_list *al_next;
void *al_pointer;
} PyArenaList;
/* There are two linked lists in an arena, one for malloc pointers and
one for PyObject. For each list, there is a pointer to the head
and to the tail. The head is used to free the list. The tail is
used to add a new element to the list.
The list always keeps one un-used node at the end of the list.
*/
struct _arena {
PyArenaList *a_malloc_head;
PyArenaList *a_malloc_tail;
PyArenaList *a_object_head;
PyArenaList *a_object_tail;
};
static PyArenaList*
PyArenaList_New(void)
{
PyArenaList *alist = (PyArenaList *)malloc(sizeof(PyArenaList));
if (!alist)
return NULL;
alist->al_next = NULL;
alist->al_pointer = NULL;
return alist;
}
static void
PyArenaList_FreeObject(PyArenaList *alist)
{
if (!alist)
return;
while (alist) {
PyArenaList *prev;
Py_XDECREF((PyObject *)alist->al_pointer);
alist->al_pointer = NULL;
prev = alist;
alist = alist->al_next;
free(prev);
}
}
static void
PyArenaList_FreeMalloc(PyArenaList *alist)
{
if (!alist)
return;
while (alist) {
PyArenaList *prev;
if (alist->al_pointer) {
free(alist->al_pointer);
}
alist->al_pointer = NULL;
prev = alist;
alist = alist->al_next;
free(prev);
}
}
PyArena *
PyArena_New()
{
PyArena* arena = (PyArena *)malloc(sizeof(PyArena));
if (!arena)
return NULL;
arena->a_object_head = PyArenaList_New();
arena->a_object_tail = arena->a_object_head;
arena->a_malloc_head = PyArenaList_New();
arena->a_malloc_tail = arena->a_malloc_head;
return arena;
}
void
PyArena_Free(PyArena *arena)
{
assert(arena);
PyArenaList_FreeObject(arena->a_object_head);
PyArenaList_FreeMalloc(arena->a_malloc_head);
free(arena);
}
void *
PyArena_Malloc(PyArena *arena, size_t size)
{
/* A better implementation might actually use an arena. The current
approach is just a trivial implementation of the API that allows
it to be tested.
*/
void *p;
assert(size != 0);
p = malloc(size);
PyArena_AddMallocPointer(arena, p);
return p;
}
int
PyArena_AddMallocPointer(PyArena *arena, void *pointer)
{
assert(pointer);
PyArenaList *tail = arena->a_malloc_tail;
assert(tail->al_pointer != pointer);
tail->al_next = PyArenaList_New();
tail->al_pointer = pointer;
arena->a_malloc_tail = tail->al_next;
return 1;
}
int
PyArena_AddPyObject(PyArena *arena, PyObject *pointer)
{
assert(pointer);
PyArenaList *tail = arena->a_object_tail;
tail->al_next = PyArenaList_New();
tail->al_pointer = pointer;
arena->a_object_tail = tail->al_next;
return 1;
}
...@@ -12,6 +12,7 @@ ...@@ -12,6 +12,7 @@
#include "code.h" #include "code.h"
#include "compile.h" #include "compile.h"
#include "symtable.h" #include "symtable.h"
#include "pyarena.h"
#include "ast.h" #include "ast.h"
#include "eval.h" #include "eval.h"
#include "marshal.h" #include "marshal.h"
...@@ -36,9 +37,9 @@ extern grammar _PyParser_Grammar; /* From graminit.c */ ...@@ -36,9 +37,9 @@ extern grammar _PyParser_Grammar; /* From graminit.c */
static void initmain(void); static void initmain(void);
static void initsite(void); static void initsite(void);
static PyObject *run_err_mod(mod_ty, const char *, PyObject *, PyObject *, static PyObject *run_err_mod(mod_ty, const char *, PyObject *, PyObject *,
PyCompilerFlags *); PyCompilerFlags *, PyArena *arena);
static PyObject *run_mod(mod_ty, const char *, PyObject *, PyObject *, static PyObject *run_mod(mod_ty, const char *, PyObject *, PyObject *,
PyCompilerFlags *); PyCompilerFlags *, PyArena *);
static PyObject *run_pyc_file(FILE *, const char *, PyObject *, PyObject *, static PyObject *run_pyc_file(FILE *, const char *, PyObject *, PyObject *,
PyCompilerFlags *); PyCompilerFlags *);
static void err_input(perrdetail *); static void err_input(perrdetail *);
...@@ -697,6 +698,7 @@ PyRun_InteractiveOneFlags(FILE *fp, const char *filename, PyCompilerFlags *flags ...@@ -697,6 +698,7 @@ PyRun_InteractiveOneFlags(FILE *fp, const char *filename, PyCompilerFlags *flags
{ {
PyObject *m, *d, *v, *w; PyObject *m, *d, *v, *w;
mod_ty mod; mod_ty mod;
PyArena *arena;
char *ps1 = "", *ps2 = ""; char *ps1 = "", *ps2 = "";
int errcode = 0; int errcode = 0;
...@@ -716,12 +718,14 @@ PyRun_InteractiveOneFlags(FILE *fp, const char *filename, PyCompilerFlags *flags ...@@ -716,12 +718,14 @@ PyRun_InteractiveOneFlags(FILE *fp, const char *filename, PyCompilerFlags *flags
else if (PyString_Check(w)) else if (PyString_Check(w))
ps2 = PyString_AsString(w); ps2 = PyString_AsString(w);
} }
arena = PyArena_New();
mod = PyParser_ASTFromFile(fp, filename, mod = PyParser_ASTFromFile(fp, filename,
Py_single_input, ps1, ps2, Py_single_input, ps1, ps2,
flags, &errcode); flags, &errcode, arena);
Py_XDECREF(v); Py_XDECREF(v);
Py_XDECREF(w); Py_XDECREF(w);
if (mod == NULL) { if (mod == NULL) {
PyArena_Free(arena);
if (errcode == E_EOF) { if (errcode == E_EOF) {
PyErr_Clear(); PyErr_Clear();
return E_EOF; return E_EOF;
...@@ -730,11 +734,13 @@ PyRun_InteractiveOneFlags(FILE *fp, const char *filename, PyCompilerFlags *flags ...@@ -730,11 +734,13 @@ PyRun_InteractiveOneFlags(FILE *fp, const char *filename, PyCompilerFlags *flags
return -1; return -1;
} }
m = PyImport_AddModule("__main__"); m = PyImport_AddModule("__main__");
if (m == NULL) if (m == NULL) {
PyArena_Free(arena);
return -1; return -1;
}
d = PyModule_GetDict(m); d = PyModule_GetDict(m);
v = run_mod(mod, filename, d, d, flags); v = run_mod(mod, filename, d, d, flags, arena);
free_mod(mod); PyArena_Free(arena);
if (v == NULL) { if (v == NULL) {
PyErr_Print(); PyErr_Print();
return -1; return -1;
...@@ -1155,9 +1161,11 @@ PyRun_StringFlags(const char *str, int start, PyObject *globals, ...@@ -1155,9 +1161,11 @@ PyRun_StringFlags(const char *str, int start, PyObject *globals,
PyObject *locals, PyCompilerFlags *flags) PyObject *locals, PyCompilerFlags *flags)
{ {
PyObject *ret; PyObject *ret;
mod_ty mod = PyParser_ASTFromString(str, "<string>", start, flags); PyArena *arena = PyArena_New();
ret = run_err_mod(mod, "<string>", globals, locals, flags); mod_ty mod = PyParser_ASTFromString(str, "<string>", start, flags,
free_mod(mod); arena);
ret = run_err_mod(mod, "<string>", globals, locals, flags, arena);
PyArena_Free(arena);
return ret; return ret;
} }
...@@ -1166,33 +1174,36 @@ PyRun_FileExFlags(FILE *fp, const char *filename, int start, PyObject *globals, ...@@ -1166,33 +1174,36 @@ PyRun_FileExFlags(FILE *fp, const char *filename, int start, PyObject *globals,
PyObject *locals, int closeit, PyCompilerFlags *flags) PyObject *locals, int closeit, PyCompilerFlags *flags)
{ {
PyObject *ret; PyObject *ret;
PyArena *arena = PyArena_New();
mod_ty mod = PyParser_ASTFromFile(fp, filename, start, 0, 0, mod_ty mod = PyParser_ASTFromFile(fp, filename, start, 0, 0,
flags, NULL); flags, NULL, arena);
if (mod == NULL) if (mod == NULL) {
PyArena_Free(arena);
return NULL; return NULL;
}
if (closeit) if (closeit)
fclose(fp); fclose(fp);
ret = run_err_mod(mod, filename, globals, locals, flags); ret = run_err_mod(mod, filename, globals, locals, flags, arena);
free_mod(mod); PyArena_Free(arena);
return ret; return ret;
} }
static PyObject * static PyObject *
run_err_mod(mod_ty mod, const char *filename, PyObject *globals, run_err_mod(mod_ty mod, const char *filename, PyObject *globals,
PyObject *locals, PyCompilerFlags *flags) PyObject *locals, PyCompilerFlags *flags, PyArena *arena)
{ {
if (mod == NULL) if (mod == NULL)
return NULL; return NULL;
return run_mod(mod, filename, globals, locals, flags); return run_mod(mod, filename, globals, locals, flags, arena);
} }
static PyObject * static PyObject *
run_mod(mod_ty mod, const char *filename, PyObject *globals, PyObject *locals, run_mod(mod_ty mod, const char *filename, PyObject *globals, PyObject *locals,
PyCompilerFlags *flags) PyCompilerFlags *flags, PyArena *arena)
{ {
PyCodeObject *co; PyCodeObject *co;
PyObject *v; PyObject *v;
co = PyAST_Compile(mod, filename, flags); co = PyAST_Compile(mod, filename, flags, arena);
if (co == NULL) if (co == NULL)
return NULL; return NULL;
v = PyEval_EvalCode(co, globals, locals); v = PyEval_EvalCode(co, globals, locals);
...@@ -1236,43 +1247,45 @@ PyObject * ...@@ -1236,43 +1247,45 @@ PyObject *
Py_CompileStringFlags(const char *str, const char *filename, int start, Py_CompileStringFlags(const char *str, const char *filename, int start,
PyCompilerFlags *flags) PyCompilerFlags *flags)
{ {
mod_ty mod;
PyCodeObject *co; PyCodeObject *co;
mod = PyParser_ASTFromString(str, filename, start, flags); PyArena *arena = PyArena_New();
if (mod == NULL) mod_ty mod = PyParser_ASTFromString(str, filename, start, flags, arena);
if (mod == NULL) {
PyArena_Free(arena);
return NULL; return NULL;
co = PyAST_Compile(mod, filename, flags); }
free_mod(mod); co = PyAST_Compile(mod, filename, flags, arena);
PyArena_Free(arena);
return (PyObject *)co; return (PyObject *)co;
} }
struct symtable * struct symtable *
Py_SymtableString(const char *str, const char *filename, int start) Py_SymtableString(const char *str, const char *filename, int start)
{ {
mod_ty mod;
struct symtable *st; struct symtable *st;
PyArena *arena = PyArena_New();
mod = PyParser_ASTFromString(str, filename, start, NULL); mod_ty mod = PyParser_ASTFromString(str, filename, start, NULL, arena);
if (mod == NULL) if (mod == NULL) {
PyArena_Free(arena);
return NULL; return NULL;
}
st = PySymtable_Build(mod, filename, 0); st = PySymtable_Build(mod, filename, 0);
free_mod(mod); PyArena_Free(arena);
return st; return st;
} }
/* Preferred access to parser is through AST. */ /* Preferred access to parser is through AST. */
mod_ty mod_ty
PyParser_ASTFromString(const char *s, const char *filename, int start, PyParser_ASTFromString(const char *s, const char *filename, int start,
PyCompilerFlags *flags) PyCompilerFlags *flags, PyArena *arena)
{ {
node *n;
mod_ty mod; mod_ty mod;
perrdetail err; perrdetail err;
n = PyParser_ParseStringFlagsFilename(s, filename, &_PyParser_Grammar, node *n = PyParser_ParseStringFlagsFilename(s, filename,
start, &err, &_PyParser_Grammar, start, &err,
PARSER_FLAGS(flags)); PARSER_FLAGS(flags));
if (n) { if (n) {
mod = PyAST_FromNode(n, flags, filename); mod = PyAST_FromNode(n, flags, filename, arena);
PyNode_Free(n); PyNode_Free(n);
return mod; return mod;
} }
...@@ -1284,15 +1297,15 @@ PyParser_ASTFromString(const char *s, const char *filename, int start, ...@@ -1284,15 +1297,15 @@ PyParser_ASTFromString(const char *s, const char *filename, int start,
mod_ty mod_ty
PyParser_ASTFromFile(FILE *fp, const char *filename, int start, char *ps1, PyParser_ASTFromFile(FILE *fp, const char *filename, int start, char *ps1,
char *ps2, PyCompilerFlags *flags, int *errcode) char *ps2, PyCompilerFlags *flags, int *errcode,
PyArena *arena)
{ {
node *n;
mod_ty mod; mod_ty mod;
perrdetail err; perrdetail err;
n = PyParser_ParseFileFlags(fp, filename, &_PyParser_Grammar, start, node *n = PyParser_ParseFileFlags(fp, filename, &_PyParser_Grammar,
ps1, ps2, &err, PARSER_FLAGS(flags)); start, ps1, ps2, &err, PARSER_FLAGS(flags));
if (n) { if (n) {
mod = PyAST_FromNode(n, flags, filename); mod = PyAST_FromNode(n, flags, filename, arena);
PyNode_Free(n); PyNode_Free(n);
return mod; return mod;
} }
...@@ -1309,10 +1322,9 @@ PyParser_ASTFromFile(FILE *fp, const char *filename, int start, char *ps1, ...@@ -1309,10 +1322,9 @@ PyParser_ASTFromFile(FILE *fp, const char *filename, int start, char *ps1,
node * node *
PyParser_SimpleParseFileFlags(FILE *fp, const char *filename, int start, int flags) PyParser_SimpleParseFileFlags(FILE *fp, const char *filename, int start, int flags)
{ {
node *n;
perrdetail err; perrdetail err;
n = PyParser_ParseFileFlags(fp, filename, &_PyParser_Grammar, start, node *n = PyParser_ParseFileFlags(fp, filename, &_PyParser_Grammar,
(char *)0, (char *)0, &err, flags); start, NULL, NULL, &err, flags);
if (n == NULL) if (n == NULL)
err_input(&err); err_input(&err);
...@@ -1324,10 +1336,9 @@ PyParser_SimpleParseFileFlags(FILE *fp, const char *filename, int start, int fla ...@@ -1324,10 +1336,9 @@ PyParser_SimpleParseFileFlags(FILE *fp, const char *filename, int start, int fla
node * node *
PyParser_SimpleParseStringFlags(const char *str, int start, int flags) PyParser_SimpleParseStringFlags(const char *str, int start, int flags)
{ {
node *n;
perrdetail err; perrdetail err;
n = PyParser_ParseStringFlags(str, &_PyParser_Grammar, start, &err, node *n = PyParser_ParseStringFlags(str, &_PyParser_Grammar,
flags); start, &err, flags);
if (n == NULL) if (n == NULL)
err_input(&err); err_input(&err);
return n; return n;
...@@ -1337,12 +1348,9 @@ node * ...@@ -1337,12 +1348,9 @@ node *
PyParser_SimpleParseStringFlagsFilename(const char *str, const char *filename, PyParser_SimpleParseStringFlagsFilename(const char *str, const char *filename,
int start, int flags) int start, int flags)
{ {
node *n;
perrdetail err; perrdetail err;
node *n = PyParser_ParseStringFlagsFilename(str, filename,
n = PyParser_ParseStringFlagsFilename(str, filename, &_PyParser_Grammar, start, &err, flags);
&_PyParser_Grammar,
start, &err, flags);
if (n == NULL) if (n == NULL)
err_input(&err); err_input(&err);
return n; return n;
...@@ -1351,8 +1359,7 @@ PyParser_SimpleParseStringFlagsFilename(const char *str, const char *filename, ...@@ -1351,8 +1359,7 @@ PyParser_SimpleParseStringFlagsFilename(const char *str, const char *filename,
node * node *
PyParser_SimpleParseStringFilename(const char *str, const char *filename, int start) PyParser_SimpleParseStringFilename(const char *str, const char *filename, int start)
{ {
return PyParser_SimpleParseStringFlagsFilename(str, filename, return PyParser_SimpleParseStringFlagsFilename(str, filename, start, 0);
start, 0);
} }
/* May want to move a more generalized form of this to parsetok.c or /* May want to move a more generalized form of this to parsetok.c or
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment