| 
									
										
										
										
											2000-02-08 19:01:29 +00:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | import marshal | 
					
						
							| 
									
										
										
										
											2000-02-08 19:01:29 +00:00
										 |  |  | import stat | 
					
						
							|  |  |  | import struct | 
					
						
							| 
									
										
										
										
											2000-02-16 00:55:44 +00:00
										 |  |  | import types | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | from cStringIO import StringIO | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | from compiler import ast, parse, walk | 
					
						
							|  |  |  | from compiler import pyassem, misc | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  | from compiler.pyassem import CO_VARARGS, CO_VARKEYWORDS, CO_NEWLOCALS, TupleArg | 
					
						
							| 
									
										
										
										
											2000-02-11 20:27:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-02 22:32:59 +00:00
										 |  |  | callfunc_opcode_info = { | 
					
						
							|  |  |  |     # (Have *args, Have **args) : opcode | 
					
						
							|  |  |  |     (0,0) : "CALL_FUNCTION", | 
					
						
							|  |  |  |     (1,0) : "CALL_FUNCTION_VAR", | 
					
						
							|  |  |  |     (0,1) : "CALL_FUNCTION_KW", | 
					
						
							|  |  |  |     (1,1) : "CALL_FUNCTION_VAR_KW", | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | def compile(filename): | 
					
						
							|  |  |  |     f = open(filename) | 
					
						
							|  |  |  |     buf = f.read() | 
					
						
							|  |  |  |     f.close() | 
					
						
							|  |  |  |     mod = Module(buf, filename) | 
					
						
							|  |  |  |     mod.compile() | 
					
						
							|  |  |  |     f = open(filename + "c", "wb") | 
					
						
							|  |  |  |     mod.dump(f) | 
					
						
							|  |  |  |     f.close() | 
					
						
							| 
									
										
										
										
											2000-02-16 00:55:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | class Module: | 
					
						
							|  |  |  |     def __init__(self, source, filename): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.filename = filename | 
					
						
							|  |  |  |         self.source = source | 
					
						
							|  |  |  |         self.code = None | 
					
						
							| 
									
										
										
										
											2000-02-16 00:55:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def compile(self): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         ast = parse(self.source) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         root, filename = os.path.split(self.filename) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         gen = ModuleCodeGenerator(filename) | 
					
						
							|  |  |  |         walk(ast, gen, 1) | 
					
						
							|  |  |  |         self.code = gen.getCode() | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def dump(self, f): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         f.write(self.getPycHeader()) | 
					
						
							|  |  |  |         marshal.dump(self.code, f) | 
					
						
							| 
									
										
										
										
											2000-02-12 00:12:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |     MAGIC = (50811 | (ord('\r')<<16) | (ord('\n')<<24)) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def getPycHeader(self): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         # compile.c uses marshal to write a long directly, with | 
					
						
							|  |  |  |         # calling the interface that would also generate a 1-byte code | 
					
						
							|  |  |  |         # to indicate the type of the value.  simplest way to get the | 
					
						
							|  |  |  |         # same effect is to call marshal and then skip the code. | 
					
						
							|  |  |  |         magic = marshal.dumps(self.MAGIC)[1:] | 
					
						
							|  |  |  |         mtime = os.stat(self.filename)[stat.ST_MTIME] | 
					
						
							|  |  |  |         mtime = struct.pack('i', mtime) | 
					
						
							|  |  |  |         return magic + mtime | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class CodeGenerator: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     optimized = 0 # is namespace access optimized? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							|  |  |  | ## Subclasses must define a constructor that intializes self.graph | 
					
						
							|  |  |  | ## before calling this init function | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  | ##         self.graph = pyassem.PyFlowGraph() | 
					
						
							|  |  |  |         self.filename = filename | 
					
						
							|  |  |  |         self.locals = misc.Stack() | 
					
						
							|  |  |  |         self.loops = misc.Stack() | 
					
						
							|  |  |  |         self.curStack = 0 | 
					
						
							|  |  |  |         self.maxStack = 0 | 
					
						
							|  |  |  |         self._setupGraphDelegation() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _setupGraphDelegation(self): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.emit = self.graph.emit | 
					
						
							|  |  |  |         self.newBlock = self.graph.newBlock | 
					
						
							|  |  |  |         self.startBlock = self.graph.startBlock | 
					
						
							|  |  |  |         self.nextBlock = self.graph.nextBlock | 
					
						
							|  |  |  |         self.setDocstring = self.graph.setDocstring | 
					
						
							| 
									
										
										
										
											2000-02-11 20:27:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def getCode(self): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         """Return a code object""" | 
					
						
							|  |  |  |         return self.graph.getCode() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Next five methods handle name access | 
					
						
							| 
									
										
										
										
											2000-02-08 19:01:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |     def isLocalName(self, name): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.locals.top().has_elt(name) | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-10 17:20:39 +00:00
										 |  |  |     def storeName(self, name): | 
					
						
							|  |  |  |         self._nameOp('STORE', name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def loadName(self, name): | 
					
						
							|  |  |  |         self._nameOp('LOAD', name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def delName(self, name): | 
					
						
							|  |  |  |         self._nameOp('DELETE', name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def _nameOp(self, prefix, name): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         if not self.optimized: | 
					
						
							|  |  |  |             self.emit(prefix + '_NAME', name) | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         if self.isLocalName(name): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |             self.emit(prefix + '_FAST', name) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self.emit(prefix + '_GLOBAL', name) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |     def set_lineno(self, node): | 
					
						
							|  |  |  |         """Emit SET_LINENO if node has lineno attribute
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Returns true if SET_LINENO was emitted. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         There are no rules for when an AST node should have a lineno | 
					
						
							|  |  |  |         attribute.  The transformer and AST code need to be reviewed | 
					
						
							|  |  |  |         and a consistent policy implemented and documented.  Until | 
					
						
							|  |  |  |         then, this method works around missing line numbers. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         lineno = getattr(node, 'lineno', None) | 
					
						
							|  |  |  |         if lineno is not None: | 
					
						
							|  |  |  |             self.emit('SET_LINENO', lineno) | 
					
						
							|  |  |  |             return 1 | 
					
						
							|  |  |  |         return 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     # The first few visitor methods handle nodes that generator new | 
					
						
							|  |  |  |     # code objects  | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  |     def visitModule(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         lnf = walk(node.node, LocalNameFinder(), 0) | 
					
						
							|  |  |  |         self.locals.push(lnf.getLocals()) | 
					
						
							|  |  |  |         self.setDocstring(node.doc) | 
					
						
							|  |  |  |         self.visit(node.node) | 
					
						
							|  |  |  |         self.emit('LOAD_CONST', None) | 
					
						
							|  |  |  |         self.emit('RETURN_VALUE') | 
					
						
							| 
									
										
										
										
											2000-02-04 19:37:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitFunction(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self._visitFuncOrLambda(node, isLambda=0) | 
					
						
							|  |  |  |         self.storeName(node.name) | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitLambda(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self._visitFuncOrLambda(node, isLambda=1) | 
					
						
							|  |  |  | ##        self.storeName("<lambda>") | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _visitFuncOrLambda(self, node, isLambda): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         gen = FunctionCodeGenerator(node, self.filename, isLambda) | 
					
						
							|  |  |  |         walk(node.code, gen) | 
					
						
							|  |  |  |         gen.finish() | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         for default in node.defaults: | 
					
						
							|  |  |  |             self.visit(default) | 
					
						
							|  |  |  |         self.emit('LOAD_CONST', gen.getCode()) | 
					
						
							|  |  |  |         self.emit('MAKE_FUNCTION', len(node.defaults)) | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-06 19:10:54 +00:00
										 |  |  |     def visitClass(self, node): | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         gen = ClassCodeGenerator(node, self.filename) | 
					
						
							|  |  |  |         walk(node.code, gen) | 
					
						
							|  |  |  |         gen.finish() | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('LOAD_CONST', node.name) | 
					
						
							| 
									
										
										
										
											2000-02-12 00:12:38 +00:00
										 |  |  |         for base in node.bases: | 
					
						
							|  |  |  |             self.visit(base) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('BUILD_TUPLE', len(node.bases)) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('LOAD_CONST', gen.getCode()) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('MAKE_FUNCTION', 0) | 
					
						
							|  |  |  |         self.emit('CALL_FUNCTION', 0) | 
					
						
							|  |  |  |         self.emit('BUILD_CLASS') | 
					
						
							| 
									
										
										
										
											2000-02-12 00:12:38 +00:00
										 |  |  |         self.storeName(node.name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     # The rest are standard visitor methods | 
					
						
							| 
									
										
										
										
											2000-02-11 20:27:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     # The next few implement control-flow statements | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitIf(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         end = self.newBlock() | 
					
						
							|  |  |  |         numtests = len(node.tests) | 
					
						
							|  |  |  |         for i in range(numtests): | 
					
						
							|  |  |  |             test, suite = node.tests[i] | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |             self.set_lineno(test) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |             self.visit(test) | 
					
						
							|  |  |  | ##            if i == numtests - 1 and not node.else_: | 
					
						
							|  |  |  | ##                nextTest = end | 
					
						
							|  |  |  | ##            else: | 
					
						
							|  |  |  | ##                nextTest = self.newBlock() | 
					
						
							|  |  |  |             nextTest = self.newBlock() | 
					
						
							|  |  |  |             self.emit('JUMP_IF_FALSE', nextTest) | 
					
						
							|  |  |  |             self.nextBlock() | 
					
						
							|  |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |             self.visit(suite) | 
					
						
							|  |  |  |             self.emit('JUMP_FORWARD', end) | 
					
						
							|  |  |  |             self.nextBlock(nextTest) | 
					
						
							|  |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |         if node.else_: | 
					
						
							|  |  |  |             self.visit(node.else_) | 
					
						
							|  |  |  |         self.nextBlock(end) | 
					
						
							| 
									
										
										
										
											2000-02-11 20:27:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitWhile(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         loop = self.newBlock() | 
					
						
							|  |  |  |         else_ = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-02-12 00:12:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         after = self.newBlock() | 
					
						
							|  |  |  |         self.emit('SETUP_LOOP', after) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(loop) | 
					
						
							|  |  |  |         self.loops.push(loop) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.visit(node.test) | 
					
						
							|  |  |  |         self.emit('JUMP_IF_FALSE', else_ or after) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.nextBlock() | 
					
						
							|  |  |  |         self.emit('POP_TOP') | 
					
						
							|  |  |  |         self.visit(node.body) | 
					
						
							|  |  |  |         self.emit('JUMP_ABSOLUTE', loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.startBlock(else_) # or just the POPs if not else clause | 
					
						
							|  |  |  |         self.emit('POP_TOP') | 
					
						
							|  |  |  |         self.emit('POP_BLOCK') | 
					
						
							|  |  |  |         if node.else_: | 
					
						
							|  |  |  |             self.visit(node.else_) | 
					
						
							|  |  |  |         self.loops.pop() | 
					
						
							|  |  |  |         self.nextBlock(after) | 
					
						
							| 
									
										
										
										
											2000-02-10 20:55:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |     def visitFor(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         start = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         anchor = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         after = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.loops.push(start) | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.emit('SETUP_LOOP', after) | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |         self.visit(node.list) | 
					
						
							|  |  |  |         self.visit(ast.Const(0)) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(start) | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('FOR_LOOP', anchor) | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |         self.visit(node.assign) | 
					
						
							|  |  |  |         self.visit(node.body) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('JUMP_ABSOLUTE', start) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(anchor) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('POP_BLOCK') | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |         if node.else_: | 
					
						
							|  |  |  |             self.visit(node.else_) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.loops.pop() | 
					
						
							|  |  |  |         self.nextBlock(after) | 
					
						
							| 
									
										
										
										
											2000-02-10 20:55:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitBreak(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         if not self.loops: | 
					
						
							|  |  |  |             raise SyntaxError, "'break' outside loop (%s, %d)" % \ | 
					
						
							|  |  |  |                   (self.filename, node.lineno) | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.emit('BREAK_LOOP') | 
					
						
							| 
									
										
										
										
											2000-02-10 20:55:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitContinue(self, node): | 
					
						
							|  |  |  |         if not self.loops: | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |             raise SyntaxError, "'continue' outside loop (%s, %d)" % \ | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |                   (self.filename, node.lineno) | 
					
						
							| 
									
										
										
										
											2000-02-10 20:55:50 +00:00
										 |  |  |         l = self.loops.top() | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('JUMP_ABSOLUTE', l) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitTest(self, node, jump): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         end = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         for child in node.nodes[:-1]: | 
					
						
							|  |  |  |             self.visit(child) | 
					
						
							|  |  |  |             self.emit(jump, end) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |             self.nextBlock() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |         self.visit(node.nodes[-1]) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(end) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitAnd(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.visitTest(node, 'JUMP_IF_FALSE') | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitOr(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.visitTest(node, 'JUMP_IF_TRUE') | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitCompare(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.visit(node.expr) | 
					
						
							|  |  |  |         cleanup = self.newBlock() | 
					
						
							|  |  |  |         for op, code in node.ops[:-1]: | 
					
						
							|  |  |  |             self.visit(code) | 
					
						
							|  |  |  |             self.emit('DUP_TOP') | 
					
						
							|  |  |  |             self.emit('ROT_THREE') | 
					
						
							|  |  |  |             self.emit('COMPARE_OP', op) | 
					
						
							|  |  |  |             self.emit('JUMP_IF_FALSE', cleanup) | 
					
						
							|  |  |  |             self.nextBlock() | 
					
						
							|  |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |         # now do the last comparison | 
					
						
							|  |  |  |         if node.ops: | 
					
						
							|  |  |  |             op, code = node.ops[-1] | 
					
						
							|  |  |  |             self.visit(code) | 
					
						
							|  |  |  |             self.emit('COMPARE_OP', op) | 
					
						
							|  |  |  |         if len(node.ops) > 1: | 
					
						
							|  |  |  |             end = self.newBlock() | 
					
						
							|  |  |  |             self.emit('JUMP_FORWARD', end) | 
					
						
							|  |  |  |             self.nextBlock(cleanup) | 
					
						
							|  |  |  |             self.emit('ROT_TWO') | 
					
						
							|  |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |             self.nextBlock(end) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # exception related | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitAssert(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         # XXX would be interesting to implement this via a | 
					
						
							|  |  |  |         # transformation of the AST before this stage | 
					
						
							|  |  |  |         end = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         # XXX __debug__ and AssertionError appear to be special cases | 
					
						
							|  |  |  |         # -- they are always loaded as globals even if there are local | 
					
						
							|  |  |  |         # names.  I guess this is a sort of renaming op. | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.emit('LOAD_GLOBAL', '__debug__') | 
					
						
							|  |  |  |         self.emit('JUMP_IF_FALSE', end) | 
					
						
							|  |  |  |         self.nextBlock() | 
					
						
							|  |  |  |         self.emit('POP_TOP') | 
					
						
							|  |  |  |         self.visit(node.test) | 
					
						
							|  |  |  |         self.emit('JUMP_IF_TRUE', end) | 
					
						
							|  |  |  |         self.nextBlock() | 
					
						
							|  |  |  |         self.emit('LOAD_GLOBAL', 'AssertionError') | 
					
						
							|  |  |  |         self.visit(node.fail) | 
					
						
							|  |  |  |         self.emit('RAISE_VARARGS', 2) | 
					
						
							|  |  |  |         self.nextBlock(end) | 
					
						
							|  |  |  |         self.emit('POP_TOP') | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitRaise(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         n = 0 | 
					
						
							|  |  |  |         if node.expr1: | 
					
						
							|  |  |  |             self.visit(node.expr1) | 
					
						
							|  |  |  |             n = n + 1 | 
					
						
							|  |  |  |         if node.expr2: | 
					
						
							|  |  |  |             self.visit(node.expr2) | 
					
						
							|  |  |  |             n = n + 1 | 
					
						
							|  |  |  |         if node.expr3: | 
					
						
							|  |  |  |             self.visit(node.expr3) | 
					
						
							|  |  |  |             n = n + 1 | 
					
						
							|  |  |  |         self.emit('RAISE_VARARGS', n) | 
					
						
							| 
									
										
										
										
											2000-02-10 20:55:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |     def visitTryExcept(self, node): | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         handlers = self.newBlock() | 
					
						
							|  |  |  |         end = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |         if node.else_: | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |             lElse = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             lElse = end | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |         self.emit('SETUP_EXCEPT', handlers) | 
					
						
							|  |  |  |         self.visit(node.body) | 
					
						
							|  |  |  |         self.emit('POP_BLOCK') | 
					
						
							|  |  |  |         self.emit('JUMP_FORWARD', lElse) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(handlers) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |          | 
					
						
							|  |  |  |         last = len(node.handlers) - 1 | 
					
						
							|  |  |  |         for i in range(len(node.handlers)): | 
					
						
							|  |  |  |             expr, target, body = node.handlers[i] | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |             self.set_lineno(expr) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |             if expr: | 
					
						
							|  |  |  |                 self.emit('DUP_TOP') | 
					
						
							|  |  |  |                 self.visit(expr) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |                 self.emit('COMPARE_OP', 'exception match') | 
					
						
							|  |  |  |                 next = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |                 self.emit('JUMP_IF_FALSE', next) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |                 self.nextBlock() | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |                 self.emit('POP_TOP') | 
					
						
							|  |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |             if target: | 
					
						
							|  |  |  |                 self.visit(target) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.emit('POP_TOP') | 
					
						
							|  |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |             self.visit(body) | 
					
						
							|  |  |  |             self.emit('JUMP_FORWARD', end) | 
					
						
							| 
									
										
										
										
											2000-02-16 00:55:44 +00:00
										 |  |  |             if expr: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |                 self.nextBlock(next) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |             self.emit('POP_TOP') | 
					
						
							|  |  |  |         self.emit('END_FINALLY') | 
					
						
							|  |  |  |         if node.else_: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |             self.nextBlock(lElse) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |             self.visit(node.else_) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(end) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |      | 
					
						
							|  |  |  |     def visitTryFinally(self, node): | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         final = self.newBlock() | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |         self.emit('SETUP_FINALLY', final) | 
					
						
							|  |  |  |         self.visit(node.body) | 
					
						
							|  |  |  |         self.emit('POP_BLOCK') | 
					
						
							|  |  |  |         self.emit('LOAD_CONST', None) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.nextBlock(final) | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  |         self.visit(node.final) | 
					
						
							|  |  |  |         self.emit('END_FINALLY') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     # misc | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | ##     def visitStmt(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  | ##         # nothing to do except walk the children | 
					
						
							|  |  |  | ##         pass | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitDiscard(self, node): | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |         self.visit(node.expr) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('POP_TOP') | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitConst(self, node): | 
					
						
							|  |  |  |         self.emit('LOAD_CONST', node.value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitKeyword(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.emit('LOAD_CONST', node.name) | 
					
						
							|  |  |  |         self.visit(node.expr) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitGlobal(self, node): | 
					
						
							|  |  |  |         # no code to generate | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitName(self, node): | 
					
						
							|  |  |  |         self.loadName(node.name) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |     def visitPass(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitImport(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         for name in node.names: | 
					
						
							|  |  |  |             self.emit('IMPORT_NAME', name) | 
					
						
							|  |  |  |             self.storeName(name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitFrom(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('IMPORT_NAME', node.modname) | 
					
						
							|  |  |  |         for name in node.names: | 
					
						
							|  |  |  |             if name == '*': | 
					
						
							|  |  |  |                 self.namespace = 0 | 
					
						
							|  |  |  |             self.emit('IMPORT_FROM', name) | 
					
						
							|  |  |  |         self.emit('POP_TOP') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitGetattr(self, node): | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |         self.visit(node.expr) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('LOAD_ATTR', node.attrname) | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     # next five implement assignments | 
					
						
							| 
									
										
										
										
											2000-02-15 23:45:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |     def visitAssign(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |         self.visit(node.expr) | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         dups = len(node.nodes) - 1 | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         for i in range(len(node.nodes)): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |             elt = node.nodes[i] | 
					
						
							|  |  |  |             if i < dups: | 
					
						
							|  |  |  |                 self.emit('DUP_TOP') | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |             if isinstance(elt, ast.Node): | 
					
						
							|  |  |  |                 self.visit(elt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitAssName(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 22:09:35 +00:00
										 |  |  |         if node.flags == 'OP_ASSIGN': | 
					
						
							|  |  |  |             self.storeName(node.name) | 
					
						
							|  |  |  |         elif node.flags == 'OP_DELETE': | 
					
						
							|  |  |  |             self.delName(node.name) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |             print "oops", node.flags | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |     def visitAssAttr(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-16 00:55:44 +00:00
										 |  |  |         self.visit(node.expr) | 
					
						
							|  |  |  |         if node.flags == 'OP_ASSIGN': | 
					
						
							|  |  |  |             self.emit('STORE_ATTR', node.attrname) | 
					
						
							|  |  |  |         elif node.flags == 'OP_DELETE': | 
					
						
							|  |  |  |             self.emit('DELETE_ATTR', node.attrname) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |             print "warning: unexpected flags:", node.flags | 
					
						
							|  |  |  |             print node | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitAssTuple(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 22:09:35 +00:00
										 |  |  |         if findOp(node) != 'OP_DELETE': | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |             self.emit('UNPACK_SEQUENCE', len(node.nodes)) | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |         for child in node.nodes: | 
					
						
							|  |  |  |             self.visit(child) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     visitAssList = visitAssTuple | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def visitExec(self, node): | 
					
						
							|  |  |  |         self.visit(node.expr) | 
					
						
							|  |  |  |         if node.locals is None: | 
					
						
							|  |  |  |             self.emit('LOAD_CONST', None) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.visit(node.locals) | 
					
						
							|  |  |  |         if node.globals is None: | 
					
						
							|  |  |  |             self.emit('DUP_TOP') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.visit(node.globals) | 
					
						
							|  |  |  |         self.emit('EXEC_STMT') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitCallFunc(self, node): | 
					
						
							|  |  |  |         pos = 0 | 
					
						
							|  |  |  |         kw = 0 | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.visit(node.node) | 
					
						
							|  |  |  |         for arg in node.args: | 
					
						
							|  |  |  |             self.visit(arg) | 
					
						
							|  |  |  |             if isinstance(arg, ast.Keyword): | 
					
						
							|  |  |  |                 kw = kw + 1 | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 pos = pos + 1 | 
					
						
							| 
									
										
										
										
											2000-05-02 22:32:59 +00:00
										 |  |  |         if node.star_args is not None: | 
					
						
							|  |  |  |             self.visit(node.star_args) | 
					
						
							|  |  |  |         if node.dstar_args is not None: | 
					
						
							|  |  |  |             self.visit(node.dstar_args) | 
					
						
							|  |  |  |         have_star = node.star_args is not None | 
					
						
							|  |  |  |         have_dstar = node.dstar_args is not None | 
					
						
							|  |  |  |         opcode = callfunc_opcode_info[have_star, have_dstar] | 
					
						
							|  |  |  |         self.emit(opcode, kw << 8 | pos) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitPrint(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         for child in node.nodes: | 
					
						
							|  |  |  |             self.visit(child) | 
					
						
							|  |  |  |             self.emit('PRINT_ITEM') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitPrintnl(self, node): | 
					
						
							|  |  |  |         self.visitPrint(node) | 
					
						
							|  |  |  |         self.emit('PRINT_NEWLINE') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitReturn(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(node) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.visit(node.value) | 
					
						
							|  |  |  |         self.emit('RETURN_VALUE') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # slice and subscript stuff | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitSlice(self, node): | 
					
						
							|  |  |  |         self.visit(node.expr) | 
					
						
							|  |  |  |         slice = 0 | 
					
						
							|  |  |  |         if node.lower: | 
					
						
							|  |  |  |             self.visit(node.lower) | 
					
						
							|  |  |  |             slice = slice | 1 | 
					
						
							|  |  |  |         if node.upper: | 
					
						
							|  |  |  |             self.visit(node.upper) | 
					
						
							|  |  |  |             slice = slice | 2 | 
					
						
							|  |  |  |         if node.flags == 'OP_APPLY': | 
					
						
							|  |  |  |             self.emit('SLICE+%d' % slice) | 
					
						
							|  |  |  |         elif node.flags == 'OP_ASSIGN': | 
					
						
							|  |  |  |             self.emit('STORE_SLICE+%d' % slice) | 
					
						
							|  |  |  |         elif node.flags == 'OP_DELETE': | 
					
						
							|  |  |  |             self.emit('DELETE_SLICE+%d' % slice) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             print "weird slice", node.flags | 
					
						
							|  |  |  |             raise | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitSubscript(self, node): | 
					
						
							|  |  |  |         self.visit(node.expr) | 
					
						
							|  |  |  |         for sub in node.subs: | 
					
						
							|  |  |  |             self.visit(sub) | 
					
						
							|  |  |  |         if len(node.subs) > 1: | 
					
						
							|  |  |  |             self.emit('BUILD_TUPLE', len(node.subs)) | 
					
						
							|  |  |  |         if node.flags == 'OP_APPLY': | 
					
						
							|  |  |  |             self.emit('BINARY_SUBSCR') | 
					
						
							|  |  |  |         elif node.flags == 'OP_ASSIGN': | 
					
						
							|  |  |  |             self.emit('STORE_SUBSCR') | 
					
						
							|  |  |  |         elif node.flags == 'OP_DELETE': | 
					
						
							|  |  |  |             self.emit('DELETE_SUBSCR') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-06 19:10:54 +00:00
										 |  |  |     # binary ops | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  |     def binaryOp(self, node, op): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.visit(node.left) | 
					
						
							|  |  |  |         self.visit(node.right) | 
					
						
							|  |  |  |         self.emit(op) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitAdd(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_ADD') | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitSub(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_SUBTRACT') | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitMul(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_MULTIPLY') | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitDiv(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_DIVIDE') | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |     def visitMod(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_MODULO') | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-14 21:33:10 +00:00
										 |  |  |     def visitPower(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_POWER') | 
					
						
							| 
									
										
										
										
											2000-02-14 21:33:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitLeftShift(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_LSHIFT') | 
					
						
							| 
									
										
										
										
											2000-02-14 21:33:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitRightShift(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.binaryOp(node, 'BINARY_RSHIFT') | 
					
						
							| 
									
										
										
										
											2000-02-14 21:33:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-06 19:10:54 +00:00
										 |  |  |     # unary ops | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def unaryOp(self, node, op): | 
					
						
							|  |  |  |         self.visit(node.expr) | 
					
						
							|  |  |  |         self.emit(op) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-14 21:33:10 +00:00
										 |  |  |     def visitInvert(self, node): | 
					
						
							|  |  |  |         return self.unaryOp(node, 'UNARY_INVERT') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |     def visitUnarySub(self, node): | 
					
						
							|  |  |  |         return self.unaryOp(node, 'UNARY_NEGATIVE') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitUnaryAdd(self, node): | 
					
						
							|  |  |  |         return self.unaryOp(node, 'UNARY_POSITIVE') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def visitUnaryInvert(self, node): | 
					
						
							|  |  |  |         return self.unaryOp(node, 'UNARY_INVERT') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |     def visitNot(self, node): | 
					
						
							|  |  |  |         return self.unaryOp(node, 'UNARY_NOT') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |     def visitBackquote(self, node): | 
					
						
							|  |  |  |         return self.unaryOp(node, 'UNARY_CONVERT') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-06 19:10:54 +00:00
										 |  |  |     # bit ops | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |     def bitOp(self, nodes, op): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.visit(nodes[0]) | 
					
						
							|  |  |  |         for node in nodes[1:]: | 
					
						
							|  |  |  |             self.visit(node) | 
					
						
							|  |  |  |             self.emit(op) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitBitand(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.bitOp(node.nodes, 'BINARY_AND') | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitBitor(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.bitOp(node.nodes, 'BINARY_OR') | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitBitxor(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.bitOp(node.nodes, 'BINARY_XOR') | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     # object constructors | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |     def visitEllipsis(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.emit('LOAD_CONST', Ellipsis) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-08 21:15:48 +00:00
										 |  |  |     def visitTuple(self, node): | 
					
						
							|  |  |  |         for elt in node.nodes: | 
					
						
							|  |  |  |             self.visit(elt) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('BUILD_TUPLE', len(node.nodes)) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |     def visitList(self, node): | 
					
						
							|  |  |  |         for elt in node.nodes: | 
					
						
							|  |  |  |             self.visit(elt) | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |         self.emit('BUILD_LIST', len(node.nodes)) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitSliceobj(self, node): | 
					
						
							|  |  |  |         for child in node.nodes: | 
					
						
							|  |  |  |             self.visit(child) | 
					
						
							|  |  |  |         self.emit('BUILD_SLICE', len(node.nodes)) | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |     def visitDict(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         lineno = getattr(node, 'lineno', None) | 
					
						
							|  |  |  |         if lineno: | 
					
						
							|  |  |  |             set.emit('SET_LINENO', lineno) | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.emit('BUILD_MAP', 0) | 
					
						
							|  |  |  |         for k, v in node.items: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |             lineno2 = getattr(node, 'lineno', None) | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |             if lineno2 is not None and lineno != lineno2: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |                 self.emit('SET_LINENO', lineno2) | 
					
						
							|  |  |  |                 lineno = lineno2 | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |             self.emit('DUP_TOP') | 
					
						
							|  |  |  |             self.visit(v) | 
					
						
							|  |  |  |             self.emit('ROT_TWO') | 
					
						
							|  |  |  |             self.visit(k) | 
					
						
							|  |  |  |             self.emit('STORE_SUBSCR') | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | class ModuleCodeGenerator(CodeGenerator): | 
					
						
							|  |  |  |     super_init = CodeGenerator.__init__ | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         # XXX <module> is ? in compile.c | 
					
						
							|  |  |  |         self.graph = pyassem.PyFlowGraph("<module>", filename) | 
					
						
							|  |  |  |         self.super_init(filename) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class FunctionCodeGenerator(CodeGenerator): | 
					
						
							|  |  |  |     super_init = CodeGenerator.__init__ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     optimized = 1 | 
					
						
							|  |  |  |     lambdaCount = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, func, filename, isLambda=0): | 
					
						
							|  |  |  |         if isLambda: | 
					
						
							|  |  |  |             klass = FunctionCodeGenerator | 
					
						
							|  |  |  |             name = "<lambda.%d>" % klass.lambdaCount | 
					
						
							|  |  |  |             klass.lambdaCount = klass.lambdaCount + 1 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             name = func.name | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         args, hasTupleArg = generateArgList(func.argnames) | 
					
						
							|  |  |  |         self.graph = pyassem.PyFlowGraph(name, filename, args,  | 
					
						
							|  |  |  |                                            optimized=1)  | 
					
						
							|  |  |  |         self.isLambda = isLambda | 
					
						
							|  |  |  |         self.super_init(filename) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         lnf = walk(func.code, LocalNameFinder(args), 0) | 
					
						
							|  |  |  |         self.locals.push(lnf.getLocals()) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         if func.varargs: | 
					
						
							|  |  |  |             self.graph.setFlag(CO_VARARGS) | 
					
						
							|  |  |  |         if func.kwargs: | 
					
						
							|  |  |  |             self.graph.setFlag(CO_VARKEYWORDS) | 
					
						
							| 
									
										
										
										
											2000-08-04 16:56:51 +00:00
										 |  |  |         self.set_lineno(func) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         if hasTupleArg: | 
					
						
							|  |  |  |             self.generateArgUnpack(func.argnames) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def finish(self): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.graph.startExitBlock() | 
					
						
							|  |  |  |         if not self.isLambda: | 
					
						
							|  |  |  |             self.emit('LOAD_CONST', None) | 
					
						
							|  |  |  |         self.emit('RETURN_VALUE') | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     def generateArgUnpack(self, args): | 
					
						
							|  |  |  |         count = 0 | 
					
						
							|  |  |  |         for arg in args: | 
					
						
							|  |  |  |             if type(arg) == types.TupleType: | 
					
						
							|  |  |  |                 self.emit('LOAD_FAST', '.nested%d' % count) | 
					
						
							|  |  |  |                 count = count + 1 | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |                 self.unpackSequence(arg) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |                          | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |     def unpackSequence(self, tup): | 
					
						
							|  |  |  |         self.emit('UNPACK_SEQUENCE', len(tup)) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         for elt in tup: | 
					
						
							|  |  |  |             if type(elt) == types.TupleType: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |                 self.unpackSequence(elt) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 self.emit('STORE_FAST', elt) | 
					
						
							| 
									
										
										
										
											2000-02-04 19:37:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |     unpackTuple = unpackSequence | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | class ClassCodeGenerator(CodeGenerator): | 
					
						
							|  |  |  |     super_init = CodeGenerator.__init__ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, klass, filename): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.graph = pyassem.PyFlowGraph(klass.name, filename, | 
					
						
							|  |  |  |                                            optimized=0) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.super_init(filename) | 
					
						
							|  |  |  |         lnf = walk(klass.code, LocalNameFinder(), 0) | 
					
						
							|  |  |  |         self.locals.push(lnf.getLocals()) | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.graph.setFlag(CO_NEWLOCALS) | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def finish(self): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.graph.startExitBlock() | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         self.emit('LOAD_LOCALS') | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         self.emit('RETURN_VALUE') | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def generateArgList(arglist): | 
					
						
							|  |  |  |     """Generate an arg list marking TupleArgs""" | 
					
						
							|  |  |  |     args = [] | 
					
						
							|  |  |  |     extra = [] | 
					
						
							|  |  |  |     count = 0 | 
					
						
							|  |  |  |     for elt in arglist: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         if type(elt) == types.StringType: | 
					
						
							|  |  |  |             args.append(elt) | 
					
						
							|  |  |  |         elif type(elt) == types.TupleType: | 
					
						
							|  |  |  |             args.append(TupleArg(count, elt)) | 
					
						
							|  |  |  |             count = count + 1 | 
					
						
							|  |  |  |             extra.extend(misc.flatten(elt)) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise ValueError, "unexpect argument type:", elt | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     return args + extra, count | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | class LocalNameFinder: | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     """Find local names in scope""" | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  |     def __init__(self, names=()): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.names = misc.Set() | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |         self.globals = misc.Set() | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         for name in names: | 
					
						
							|  |  |  |             self.names.add(name) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def getLocals(self): | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |         for elt in self.globals.elements(): | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |             if self.names.has_elt(elt): | 
					
						
							|  |  |  |                 self.names.remove(elt) | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         return self.names | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  |     def visitDict(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											2000-02-14 14:14:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
											
												add ExampleASTVisitor:
* prints out examples of nodes that are handled by visitor.  simply a
  development convenience
remove NestedCodeGenerator -- it was bogus after all
replace with generateFunctionCode, a method to call to generate code
  for a function instead of a top-level module
fix impl of visitDiscard (most pop stack)
emit lineno for pass
handle the following new node types: Import, From, Getattr, Subscript,
Slice, AssAttr, AssTuple, Mod, Not, And, Or, List
LocalNameFinder: remove names declared as globals for locals
PythonVMCode: pass arg names to constructor, force varnames to contain
them all (even if they aren't referenced)
add -q option on command line to disable stdout
											
										 
											2000-02-10 00:47:08 +00:00
										 |  |  |     def visitGlobal(self, node): | 
					
						
							|  |  |  |         for name in node.names: | 
					
						
							|  |  |  |             self.globals.add(name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  |     def visitFunction(self, node): | 
					
						
							|  |  |  |         self.names.add(node.name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-11 20:27:07 +00:00
										 |  |  |     def visitLambda(self, node): | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											2000-02-11 20:27:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  |     def visitImport(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         for name in node.names: | 
					
						
							|  |  |  |             self.names.add(name) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitFrom(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         for name in node.names: | 
					
						
							|  |  |  |             self.names.add(name) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-06 19:10:54 +00:00
										 |  |  |     def visitClass(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.names.add(node.name) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def visitAssName(self, node): | 
					
						
							| 
									
										
										
										
											2000-02-17 17:56:29 +00:00
										 |  |  |         self.names.add(node.name) | 
					
						
							| 
									
										
										
										
											2000-02-04 00:28:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | def findOp(node): | 
					
						
							|  |  |  |     """Find the op (DELETE, LOAD, STORE) in an AssTuple tree""" | 
					
						
							|  |  |  |     v = OpFinder() | 
					
						
							|  |  |  |     walk(node, v, 0) | 
					
						
							|  |  |  |     return v.op | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-17 22:09:35 +00:00
										 |  |  | class OpFinder: | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         self.op = None | 
					
						
							|  |  |  |     def visitAssName(self, node): | 
					
						
							|  |  |  |         if self.op is None: | 
					
						
							|  |  |  |             self.op = node.flags | 
					
						
							|  |  |  |         elif self.op != node.flags: | 
					
						
							|  |  |  |             raise ValueError, "mixed ops in stmt" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     import sys | 
					
						
							| 
									
										
										
										
											2000-02-08 19:01:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-16 20:06:59 +00:00
										 |  |  |     for file in sys.argv[1:]: | 
					
						
							| 
									
										
										
										
											2000-08-12 20:32:46 +00:00
										 |  |  |         compile(file) |