mirror of
https://github.com/python/cpython.git
synced 2025-10-31 21:51:50 +00:00
gh-104584: Emit macro expansions to opcode_metadata.h (#106163)
This produces longer traces (superblocks?). Also improved debug output (uop names are now printed instead of numeric opcodes). This would be simpler if the numeric opcode values were generated by generate_cases.py, but that's another project. Refactored some code in generate_cases.py so the essential algorithm for cache effects is only run once. (Deciding which effects are used and what the total cache size is, regardless of what's used.)
This commit is contained in:
parent
c283a0cff5
commit
11731434df
4 changed files with 153 additions and 78 deletions
|
|
@ -300,6 +300,13 @@ def emit_macros(cls, out: Formatter):
|
|||
f"(_PyOpcode_opcode_metadata[(OP)].flags & ({name}))")
|
||||
|
||||
|
||||
@dataclasses.dataclass
|
||||
class ActiveCacheEffect:
|
||||
"""Wraps a CacheEffect that is actually used, in context."""
|
||||
effect: parser.CacheEffect
|
||||
offset: int
|
||||
|
||||
|
||||
FORBIDDEN_NAMES_IN_UOPS = (
|
||||
"resume_with_error", # Proxy for "goto", which isn't an IDENTIFIER
|
||||
"unbound_local_error",
|
||||
|
|
@ -344,6 +351,7 @@ class Instruction:
|
|||
unmoved_names: frozenset[str]
|
||||
instr_fmt: str
|
||||
instr_flags: InstructionFlags
|
||||
active_caches: list[ActiveCacheEffect]
|
||||
|
||||
# Set later
|
||||
family: parser.Family | None = None
|
||||
|
|
@ -375,15 +383,19 @@ def __init__(self, inst: parser.InstDef):
|
|||
|
||||
self.instr_flags = InstructionFlags.fromInstruction(inst)
|
||||
|
||||
self.active_caches = []
|
||||
offset = 0
|
||||
for effect in self.cache_effects:
|
||||
if effect.name != UNUSED:
|
||||
self.active_caches.append(ActiveCacheEffect(effect, offset))
|
||||
offset += effect.size
|
||||
|
||||
if self.instr_flags.HAS_ARG_FLAG:
|
||||
fmt = "IB"
|
||||
else:
|
||||
fmt = "IX"
|
||||
cache = "C"
|
||||
for ce in self.cache_effects:
|
||||
for _ in range(ce.size):
|
||||
fmt += cache
|
||||
cache = "0"
|
||||
if offset:
|
||||
fmt += "C" + "0"*(offset-1)
|
||||
self.instr_fmt = fmt
|
||||
|
||||
def is_viable_uop(self) -> bool:
|
||||
|
|
@ -392,18 +404,11 @@ def is_viable_uop(self) -> bool:
|
|||
return False
|
||||
if self.instr_flags.HAS_ARG_FLAG:
|
||||
# If the instruction uses oparg, it cannot use any caches
|
||||
for c in self.cache_effects:
|
||||
if c.name != UNUSED:
|
||||
return False
|
||||
if self.active_caches:
|
||||
return False
|
||||
else:
|
||||
# If it doesn't use oparg, it can have one cache entry
|
||||
caches: list[parser.CacheEffect] = []
|
||||
cache_offset = 0
|
||||
for c in self.cache_effects:
|
||||
if c.name != UNUSED:
|
||||
caches.append(c)
|
||||
cache_offset += c.size
|
||||
if len(caches) > 1:
|
||||
if len(self.active_caches) > 1:
|
||||
return False
|
||||
for forbidden in FORBIDDEN_NAMES_IN_UOPS:
|
||||
# TODO: Don't check in '#ifdef ENABLE_SPECIALIZATION' regions
|
||||
|
|
@ -458,7 +463,7 @@ def write(self, out: Formatter, tier: Tiers = TIER_ONE) -> None:
|
|||
|
||||
# out.emit(f"next_instr += OPSIZE({self.inst.name}) - 1;")
|
||||
|
||||
self.write_body(out, 0, tier=tier)
|
||||
self.write_body(out, 0, self.active_caches, tier=tier)
|
||||
|
||||
# Skip the rest if the block always exits
|
||||
if self.always_exits:
|
||||
|
|
@ -492,33 +497,30 @@ def write_body(
|
|||
self,
|
||||
out: Formatter,
|
||||
dedent: int,
|
||||
cache_adjust: int = 0,
|
||||
active_caches: list[ActiveCacheEffect],
|
||||
tier: Tiers = TIER_ONE,
|
||||
) -> None:
|
||||
"""Write the instruction body."""
|
||||
# Write cache effect variable declarations and initializations
|
||||
cache_offset = cache_adjust
|
||||
for ceffect in self.cache_effects:
|
||||
if ceffect.name != UNUSED:
|
||||
bits = ceffect.size * BITS_PER_CODE_UNIT
|
||||
if bits == 64:
|
||||
# NOTE: We assume that 64-bit data in the cache
|
||||
# is always an object pointer.
|
||||
# If this becomes false, we need a way to specify
|
||||
# syntactically what type the cache data is.
|
||||
typ = "PyObject *"
|
||||
func = "read_obj"
|
||||
else:
|
||||
typ = f"uint{bits}_t "
|
||||
func = f"read_u{bits}"
|
||||
if tier == TIER_ONE:
|
||||
out.emit(
|
||||
f"{typ}{ceffect.name} = {func}(&next_instr[{cache_offset}].cache);"
|
||||
)
|
||||
else:
|
||||
out.emit(f"{typ}{ceffect.name} = operand;")
|
||||
cache_offset += ceffect.size
|
||||
assert cache_offset == self.cache_offset + cache_adjust
|
||||
for active in active_caches:
|
||||
ceffect = active.effect
|
||||
bits = ceffect.size * BITS_PER_CODE_UNIT
|
||||
if bits == 64:
|
||||
# NOTE: We assume that 64-bit data in the cache
|
||||
# is always an object pointer.
|
||||
# If this becomes false, we need a way to specify
|
||||
# syntactically what type the cache data is.
|
||||
typ = "PyObject *"
|
||||
func = "read_obj"
|
||||
else:
|
||||
typ = f"uint{bits}_t "
|
||||
func = f"read_u{bits}"
|
||||
if tier == TIER_ONE:
|
||||
out.emit(
|
||||
f"{typ}{ceffect.name} = {func}(&next_instr[{active.offset}].cache);"
|
||||
)
|
||||
else:
|
||||
out.emit(f"{typ}{ceffect.name} = operand;")
|
||||
|
||||
# Write the body, substituting a goto for ERROR_IF() and other stuff
|
||||
assert dedent <= 0
|
||||
|
|
@ -583,8 +585,9 @@ class Component:
|
|||
instr: Instruction
|
||||
input_mapping: StackEffectMapping
|
||||
output_mapping: StackEffectMapping
|
||||
active_caches: list[ActiveCacheEffect]
|
||||
|
||||
def write_body(self, out: Formatter, cache_adjust: int) -> None:
|
||||
def write_body(self, out: Formatter) -> None:
|
||||
with out.block(""):
|
||||
input_names = {ieffect.name for _, ieffect in self.input_mapping}
|
||||
for var, ieffect in self.input_mapping:
|
||||
|
|
@ -593,7 +596,7 @@ def write_body(self, out: Formatter, cache_adjust: int) -> None:
|
|||
if oeffect.name not in input_names:
|
||||
out.declare(oeffect, None)
|
||||
|
||||
self.instr.write_body(out, dedent=-4, cache_adjust=cache_adjust)
|
||||
self.instr.write_body(out, -4, self.active_caches)
|
||||
|
||||
for var, oeffect in self.output_mapping:
|
||||
out.assign(var, oeffect)
|
||||
|
|
@ -611,6 +614,7 @@ class MacroInstruction:
|
|||
instr_flags: InstructionFlags
|
||||
macro: parser.Macro
|
||||
parts: list[Component | parser.CacheEffect]
|
||||
cache_offset: int
|
||||
predicted: bool = False
|
||||
|
||||
|
||||
|
|
@ -873,11 +877,11 @@ def effect_counts(self, name: str) -> tuple[int, int, int]:
|
|||
cache = instr.cache_offset
|
||||
input = len(instr.input_effects)
|
||||
output = len(instr.output_effects)
|
||||
elif macro := self.macro_instrs.get(name):
|
||||
cache, input, output = 0, 0, 0
|
||||
for part in macro.parts:
|
||||
elif mac := self.macro_instrs.get(name):
|
||||
cache = mac.cache_offset
|
||||
input, output = 0, 0
|
||||
for part in mac.parts:
|
||||
if isinstance(part, Component):
|
||||
cache += part.instr.cache_offset
|
||||
# A component may pop what the previous component pushed,
|
||||
# so we offset the input/output counts by that.
|
||||
delta_i = len(part.instr.input_effects)
|
||||
|
|
@ -885,9 +889,6 @@ def effect_counts(self, name: str) -> tuple[int, int, int]:
|
|||
offset = min(delta_i, output)
|
||||
input += delta_i - offset
|
||||
output += delta_o - offset
|
||||
else:
|
||||
assert isinstance(part, parser.CacheEffect), part
|
||||
cache += part.size
|
||||
else:
|
||||
assert False, f"Unknown instruction {name!r}"
|
||||
return cache, input, output
|
||||
|
|
@ -906,29 +907,25 @@ def analyze_macro(self, macro: parser.Macro) -> MacroInstruction:
|
|||
stack, initial_sp = self.stack_analysis(components)
|
||||
sp = initial_sp
|
||||
parts: list[Component | parser.CacheEffect] = []
|
||||
format = "IB"
|
||||
flags = InstructionFlags.newEmpty()
|
||||
cache = "C"
|
||||
offset = 0
|
||||
for component in components:
|
||||
match component:
|
||||
case parser.CacheEffect() as ceffect:
|
||||
parts.append(ceffect)
|
||||
for _ in range(ceffect.size):
|
||||
format += cache
|
||||
cache = "0"
|
||||
offset += ceffect.size
|
||||
case Instruction() as instr:
|
||||
part, sp = self.analyze_instruction(instr, stack, sp)
|
||||
part, sp, offset = self.analyze_instruction(instr, stack, sp, offset)
|
||||
parts.append(part)
|
||||
for ce in instr.cache_effects:
|
||||
for _ in range(ce.size):
|
||||
format += cache
|
||||
cache = "0"
|
||||
flags.add(instr.instr_flags)
|
||||
case _:
|
||||
typing.assert_never(component)
|
||||
final_sp = sp
|
||||
format = "IB"
|
||||
if offset:
|
||||
format += "C" + "0"*(offset-1)
|
||||
return MacroInstruction(
|
||||
macro.name, stack, initial_sp, final_sp, format, flags, macro, parts
|
||||
macro.name, stack, initial_sp, final_sp, format, flags, macro, parts, offset
|
||||
)
|
||||
|
||||
def analyze_pseudo(self, pseudo: parser.Pseudo) -> PseudoInstruction:
|
||||
|
|
@ -941,8 +938,8 @@ def analyze_pseudo(self, pseudo: parser.Pseudo) -> PseudoInstruction:
|
|||
return PseudoInstruction(pseudo.name, targets, fmts[0], targets[0].instr_flags)
|
||||
|
||||
def analyze_instruction(
|
||||
self, instr: Instruction, stack: list[StackEffect], sp: int
|
||||
) -> tuple[Component, int]:
|
||||
self, instr: Instruction, stack: list[StackEffect], sp: int, offset: int
|
||||
) -> tuple[Component, int, int]:
|
||||
input_mapping: StackEffectMapping = []
|
||||
for ieffect in reversed(instr.input_effects):
|
||||
sp -= 1
|
||||
|
|
@ -951,7 +948,12 @@ def analyze_instruction(
|
|||
for oeffect in instr.output_effects:
|
||||
output_mapping.append((stack[sp], oeffect))
|
||||
sp += 1
|
||||
return Component(instr, input_mapping, output_mapping), sp
|
||||
active_effects: list[ActiveCacheEffect] = []
|
||||
for ceffect in instr.cache_effects:
|
||||
if ceffect.name != UNUSED:
|
||||
active_effects.append(ActiveCacheEffect(ceffect, offset))
|
||||
offset += ceffect.size
|
||||
return Component(instr, input_mapping, output_mapping, active_effects), sp, offset
|
||||
|
||||
def check_macro_components(
|
||||
self, macro: parser.Macro
|
||||
|
|
@ -1030,7 +1032,7 @@ def stack_analysis(
|
|||
|
||||
def get_stack_effect_info(
|
||||
self, thing: parser.InstDef | parser.Macro | parser.Pseudo
|
||||
) -> tuple[AnyInstruction | None, str, str]:
|
||||
) -> tuple[AnyInstruction | None, str | None, str | None]:
|
||||
def effect_str(effects: list[StackEffect]) -> str:
|
||||
n_effect, sym_effect = list_effect_size(effects)
|
||||
if sym_effect:
|
||||
|
|
@ -1108,6 +1110,7 @@ def write_stack_effect_functions(self) -> None:
|
|||
continue
|
||||
instr, popped, pushed = self.get_stack_effect_info(thing)
|
||||
if instr is not None:
|
||||
assert popped is not None and pushed is not None
|
||||
popped_data.append((instr, popped))
|
||||
pushed_data.append((instr, pushed))
|
||||
|
||||
|
|
@ -1182,7 +1185,8 @@ def write_metadata(self) -> None:
|
|||
|
||||
self.write_pseudo_instrs()
|
||||
|
||||
self.write_uop_defines()
|
||||
self.out.emit("")
|
||||
self.write_uop_items(lambda name, counter: f"#define {name} {counter}")
|
||||
|
||||
self.write_stack_effect_functions()
|
||||
|
||||
|
|
@ -1213,6 +1217,9 @@ def write_metadata(self) -> None:
|
|||
self.out.emit("#ifndef NEED_OPCODE_METADATA")
|
||||
self.out.emit("extern const struct opcode_metadata _PyOpcode_opcode_metadata[512];")
|
||||
self.out.emit("extern const struct opcode_macro_expansion _PyOpcode_macro_expansion[256];")
|
||||
self.out.emit("#ifdef Py_DEBUG")
|
||||
self.out.emit("extern const char * const _PyOpcode_uop_name[512];")
|
||||
self.out.emit("#endif")
|
||||
self.out.emit("#else")
|
||||
|
||||
self.out.emit("const struct opcode_metadata _PyOpcode_opcode_metadata[512] = {")
|
||||
|
|
@ -1246,19 +1253,27 @@ def write_metadata(self) -> None:
|
|||
pass
|
||||
case parser.InstDef(name=name):
|
||||
instr = self.instrs[name]
|
||||
# Since an 'op' is not a bytecode, it has no expansion
|
||||
if instr.kind != "op" and instr.is_viable_uop():
|
||||
# Double check there aren't any used cache effects.
|
||||
# If this fails, see write_macro_expansions().
|
||||
assert not instr.active_caches, (instr.name, instr.cache_effects)
|
||||
self.out.emit(
|
||||
f"[{name}] = "
|
||||
f"{{ .nuops = 1, .uops = {{ {{ {name}, 0, 0 }} }} }},"
|
||||
)
|
||||
case parser.Macro():
|
||||
# TODO: emit expansion if all parts are viable uops
|
||||
pass
|
||||
self.write_macro_expansions(self.macro_instrs[thing.name])
|
||||
case parser.Pseudo():
|
||||
pass
|
||||
case _:
|
||||
typing.assert_never(thing)
|
||||
|
||||
self.out.emit("#ifdef Py_DEBUG")
|
||||
with self.out.block("const char * const _PyOpcode_uop_name[512] =", ";"):
|
||||
self.write_uop_items(lambda name, counter: f"[{counter}] = \"{name}\",")
|
||||
self.out.emit("#endif")
|
||||
|
||||
self.out.emit("#endif")
|
||||
|
||||
with open(self.pymetadata_filename, "w") as f:
|
||||
|
|
@ -1300,13 +1315,12 @@ def write_pseudo_instrs(self) -> None:
|
|||
self.out.emit(f" ((OP) == {op}) || \\")
|
||||
self.out.emit(f" 0")
|
||||
|
||||
def write_uop_defines(self) -> None:
|
||||
def write_uop_items(self, make_text: typing.Callable[[str, int], str]) -> None:
|
||||
"""Write '#define XXX NNN' for each uop"""
|
||||
self.out.emit("")
|
||||
counter = 300
|
||||
counter = 300 # TODO: Avoid collision with pseudo instructions
|
||||
def add(name: str) -> None:
|
||||
nonlocal counter
|
||||
self.out.emit(f"#define {name} {counter}")
|
||||
self.out.emit(make_text(name, counter))
|
||||
counter += 1
|
||||
add("EXIT_TRACE")
|
||||
add("SET_IP")
|
||||
|
|
@ -1314,6 +1328,32 @@ def add(name: str) -> None:
|
|||
if instr.kind == "op" and instr.is_viable_uop():
|
||||
add(instr.name)
|
||||
|
||||
def write_macro_expansions(self, mac: MacroInstruction) -> None:
|
||||
"""Write the macro expansions for a macro-instruction."""
|
||||
# TODO: Refactor to share code with write_cody(), is_viaible_uop(), etc.
|
||||
offset = 0 # Cache effect offset
|
||||
expansions: list[tuple[str, int, int]] = [] # [(name, size, offset), ...]
|
||||
for part in mac.parts:
|
||||
if isinstance(part, Component):
|
||||
# All component instructions must be viable uops
|
||||
if not part.instr.is_viable_uop():
|
||||
print(f"NOTE: Part {part.instr.name} of {mac.name} is not a viable uop")
|
||||
return
|
||||
if part.instr.instr_flags.HAS_ARG_FLAG or not part.active_caches:
|
||||
size, offset = 0, 0
|
||||
else:
|
||||
# If this assert triggers, is_viable_uops() lied
|
||||
assert len(part.active_caches) == 1, (mac.name, part.instr.name)
|
||||
cache = part.active_caches[0]
|
||||
size, offset = cache.effect.size, cache.offset
|
||||
expansions.append((part.instr.name, size, offset))
|
||||
assert len(expansions) > 0, f"Macro {mac.name} has empty expansion?!"
|
||||
pieces = [f"{{ {name}, {size}, {offset} }}" for name, size, offset in expansions]
|
||||
self.out.emit(
|
||||
f"[{mac.name}] = "
|
||||
f"{{ .nuops = {len(expansions)}, .uops = {{ {', '.join(pieces)} }} }},"
|
||||
)
|
||||
|
||||
def emit_metadata_entry(
|
||||
self, name: str, fmt: str, flags: InstructionFlags
|
||||
) -> None:
|
||||
|
|
@ -1379,6 +1419,7 @@ def write_executor_instructions(self) -> None:
|
|||
for thing in self.everything:
|
||||
match thing:
|
||||
case OverriddenInstructionPlaceHolder():
|
||||
# TODO: Is this helpful?
|
||||
self.write_overridden_instr_place_holder(thing)
|
||||
case parser.InstDef():
|
||||
instr = self.instrs[thing.name]
|
||||
|
|
@ -1388,7 +1429,7 @@ def write_executor_instructions(self) -> None:
|
|||
instr.write(self.out, tier=TIER_TWO)
|
||||
self.out.emit("break;")
|
||||
case parser.Macro():
|
||||
pass # TODO
|
||||
pass
|
||||
case parser.Pseudo():
|
||||
pass
|
||||
case _:
|
||||
|
|
@ -1429,7 +1470,7 @@ def write_macro(self, mac: MacroInstruction) -> None:
|
|||
cache_adjust += size
|
||||
case Component() as comp:
|
||||
last_instr = comp.instr
|
||||
comp.write_body(self.out, cache_adjust)
|
||||
comp.write_body(self.out)
|
||||
cache_adjust += comp.instr.cache_offset
|
||||
|
||||
if cache_adjust:
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue