Commit d05ff27d authored by Fred Drake's avatar Fred Drake

More micro-optimizations.

A (minor) change to the bytecode format: for "extended" attributes
(handled by TALInterpreter.attrAction()), the instruction contains the
integer previously retrieved from the actionIndex table; the bytecode
generator now takes care of the lookup.  This allows attrAction() to
no longer need to lookup the table that it used to lookup the
integers.

Many handlers are now separated into TAL and non-TAL versions -- the
TALInterpreter constructor determines which dispatch table to use when
it initializes self.tal (which is not changed later).  This allows
each of the handlers to do only what is needed without having to
decide based on the self.tal attribute.

TALInterpreter.do_startTag() no longer contains a special case for an
empty attribute list; that case is already optimized away by the
bytecode generator.  What remains would actually work in that case,
but it simply doesn't occur.  This removes one test of the attrList
parameter.
parent adac2175
......@@ -157,6 +157,8 @@ class TALGenerator:
collect = []
return output
actionIndex = {"replace":0, "insert":1, "metal":2, "tal":3, "xmlns":4,
0: 0, 1: 1, 2: 2, 3: 3, 4: 4}
def optimizeStartTag(self, collect, name, attrlist, end):
if not attrlist:
collect.append("<%s%s" % (name, end))
......@@ -167,6 +169,9 @@ class TALGenerator:
item = attrlist[i]
if len(item) > 2:
opt = 0
name, value, action = item[:3]
action = self.actionIndex[action]
attrlist[i] = (name, value, action) + item[3:]
else:
if item[1] is None:
s = item[0]
......
......@@ -170,6 +170,10 @@ class TALInterpreter:
self.wrap = wrap
self.metal = metal
self.tal = tal
if tal:
self.dispatch = self.bytecode_handlers_tal
else:
self.dispatch = self.bytecode_handlers
assert showtal in (-1, 0, 1)
if showtal == -1:
showtal = (not tal)
......@@ -178,7 +182,9 @@ class TALInterpreter:
self.stackLimit = stackLimit
self.html = 0
self.endsep = "/>"
self.endlen = len(self.endsep)
self.macroStack = []
self.popMacro = self.macroStack.pop
self.position = None, None # (lineno, offset)
self.col = 0
self.level = 0
......@@ -229,7 +235,8 @@ class TALInterpreter:
self._stream_write("\n")
self.col = 0
def stream_write(self, s):
def stream_write(self, s,
len=len, rfind=rfind):
self._stream_write(s)
i = rfind(s, '\n')
if i < 0:
......@@ -241,7 +248,7 @@ class TALInterpreter:
def interpret(self, program):
self.level = self.level + 1
handlers = self.bytecode_handlers
handlers = self.dispatch
_apply = apply
_tuple = tuple
tup = (self,)
......@@ -272,29 +279,29 @@ class TALInterpreter:
self.endsep = " />"
else:
self.endsep = "/>"
self.endlen = len(self.endsep)
bytecode_handlers["mode"] = do_mode
def do_setPosition(self, position):
self.position = position
bytecode_handlers["setPosition"] = do_setPosition
def do_startEndTag(self, (name, attrList)):
self.do_startTag((name, attrList), self.endsep)
def do_startEndTag(self, stuff):
self.do_startTag(stuff, self.endsep, self.endlen)
bytecode_handlers["startEndTag"] = do_startEndTag
def do_startTag(self, (name, attrList), end=">"):
if not attrList:
s = "<%s%s" % (name, end)
self._stream_write(s)
self.col = self.col + len(s)
return
_len = len
_quote = quote
def do_startTag(self, (name, attrList),
end=">", endlen=1, _len=len, _quote=quote):
# The bytecode generator does not cause calls to this method
# for start tags with no attributes; those are optimized down
# to rawtext events. Hence, there is no special "fast path"
# for that case.
_stream_write = self._stream_write
_stream_write("<" + name)
col = self.col + _len(name) + 1
namelen = _len(name)
col = self.col + namelen + 1
wrap = self.wrap
align = col + 1 + _len(name)
align = col + 1 + namelen
if align >= wrap/2:
align = 4 # Avoid a narrow column far to the right
try:
......@@ -309,25 +316,24 @@ class TALInterpreter:
s = name
else:
s = "%s=%s" % (name, _quote(value))
slen = _len(s)
if (wrap and
col >= align and
col + 1 + _len(s) > wrap):
col + 1 + slen > wrap):
_stream_write("\n" + " "*align)
col = align + _len(s)
col = align + slen
else:
s = " " + s
col = col + 1 + _len(s)
col = col + 1 + slen
_stream_write(s)
_stream_write(end)
col = col + _len(end)
col = col + endlen
finally:
self.col = col
bytecode_handlers["startTag"] = do_startTag
actionIndex = {"replace":0, "insert":1, "metal":2, "tal":3, "xmlns":4}
def attrAction(self, item):
name, value = item[:2]
action = self.actionIndex[item[2]]
name, value, action = item[:3]
if not self.showtal and action > 1:
return 0, name, value
ok = 1
......@@ -349,7 +355,6 @@ class TALInterpreter:
else:
if evalue is None:
ok = 0
else:
value = evalue
elif action == 2 and self.metal:
i = rfind(name, ":") + 1
......@@ -387,12 +392,13 @@ class TALInterpreter:
sys.stderr.write("+--------------------------------------\n")
def do_beginScope(self, dict):
if self.tal:
self.engine.beginScope()
self.scopeLevel = self.scopeLevel + 1
def do_beginScope_tal(self, dict):
engine = self.engine
engine.beginScope()
engine.setLocal("attrs", dict)
else:
self.engine.beginScope()
self.scopeLevel = self.scopeLevel + 1
bytecode_handlers["beginScope"] = do_beginScope
......@@ -401,22 +407,23 @@ class TALInterpreter:
self.scopeLevel = self.scopeLevel - 1
bytecode_handlers["endScope"] = do_endScope
def do_setLocal(self, (name, expr)):
if self.tal:
def do_setLocal(self, junk):
pass
def do_setLocal_tal(self, (name, expr)):
value = self.engine.evaluateValue(expr)
self.engine.setLocal(name, value)
bytecode_handlers["setLocal"] = do_setLocal
def do_setGlobal(self, (name, expr)):
if self.tal:
def do_setGlobal_tal(self, (name, expr)):
value = self.engine.evaluateValue(expr)
self.engine.setGlobal(name, value)
bytecode_handlers["setGlobal"] = do_setGlobal
bytecode_handlers["setGlobal"] = do_setLocal
def do_insertText(self, (expr, block)):
if not self.tal:
self.interpret(block)
return
def do_insertText_tal(self, (expr, block)):
text = self.engine.evaluateText(expr)
if text is None:
return
......@@ -469,9 +476,9 @@ class TALInterpreter:
self.interpret(program)
def do_loop(self, (name, expr, block)):
if not self.tal:
self.interpret(block)
return
def do_loop_tal(self, (name, expr, block)):
iterator = self.engine.setRepeat(name, expr)
while iterator.next():
self.interpret(block)
......@@ -548,9 +555,9 @@ class TALInterpreter:
bytecode_handlers["defineSlot"] = do_defineSlot
def do_onError(self, (block, handler)):
if not self.tal:
self.interpret(block)
return
def do_onError_tal(self, (block, handler)):
state = self.saveState()
self.stream = stream = StringIO()
self._stream_write = stream.write
......@@ -569,6 +576,14 @@ class TALInterpreter:
self.stream_write(stream.getvalue())
bytecode_handlers["onError"] = do_onError
bytecode_handlers_tal = bytecode_handlers.copy()
bytecode_handlers_tal["beginScope"] = do_beginScope_tal
bytecode_handlers_tal["setLocal"] = do_setLocal_tal
bytecode_handlers_tal["setGlobal"] = do_setGlobal_tal
bytecode_handlers_tal["insertText"] = do_insertText_tal
bytecode_handlers_tal["loop"] = do_loop_tal
bytecode_handlers_tal["onError"] = do_onError_tal
def test():
from driver import FILE, parsefile
......
......@@ -155,7 +155,7 @@ class METALGeneratorTestCases(TestCaseBase):
def check_define_macro(self):
macro = self.initial_program + [
('startTag', ('p', [('metal:define-macro', 'M', 'metal')])),
('startTag', ('p', [('metal:define-macro', 'M', 2)])),
rawtext('booh</p>'),
]
program = [
......@@ -170,17 +170,17 @@ class METALGeneratorTestCases(TestCaseBase):
('setPosition', (1, 0)),
('useMacro',
('M', '$M$', {},
[('startTag', ('p', [('metal:use-macro', 'M', 'metal')])),
[('startTag', ('p', [('metal:use-macro', 'M', 2)])),
rawtext('booh</p>')])),
])
def check_define_slot(self):
macro = self.initial_program + [
('startTag', ('p', [('metal:define-macro', 'M', 'metal')])),
('startTag', ('p', [('metal:define-macro', 'M', 2)])),
rawtext('foo'),
('setPosition', (1, 29)),
('defineSlot', ('S',
[('startTag', ('span', [('metal:define-slot', 'S', 'metal')])),
[('startTag', ('span', [('metal:define-slot', 'S', 2)])),
rawtext('spam</span>')])),
rawtext('bar</p>'),
]
......@@ -198,13 +198,13 @@ class METALGeneratorTestCases(TestCaseBase):
('useMacro',
('M', '$M$',
{'S': [('startTag', ('span',
[('metal:fill-slot', 'S', 'metal')])),
[('metal:fill-slot', 'S', 2)])),
rawtext('spam</span>')]},
[('startTag', ('p', [('metal:use-macro', 'M', 'metal')])),
[('startTag', ('p', [('metal:use-macro', 'M', 2)])),
rawtext('foo'),
('setPosition', (1, 26)),
('fillSlot', ('S',
[('startTag', ('span', [('metal:fill-slot', 'S', 'metal')])),
[('startTag', ('span', [('metal:fill-slot', 'S', 2)])),
rawtext('spam</span>')])),
rawtext('bar</p>')])),
])
......@@ -220,7 +220,7 @@ class TALGeneratorTestCases(TestCaseBase):
('setPosition', (1, 0)),
('beginScope', {'tal:define': 'xyzzy string:spam'}),
('setLocal', ('xyzzy', '$string:spam$')),
('startTag', ('p', [('tal:define', 'xyzzy string:spam', 'tal')])),
('startTag', ('p', [('tal:define', 'xyzzy string:spam', 3)])),
rawtext('</p>'),
('endScope', ()),
])
......@@ -231,7 +231,7 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:define': 'local xyzzy string:spam'}),
('setLocal', ('xyzzy', '$string:spam$')),
('startTag', ('p',
[('tal:define', 'local xyzzy string:spam', 'tal')])),
[('tal:define', 'local xyzzy string:spam', 3)])),
rawtext('</p>'),
('endScope', ()),
])
......@@ -242,7 +242,7 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:define': 'global xyzzy string:spam'}),
('setGlobal', ('xyzzy', '$string:spam$')),
('startTag', ('p',
[('tal:define', 'global xyzzy string:spam', 'tal')])),
[('tal:define', 'global xyzzy string:spam', 3)])),
rawtext('</p>'),
('endScope', ()),
])
......@@ -253,7 +253,7 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:define': 'x string:spam; y x'}),
('setLocal', ('x', '$string:spam$')),
('setLocal', ('y', '$x$')),
('startTag', ('p', [('tal:define', 'x string:spam; y x', 'tal')])),
('startTag', ('p', [('tal:define', 'x string:spam; y x', 3)])),
rawtext('</p>'),
('endScope', ()),
])
......@@ -264,7 +264,7 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:define': 'x string:;;;;; y x'}),
('setLocal', ('x', '$string:;;$')),
('setLocal', ('y', '$x$')),
('startTag', ('p', [('tal:define', 'x string:;;;;; y x', 'tal')])),
('startTag', ('p', [('tal:define', 'x string:;;;;; y x', 3)])),
rawtext('</p>'),
('endScope', ()),
])
......@@ -279,7 +279,7 @@ class TALGeneratorTestCases(TestCaseBase):
('setGlobal', ('y', '$x$')),
('setLocal', ('z', '$y$')),
('startTag', ('p',
[('tal:define', 'x string:spam; global y x; local z y', 'tal')])),
[('tal:define', 'x string:spam; global y x; local z y', 3)])),
rawtext('</p>'),
('endScope', ()),
])
......@@ -291,7 +291,7 @@ class TALGeneratorTestCases(TestCaseBase):
('setPosition', (1, 3)),
('beginScope', {'tal:condition': 'python:1'}),
('condition', ('$python:1$',
[('startTag', ('span', [('tal:condition', 'python:1', 'tal')])),
[('startTag', ('span', [('tal:condition', 'python:1', 3)])),
rawtext('<b>foo</b></span>')])),
('endScope', ()),
rawtext('</p>'),
......@@ -301,7 +301,7 @@ class TALGeneratorTestCases(TestCaseBase):
self._run_check("<p tal:content='string:foo'>bar</p>", [
('setPosition', (1, 0)),
('beginScope', {'tal:content': 'string:foo'}),
('startTag', ('p', [('tal:content', 'string:foo', 'tal')])),
('startTag', ('p', [('tal:content', 'string:foo', 3)])),
('insertText', ('$string:foo$', [rawtext('bar')])),
rawtext('</p>'),
('endScope', ()),
......@@ -311,7 +311,7 @@ class TALGeneratorTestCases(TestCaseBase):
self._run_check("<p tal:content='text string:foo'>bar</p>", [
('setPosition', (1, 0)),
('beginScope', {'tal:content': 'text string:foo'}),
('startTag', ('p', [('tal:content', 'text string:foo', 'tal')])),
('startTag', ('p', [('tal:content', 'text string:foo', 3)])),
('insertText', ('$string:foo$', [rawtext('bar')])),
rawtext('</p>'),
('endScope', ()),
......@@ -322,7 +322,7 @@ class TALGeneratorTestCases(TestCaseBase):
('setPosition', (1, 0)),
('beginScope', {'tal:content': 'structure string:<br>'}),
('startTag', ('p',
[('tal:content', 'structure string:<br>', 'tal')])),
[('tal:content', 'structure string:<br>', 3)])),
('insertStructure',
('$string:<br>$', {}, [rawtext('bar')])),
rawtext('</p>'),
......@@ -334,7 +334,7 @@ class TALGeneratorTestCases(TestCaseBase):
('setPosition', (1, 0)),
('beginScope', {'tal:replace': 'string:foo'}),
('insertText', ('$string:foo$',
[('startTag', ('p', [('tal:replace', 'string:foo', 'tal')])),
[('startTag', ('p', [('tal:replace', 'string:foo', 3)])),
rawtext('bar</p>')])),
('endScope', ()),
])
......@@ -345,7 +345,7 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:replace': 'text string:foo'}),
('insertText', ('$string:foo$',
[('startTag', ('p',
[('tal:replace', 'text string:foo', 'tal')])),
[('tal:replace', 'text string:foo', 3)])),
rawtext('bar</p>')])),
('endScope', ()),
])
......@@ -356,7 +356,7 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:replace': 'structure string:<br>'}),
('insertStructure', ('$string:<br>$', {},
[('startTag', ('p',
[('tal:replace', 'structure string:<br>', 'tal')])),
[('tal:replace', 'structure string:<br>', 3)])),
rawtext('bar</p>')])),
('endScope', ()),
])
......@@ -368,11 +368,11 @@ class TALGeneratorTestCases(TestCaseBase):
('beginScope', {'tal:repeat': 'x python:(1,2,3)'}),
('loop', ('x', '$python:(1,2,3)$',
[('startTag', ('p',
[('tal:repeat', 'x python:(1,2,3)', 'tal')])),
[('tal:repeat', 'x python:(1,2,3)', 3)])),
('setPosition', (1, 33)),
('beginScope', {'tal:replace': 'x'}),
('insertText', ('$x$',
[('startTag', ('span', [('tal:replace', 'x', 'tal')])),
[('startTag', ('span', [('tal:replace', 'x', 3)])),
rawtext('dummy</span>')])),
('endScope', ()),
rawtext('</p>')])),
......@@ -388,11 +388,11 @@ class TALGeneratorTestCases(TestCaseBase):
{'tal:attributes': 'href string:http://www.zope.org; x string:y',
'name': 'bar', 'href': 'foo'}),
('startTag', ('a',
[('href', 'foo', 'replace', '$string:http://www.zope.org$'),
[('href', 'foo', 0, '$string:http://www.zope.org$'),
('name', 'name="bar"'),
('tal:attributes',
'href string:http://www.zope.org; x string:y', 'tal'),
('x', None, 'insert', '$string:y$')])),
'href string:http://www.zope.org; x string:y', 3),
('x', None, 1, '$string:y$')])),
rawtext('link</a>'),
('endScope', ()),
])
......@@ -407,8 +407,8 @@ class TALGeneratorTestCases(TestCaseBase):
('insertStructure', ('$string:<img>$',
{'src': '$string:foo.png$'},
[('startTag', ('p',
[('tal:replace', 'structure string:<img>', 'tal'),
('tal:attributes', 'src string:foo.png', 'tal')])),
[('tal:replace', 'structure string:<img>', 3),
('tal:attributes', 'src string:foo.png', 3)])),
rawtext('duh</p>')])),
('endScope', ()),
])
......@@ -421,13 +421,13 @@ class TALGeneratorTestCases(TestCaseBase):
{'tal:content': 'notHere', 'tal:on-error': 'string:error'}),
('onError',
([('startTag', ('p',
[('tal:on-error', 'string:error', 'tal'),
('tal:content', 'notHere', 'tal')])),
[('tal:on-error', 'string:error', 3),
('tal:content', 'notHere', 3)])),
('insertText', ('$notHere$', [rawtext('okay')])),
rawtext('</p>')],
[('startTag', ('p',
[('tal:on-error', 'string:error', 'tal'),
('tal:content', 'notHere', 'tal')])),
[('tal:on-error', 'string:error', 3),
('tal:content', 'notHere', 3)])),
('insertText', ('$string:error$', [])),
rawtext('</p>')])),
('endScope', ()),
......@@ -442,12 +442,12 @@ class TALGeneratorTestCases(TestCaseBase):
('onError',
([('insertText', ('$notHere$',
[('startTag', ('p',
[('tal:on-error', 'string:error', 'tal'),
('tal:replace', 'notHere', 'tal')])),
[('tal:on-error', 'string:error', 3),
('tal:replace', 'notHere', 3)])),
rawtext('okay</p>')]))],
[('startTag', ('p',
[('tal:on-error', 'string:error', 'tal'),
('tal:replace', 'notHere', 'tal')])),
[('tal:on-error', 'string:error', 3),
('tal:replace', 'notHere', 3)])),
('insertText', ('$string:error$', [])),
rawtext('</p>')])),
('endScope', ()),
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment