1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408
|
#------------------------------------------------------------------------------
# Copyright (c) 2013-2025, Nucleic Development Team.
#
# Distributed under the terms of the Modified BSD License.
#
# The full license is in the file LICENSE, distributed with this software.
#------------------------------------------------------------------------------
from atom.api import List, Typed
from . import block_compiler as block
from . import compiler_common as cmn
from .enaml_ast import ConstExpr, Template
from ..compat import PY311, PY313
def collect_local_names(node):
""" Collect the compile-time local variable names for the node.
Parameters
----------
node : Template
The enaml ast template node of interest.
Returns
-------
result : tuple
A 2-tuple of (const_names, param_names) lists for the template.
"""
const_names = []
param_names = []
params = node.parameters
for param in (params.positional + params.keywords):
param_names.append(param.name)
if params.starparam:
param_names.append(params.starparam)
for item in node.body:
if isinstance(item, ConstExpr):
const_names.append(item.name)
return const_names, param_names
class FirstPassTemplateCompiler(block.FirstPassBlockCompiler):
""" The first pass template compiler.
This compiler generates the code which builds the compiler nodes
for the template definition. The main entry point is the 'compile'
class method.
"""
#: The const names collected during traversal.
const_names = List()
@classmethod
def compile(cls, node, args, local_names, filename):
""" Invoke the compiler for the given node.
The generated code object expects the SCOPE_KEY to be passed as
one of the arguments. The code object will return a 2-tuple of
compiler node list and const expression value tuple.
Parameters
----------
node : Template
The enaml ast Template node of interest.
args : list
The list of argument names which will be passed to the
code object when it is invoked.
local_names : set
The set of local names which are available to the code
object. This should be the combination of const expression
names and template parameter names.
filename : str
The filename of the node being compiled.
Returns
-------
result : tuple
A 2-tuple of (code, index_map) which is the generated code
object for the first compiler pass, and a mapping of ast
node to relevant compiler node index.
"""
compiler = cls()
compiler.filename = filename
compiler.local_names = local_names
cg = compiler.code_generator
# Setup the block for execution.
cmn.fetch_helpers(cg)
cmn.make_node_list(cg, cmn.count_nodes(node))
# Dispatch the visitors.
compiler.visit(node)
# Setup the parameters and generate the code object.
cg.name = node.name
cg.firstlineno = node.lineno
cg.newlocals = True
cg.args = args
code = cg.to_code()
# Union the two index maps for use by the second compiler pass.
final_index_map = dict(compiler.index_map)
final_index_map.update(compiler.aux_index_map)
return (code, final_index_map)
def visit_Template(self, node):
# No pragmas are supported yet for template nodes.
cmn.warn_pragmas(node, self.filename)
# Claim the index for the compiler node
index = len(self.index_map)
self.index_map[node] = index
# Setup the line number for the template.
cg = self.code_generator
cg.set_lineno(node.lineno)
# Create the template compiler node and store in the node list.
# Python 3.11 and 3.12 requires a NULL before a function that is not a method
# Python 3.13 one after
if not PY313 and PY311:
cg.push_null()
cmn.load_helper(cg, 'template_node')
if PY313:
cg.push_null()
cg.load_fast(cmn.SCOPE_KEY)
cg.call_function(1)
cmn.store_node(cg, index)
# Visit the body of the template.
for item in node.body:
self.visit(item)
# Update the internal node ids for the hierarchy.
cmn.load_node(cg, 0)
cg.load_method('update_id_nodes')
cg.call_function(is_method=True)
cg.pop_top()
# Load the compiler node list for returning.
cg.load_fast(cmn.NODE_LIST)
# Load the const names for returning.
for name in self.const_names:
cg.load_fast(name)
cg.build_tuple(len(self.const_names))
# Create and return the return value tuple.
cg.build_tuple(2)
cg.return_value()
def visit_ConstExpr(self, node):
# Keep track of the const name for loading for return.
self.const_names.append(node.name)
# Setup the line number for the const expr.
cg = self.code_generator
cg.set_lineno(node.lineno)
# Generate the code for the expression.
names = self.local_names
cmn.safe_eval_ast(cg, node.expr.ast, node.name, node.lineno, names)
# Validate the type of the expression value.
if node.typename:
with cg.try_squash_raise():
cg.dup_top()
# Python 3.11 and 3.12 requires a NULL before a function that is not a method
# Python 3.13 one after
if not PY313 and PY311:
cg.push_null()
cg.rot_two()
cmn.load_helper(cg, 'type_check_expr')
cg.rot_two()
if PY313:
cg.push_null()
cg.rot_two()
cmn.load_typename(cg, node.typename, names)
cg.call_function(2)
cg.pop_top()
# Store the result in the fast locals.
cg.store_fast(node.name)
class SecondPassTemplateCompiler(block.SecondPassBlockCompiler):
""" The second pass template compiler.
This compiler generates code which binds the data to the compiler
nodes for the template definition. The main entry point is the
'compile' class method.
"""
#: A mapping of const names to their value index.
const_indices = Typed(dict, ())
@classmethod
def compile(cls, node, args, local_names, index_map, consts, filename):
""" Invoke the compiler for the given node.
The generated code object expects NODE_LIST and T_CONSTS to be
passed as part of the arguments.
Parameters
----------
node : Template
The enaml ast Template node of interest.
args : list
The list of argument names which will be passed to the
code object when it is invoked.
local_names : set
The set of local names which are available to the code
object. This should be the combination of const expression
names and template parameter names.
index_map : dict
A mapping of ast node to compiler node index in the node list.
consts : list
The list of const expression names for the block.
filename : str
The filename of the node being compiled.
Returns
-------
result : CodeType
The code object which will bind the data for the block.
"""
compiler = cls()
compiler.filename = filename
compiler.local_names = local_names
compiler.index_map = index_map
for index, name in enumerate(consts):
compiler.const_indices[name] = index
cg = compiler.code_generator
# Setup the block for execution.
cmn.fetch_helpers(cg)
cmn.fetch_globals(cg)
# Dispatch the visitors.
compiler.visit(node)
# Setup the parameters and generate the code object.
cg.name = node.name
cg.firstlineno = node.lineno
cg.newlocals = True
cg.args = args
return cg.to_code()
def visit_Template(self, node):
# Visit the body of the template.
for item in node.body:
self.visit(item)
# Add the return value for the code.
cg = self.code_generator
cg.load_const(None)
cg.return_value()
def visit_ConstExpr(self, node):
# Store the value of the const expr into fast locals.
cg = self.code_generator
cg.set_lineno(node.lineno)
cg.load_fast(cmn.T_CONSTS)
cg.load_const(self.const_indices[node.name])
cg.binary_subscr()
cg.store_fast(node.name)
class TemplateCompiler(cmn.CompilerBase):
""" A compiler which will compile an Enaml template definition.
The entry point is the `compile` classmethod which will compile
the ast into a python code object which will generate a template
instance compiler node when invoked.
"""
@classmethod
def compile(cls, node, filename):
""" The main entry point to the compiler.
Parameters
----------
node : Template
The enaml ast Template node to compile.
filename : str
The filename of the node being compiled.
Returns
-------
result : CodeType
The compiled code object for the node.
"""
assert isinstance(node, Template), 'invalid node'
compiler = cls(filename=filename)
return compiler.visit(node)
def visit_Template(self, node):
cg = self.code_generator
filename = self.filename
# Collect the data needed for the compiler passes.
const_names, param_names = collect_local_names(node)
local_names = set(const_names + param_names)
# Generate the code for the first pass.
first_args = [cmn.SCOPE_KEY] + param_names
first_code, index_map = FirstPassTemplateCompiler.compile(
node, first_args, local_names, filename
)
# Generate the code for the second pass.
second_args = [cmn.NODE_LIST, cmn.T_CONSTS] + param_names
second_code = SecondPassTemplateCompiler.compile(
node, second_args, local_names, index_map, const_names, filename
)
# Prepare the code block for execution.
cmn.fetch_helpers(cg)
# Python 3.11 and 3.12 requires a NULL before a function that is not a method
# Python 3.13 one after
if not PY313 and PY311:
cg.push_null()
cmn.load_helper(cg, 'make_object')
if PY313:
cg.push_null()
cg.call_function()
cg.store_fast(cmn.SCOPE_KEY)
# Load and invoke the first pass code object.
# Python 3.11 and 3.12 requires a NULL before a function that is not a method
# Python 3.13 one after
if not PY313 and PY311:
cg.push_null()
cg.load_const(first_code)
cg.make_function()
if PY313:
cg.push_null()
for arg in first_args:
cg.load_fast(arg)
cg.call_function(len(first_args))
cg.unpack_sequence(2)
cg.store_fast(cmn.NODE_LIST)
cg.store_fast(cmn.T_CONSTS)
# Load and invoke the second pass code object.
# Python 3.11 and 3.12 requires a NULL before a function that is not a method
# Python 3.13 one after
if not PY313 and PY311:
cg.push_null()
cg.load_const(second_code)
cg.make_function()
if PY313:
cg.push_null()
for arg in second_args:
cg.load_fast(arg)
cg.call_function(len(second_args))
cg.pop_top()
# Load the root template compiler node.
cmn.load_node(cg, 0)
# Add the template scope to the node.
cg.dup_top()
# Python 3.11 and 3.12 requires a NULL before a function that is not a method
# Python 3.13 one after
if not PY313 and PY311:
cg.push_null()
cg.rot_two()
cmn.load_helper(cg, 'add_template_scope')
cg.rot_two()
if PY313:
cg.push_null()
cg.rot_two()
cg.load_const(tuple(param_names + const_names))
for name in param_names:
cg.load_fast(name)
cg.build_tuple(len(param_names))
cg.load_fast(cmn.T_CONSTS)
cg.binary_add()
cg.call_function(3)
cg.pop_top()
# Return the template node to the caller.
cg.return_value()
# Setup the parameters and generate the code object.
cg.name = node.name
cg.firstlineno = node.lineno
cg.newlocals = True
cg.args = param_names
cg.docstring = node.docstring
cg.varargs = bool(node.parameters.starparam)
return cg.to_code()
|