From b6f46a392b7d902fc96a348c427a6166714aaba5 Mon Sep 17 00:00:00 2001 From: Dag Sverre Seljebotn <dagss@student.matnat.uio.no> Date: Wed, 30 Jul 2008 10:54:59 +0200 Subject: [PATCH] Changed name from "fork" to "insertion_point" (codewriter), introduced func context --- Cython/Compiler/Annotate.py | 2 +- Cython/Compiler/Code.py | 227 +++++++++++++++++++--------------- Cython/Compiler/ModuleNode.py | 2 +- Cython/StringIOTree.py | 35 +++--- 4 files changed, 149 insertions(+), 117 deletions(-) diff --git a/Cython/Compiler/Annotate.py b/Cython/Compiler/Annotate.py index 7e2d8c70..6336b142 100644 --- a/Cython/Compiler/Annotate.py +++ b/Cython/Compiler/Annotate.py @@ -25,7 +25,7 @@ class AnnotationCCodeWriter(CCodeWriter): self.last_pos = None self.code = {} else: - # When forking, keep references to the same database + # When creating an insertion point, keep references to the same database self.annotation_buffer = create_from.annotation_buffer self.annotations = create_from.annotations self.code = create_from.code diff --git a/Cython/Compiler/Code.py b/Cython/Compiler/Code.py index f837dda8..bc7ef507 100644 --- a/Cython/Compiler/Code.py +++ b/Cython/Compiler/Code.py @@ -11,27 +11,107 @@ from TypeSlots import method_coexist from Scanning import SourceDescriptor from Cython.StringIOTree import StringIOTree +class FunctionContext(object): + # Not used for now, perhaps later + def __init__(self): + self.error_label = None + self.label_counter = 0 + self.labels_used = {} + self.return_label = self.new_label() + self.new_error_label() + self.continue_label = None + self.break_label = None + + self.temps_allocated = [] + self.temps_free = {} # type -> list of free vars + + def new_label(self): + n = self.label_counter + self.label_counter = n + 1 + return "%s%d" % (Naming.label_prefix, n) + + def new_error_label(self): + old_err_lbl = self.error_label + self.error_label = self.new_label() + return old_err_lbl + + def get_loop_labels(self): + return ( + self.continue_label, + self.break_label) + + def set_loop_labels(self, labels): + (self.continue_label, + self.break_label) = labels + + def new_loop_labels(self): + old_labels = self.get_loop_labels() + self.set_loop_labels( + (self.new_label(), + self.new_label())) + return old_labels + + def get_all_labels(self): + return ( + self.continue_label, + self.break_label, + self.return_label, + self.error_label) + + def set_all_labels(self, labels): + (self.continue_label, + self.break_label, + self.return_label, + self.error_label) = labels + + def all_new_labels(self): + old_labels = self.get_all_labels() + new_labels = [] + for old_label in old_labels: + if old_label: + new_labels.append(self.new_label()) + else: + new_labels.append(old_label) + self.set_all_labels(new_labels) + return old_labels + + def use_label(self, lbl): + self.labels_used[lbl] = 1 + + def label_used(self, lbl): + return lbl in self.labels_used + + def allocate_temp(self, type): + freelist = self.temps_free.get(type) + if freelist is not None and len(freelist) > 0: + return freelist.pop() + else: + pass + + + +def funccontext_property(name): + def get(self): + return getattr(self.func, name) + def set(self, value): + setattr(self.func, name, value) + return property(get, set) + class CCodeWriter(object): """ - Utility class to output C code. Each codewriter is forkable (see - StringIOTree). + Utility class to output C code. - When forking a code writer one must care about the state that is + When creating an insertion point one must care about the state that is kept: - - formatting state (level, bol) is cloned and modifyable in - all forked copies + - formatting state (level, bol) is cloned and used in insertion points + as well - labels, temps, exc_vars: One must construct a scope in which these can exist by calling enter_cfunc_scope/exit_cfunc_scope (these are for - sanity checking and forward compatabilty). When a fork happens, only - the *last* fork will maintain this created scope, while the other - instances "looses" their ability to use temps and labels (as this - is sufficient for current usecases). - - utility code: Same story as with labels and temps; use enter_implementation - and exit_implementation. - - marker: Only kept in last fork. - - filename_table, filename_list, input_file_contents: All forks share - the same instances simultaneously. - - + sanity checking and forward compatabilty). Created insertion points + looses this scope and cannot access it. + - marker: Not copied to insertion point + - filename_table, filename_list, input_file_contents: All codewriters + coming from the same root share the same instances simultaneously. """ # f file output file @@ -52,7 +132,8 @@ class CCodeWriter(object): # exc_vars (string * 3) exception variables for reraise, or None # input_file_contents dict contents (=list of lines) of any file that was used as input # to create this output C code. This is - # used to annotate the comments. + # used to annotate the comments. + # func FunctionContext contains labels and temps context info in_try_finally = 0 @@ -61,6 +142,7 @@ class CCodeWriter(object): self.buffer = buffer self.marker = None self.last_marker_line = 0 + self.func = None if create_from is None: # Root CCodeWriter self.level = 0 @@ -69,13 +151,11 @@ class CCodeWriter(object): self.filename_list = [] self.exc_vars = None self.input_file_contents = {} - self.in_cfunc = False else: # Clone formatting state c = create_from self.level = c.level self.bol = c.bol - self.in_cfunc = c.in_cfunc # Note: NOT copying but sharing instance self.filename_table = c.filename_table self.filename_list = [] @@ -95,31 +175,39 @@ class CCodeWriter(object): def write(self, s): self.buffer.write(s) - - def fork(self): - other = self.create_new(create_from=self, buffer=self.buffer.fork()) - # If we need to do something with our own state on fork, do it here + + def insertion_point(self): + other = self.create_new(create_from=self, buffer=self.buffer.insertion_point()) return other + + # Properties delegated to function scope + label_counter = funccontext_property("label_counter") + return_label = funccontext_property("return_label") + error_label = funccontext_property("error_label") + labels_used = funccontext_property("labels_used") + continue_label = funccontext_property("continue_label") + break_label = funccontext_property("break_label") + + + # Functions delegated to function scope + def new_label(self): return self.func.new_label() + def new_error_label(self): return self.func.new_error_label() + def get_loop_labels(self): return self.func.get_loop_labels() + def set_loop_labels(self, labels): return self.func.set_loop_labels(labels) + def new_loop_labels(self): return self.func.new_loop_labels() + def get_all_labels(self): return self.func.get_all_labels() + def set_all_labels(self, labels): return self.func.set_all_labels(labels) + def all_new_labels(self): return self.func.all_new_labels() + def use_label(self, lbl): return self.func.use_label(lbl) + def label_used(self, lbl): return self.func.label_used(lbl) + + def enter_cfunc_scope(self): - assert not self.in_cfunc - self.in_cfunc = True - self.error_label = None - self.label_counter = 0 - self.labels_used = {} - self.return_label = self.new_label() - self.new_error_label() - self.continue_label = None - self.break_label = None + self.func = FunctionContext() def exit_cfunc_scope(self): - self.in_cfunc = False - del self.error_label - del self.label_counter - del self.labels_used - del self.return_label - del self.continue_label - del self.break_label + self.func = None def putln(self, code = ""): if self.marker and self.bol: @@ -199,68 +287,13 @@ class CCodeWriter(object): source_desc.get_escaped_description(), line, u'\n'.join(lines)) self.marker = (line, marker) - def new_label(self): - n = self.label_counter - self.label_counter = n + 1 - return "%s%d" % (Naming.label_prefix, n) - - def new_error_label(self): - old_err_lbl = self.error_label - self.error_label = self.new_label() - return old_err_lbl - - def get_loop_labels(self): - return ( - self.continue_label, - self.break_label) - - def set_loop_labels(self, labels): - (self.continue_label, - self.break_label) = labels - - def new_loop_labels(self): - old_labels = self.get_loop_labels() - self.set_loop_labels( - (self.new_label(), - self.new_label())) - return old_labels - - def get_all_labels(self): - return ( - self.continue_label, - self.break_label, - self.return_label, - self.error_label) - - def set_all_labels(self, labels): - (self.continue_label, - self.break_label, - self.return_label, - self.error_label) = labels - - def all_new_labels(self): - old_labels = self.get_all_labels() - new_labels = [] - for old_label in old_labels: - if old_label: - new_labels.append(self.new_label()) - else: - new_labels.append(old_label) - self.set_all_labels(new_labels) - return old_labels - - def use_label(self, lbl): - self.labels_used[lbl] = 1 - - def label_used(self, lbl): - return lbl in self.labels_used def put_label(self, lbl): - if lbl in self.labels_used: + if lbl in self.func.labels_used: self.putln("%s:;" % lbl) def put_goto(self, lbl): - self.use_label(lbl) + self.func.use_label(lbl) self.putln("goto %s;" % lbl) def put_var_declarations(self, entries, static = 0, dll_linkage = None, @@ -412,8 +445,8 @@ class CCodeWriter(object): return cond def error_goto(self, pos): - lbl = self.error_label - self.use_label(lbl) + lbl = self.func.error_label + self.func.use_label(lbl) if Options.c_line_in_traceback: cinfo = " %s = %s;" % (Naming.clineno_cname, Naming.line_c_macro) else: diff --git a/Cython/Compiler/ModuleNode.py b/Cython/Compiler/ModuleNode.py index 6e4b20dd..f2662a7e 100644 --- a/Cython/Compiler/ModuleNode.py +++ b/Cython/Compiler/ModuleNode.py @@ -239,7 +239,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode): code = Annotate.AnnotationCCodeWriter() else: code = Code.CCodeWriter() - h_code = code.fork() + h_code = code.insertion_point() self.generate_module_preamble(env, modules, h_code) code.putln("") diff --git a/Cython/StringIOTree.py b/Cython/StringIOTree.py index 45cf6619..5f4fb879 100644 --- a/Cython/StringIOTree.py +++ b/Cython/StringIOTree.py @@ -24,7 +24,7 @@ class StringIOTree(object): def write(self, what): self.stream.write(what) - def fork(self): + def insertion_point(self): # Save what we have written until now # (would it be more efficient to check with len(self.stream.getvalue())? # leaving it out for now) @@ -36,30 +36,29 @@ class StringIOTree(object): return other __doc__ = r""" -Implements a forkable buffer. When you know you need to "get back" to a place -and write more later, simply call fork() at that spot and get a new -StringIOTree object that is "left behind", *behind* the object that is -forked. +Implements a buffer with insertion points. When you know you need to +"get back" to a place and write more later, simply call insertion_point() +at that spot and get a new StringIOTree object that is "left behind". EXAMPLE: ->>> pyrex = StringIOTree() ->>> pyrex.write('first\n') ->>> cython = pyrex.fork() ->>> pyrex.write('third\n') ->>> cython.write('second\n') ->>> print pyrex.getvalue() +>>> a = StringIOTree() +>>> a.write('first\n') +>>> b = a.insertion_point() +>>> a.write('third\n') +>>> b.write('second\n') +>>> print a.getvalue() first second third <BLANKLINE> ->>> b = cython.fork() ->>> a = b.fork() ->>> a.write('alpha\n') ->>> cython.write('gamma\n') ->>> b.write('beta\n') ->>> print cython.getvalue() +>>> c = b.insertion_point() +>>> d = c.insertion_point() +>>> d.write('alpha\n') +>>> b.write('gamma\n') +>>> c.write('beta\n') +>>> print b.getvalue() second alpha beta @@ -67,7 +66,7 @@ gamma <BLANKLINE> >>> out = StringIO() ->>> pyrex.copyto(out) +>>> a.copyto(out) >>> print out.getvalue() first second -- 2.26.2