blob: f33107ea633bc6f095436efcf36062471d182b55 [file] [log] [blame] [edit]
#!/usr/bin/env python3
#
# Copyright 2018 WebAssembly Community Group participants
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
instructions = [
("unreachable", "makeUnreachable()"),
("nop", "makeNop()"),
("block", "makeBlock(s)"),
("loop", "makeLoop(s)"),
("if", "makeIf(s)"),
("then", "makeThenOrElse(s)"),
("else", "makeThenOrElse(s)"),
("br", "makeBreak(s)"),
("br_if", "makeBreak(s)"),
("br_table", "makeBreakTable(s)"),
("return", "makeReturn(s)"),
("call", "makeCall(s)"),
("call_indirect", "makeCallIndirect(s)"),
("drop", "makeDrop(s)"),
("select", "makeSelect(s)"),
("get_local", "makeGetLocal(s)"),
("set_local", "makeSetLocal(s)"),
("tee_local", "makeTeeLocal(s)"),
("get_global", "makeGetGlobal(s)"),
("set_global", "makeSetGlobal(s)"),
("i32.load", "makeLoad(s, i32, /*isAtomic=*/false)"),
("i64.load", "makeLoad(s, i64, /*isAtomic=*/false)"),
("f32.load", "makeLoad(s, f32, /*isAtomic=*/false)"),
("f64.load", "makeLoad(s, f64, /*isAtomic=*/false)"),
("i32.load8_s", "makeLoad(s, i32, /*isAtomic=*/false)"),
("i32.load8_u", "makeLoad(s, i32, /*isAtomic=*/false)"),
("i32.load16_s", "makeLoad(s, i32, /*isAtomic=*/false)"),
("i32.load16_u", "makeLoad(s, i32, /*isAtomic=*/false)"),
("i64.load8_s", "makeLoad(s, i64, /*isAtomic=*/false)"),
("i64.load8_u", "makeLoad(s, i64, /*isAtomic=*/false)"),
("i64.load16_s", "makeLoad(s, i64, /*isAtomic=*/false)"),
("i64.load16_u", "makeLoad(s, i64, /*isAtomic=*/false)"),
("i64.load32_s", "makeLoad(s, i64, /*isAtomic=*/false)"),
("i64.load32_u", "makeLoad(s, i64, /*isAtomic=*/false)"),
("i32.store", "makeStore(s, i32, /*isAtomic=*/false)"),
("i64.store", "makeStore(s, i64, /*isAtomic=*/false)"),
("f32.store", "makeStore(s, f32, /*isAtomic=*/false)"),
("f64.store", "makeStore(s, f64, /*isAtomic=*/false)"),
("i32.store8", "makeStore(s, i32, /*isAtomic=*/false)"),
("i32.store16", "makeStore(s, i32, /*isAtomic=*/false)"),
("i64.store8", "makeStore(s, i64, /*isAtomic=*/false)"),
("i64.store16", "makeStore(s, i64, /*isAtomic=*/false)"),
("i64.store32", "makeStore(s, i64, /*isAtomic=*/false)"),
("current_memory", "makeHost(s, HostOp::CurrentMemory)"),
("grow_memory", "makeHost(s, HostOp::GrowMemory)"),
("i32.const", "makeConst(s, i32)"),
("i64.const", "makeConst(s, i64)"),
("f32.const", "makeConst(s, f32)"),
("f64.const", "makeConst(s, f64)"),
("i32.eqz", "makeUnary(s, UnaryOp::EqZInt32)"),
("i32.eq", "makeBinary(s, BinaryOp::EqInt32)"),
("i32.ne", "makeBinary(s, BinaryOp::NeInt32)"),
("i32.lt_s", "makeBinary(s, BinaryOp::LtSInt32)"),
("i32.lt_u", "makeBinary(s, BinaryOp::LtUInt32)"),
("i32.gt_s", "makeBinary(s, BinaryOp::GtSInt32)"),
("i32.gt_u", "makeBinary(s, BinaryOp::GtUInt32)"),
("i32.le_s", "makeBinary(s, BinaryOp::LeSInt32)"),
("i32.le_u", "makeBinary(s, BinaryOp::LeUInt32)"),
("i32.ge_s", "makeBinary(s, BinaryOp::GeSInt32)"),
("i32.ge_u", "makeBinary(s, BinaryOp::GeUInt32)"),
("i64.eqz", "makeUnary(s, UnaryOp::EqZInt64)"),
("i64.eq", "makeBinary(s, BinaryOp::EqInt64)"),
("i64.ne", "makeBinary(s, BinaryOp::NeInt64)"),
("i64.lt_s", "makeBinary(s, BinaryOp::LtSInt64)"),
("i64.lt_u", "makeBinary(s, BinaryOp::LtUInt64)"),
("i64.gt_s", "makeBinary(s, BinaryOp::GtSInt64)"),
("i64.gt_u", "makeBinary(s, BinaryOp::GtUInt64)"),
("i64.le_s", "makeBinary(s, BinaryOp::LeSInt64)"),
("i64.le_u", "makeBinary(s, BinaryOp::LeUInt64)"),
("i64.ge_s", "makeBinary(s, BinaryOp::GeSInt64)"),
("i64.ge_u", "makeBinary(s, BinaryOp::GeUInt64)"),
("f32.eq", "makeBinary(s, BinaryOp::EqFloat32)"),
("f32.ne", "makeBinary(s, BinaryOp::NeFloat32)"),
("f32.lt", "makeBinary(s, BinaryOp::LtFloat32)"),
("f32.gt", "makeBinary(s, BinaryOp::GtFloat32)"),
("f32.le", "makeBinary(s, BinaryOp::LeFloat32)"),
("f32.ge", "makeBinary(s, BinaryOp::GeFloat32)"),
("f64.eq", "makeBinary(s, BinaryOp::EqFloat64)"),
("f64.ne", "makeBinary(s, BinaryOp::NeFloat64)"),
("f64.lt", "makeBinary(s, BinaryOp::LtFloat64)"),
("f64.gt", "makeBinary(s, BinaryOp::GtFloat64)"),
("f64.le", "makeBinary(s, BinaryOp::LeFloat64)"),
("f64.ge", "makeBinary(s, BinaryOp::GeFloat64)"),
("i32.clz", "makeUnary(s, UnaryOp::ClzInt32)"),
("i32.ctz", "makeUnary(s, UnaryOp::CtzInt32)"),
("i32.popcnt", "makeUnary(s, UnaryOp::PopcntInt32)"),
("i32.add", "makeBinary(s, BinaryOp::AddInt32)"),
("i32.sub", "makeBinary(s, BinaryOp::SubInt32)"),
("i32.mul", "makeBinary(s, BinaryOp::MulInt32)"),
("i32.div_s", "makeBinary(s, BinaryOp::DivSInt32)"),
("i32.div_u", "makeBinary(s, BinaryOp::DivUInt32)"),
("i32.rem_s", "makeBinary(s, BinaryOp::RemSInt32)"),
("i32.rem_u", "makeBinary(s, BinaryOp::RemUInt32)"),
("i32.and", "makeBinary(s, BinaryOp::AndInt32)"),
("i32.or", "makeBinary(s, BinaryOp::OrInt32)"),
("i32.xor", "makeBinary(s, BinaryOp::XorInt32)"),
("i32.shl", "makeBinary(s, BinaryOp::ShlInt32)"),
("i32.shr_s", "makeBinary(s, BinaryOp::ShrSInt32)"),
("i32.shr_u", "makeBinary(s, BinaryOp::ShrUInt32)"),
("i32.rotl", "makeBinary(s, BinaryOp::RotLInt32)"),
("i32.rotr", "makeBinary(s, BinaryOp::RotRInt32)"),
("i64.clz", "makeUnary(s, UnaryOp::ClzInt64)"),
("i64.ctz", "makeUnary(s, UnaryOp::CtzInt64)"),
("i64.popcnt", "makeUnary(s, UnaryOp::PopcntInt64)"),
("i64.add", "makeBinary(s, BinaryOp::AddInt64)"),
("i64.sub", "makeBinary(s, BinaryOp::SubInt64)"),
("i64.mul", "makeBinary(s, BinaryOp::MulInt64)"),
("i64.div_s", "makeBinary(s, BinaryOp::DivSInt64)"),
("i64.div_u", "makeBinary(s, BinaryOp::DivUInt64)"),
("i64.rem_s", "makeBinary(s, BinaryOp::RemSInt64)"),
("i64.rem_u", "makeBinary(s, BinaryOp::RemUInt64)"),
("i64.and", "makeBinary(s, BinaryOp::AndInt64)"),
("i64.or", "makeBinary(s, BinaryOp::OrInt64)"),
("i64.xor", "makeBinary(s, BinaryOp::XorInt64)"),
("i64.shl", "makeBinary(s, BinaryOp::ShlInt64)"),
("i64.shr_s", "makeBinary(s, BinaryOp::ShrSInt64)"),
("i64.shr_u", "makeBinary(s, BinaryOp::ShrUInt64)"),
("i64.rotl", "makeBinary(s, BinaryOp::RotLInt64)"),
("i64.rotr", "makeBinary(s, BinaryOp::RotRInt64)"),
("f32.abs", "makeUnary(s, UnaryOp::AbsFloat32)"),
("f32.neg", "makeUnary(s, UnaryOp::NegFloat32)"),
("f32.ceil", "makeUnary(s, UnaryOp::CeilFloat32)"),
("f32.floor", "makeUnary(s, UnaryOp::FloorFloat32)"),
("f32.trunc", "makeUnary(s, UnaryOp::TruncFloat32)"),
("f32.nearest", "makeUnary(s, UnaryOp::NearestFloat32)"),
("f32.sqrt", "makeUnary(s, UnaryOp::SqrtFloat32)"),
("f32.add", "makeBinary(s, BinaryOp::AddFloat32)"),
("f32.sub", "makeBinary(s, BinaryOp::SubFloat32)"),
("f32.mul", "makeBinary(s, BinaryOp::MulFloat32)"),
("f32.div", "makeBinary(s, BinaryOp::DivFloat32)"),
("f32.min", "makeBinary(s, BinaryOp::MinFloat32)"),
("f32.max", "makeBinary(s, BinaryOp::MaxFloat32)"),
("f32.copysign", "makeBinary(s, BinaryOp::CopySignFloat32)"),
("f64.abs", "makeUnary(s, UnaryOp::AbsFloat64)"),
("f64.neg", "makeUnary(s, UnaryOp::NegFloat64)"),
("f64.ceil", "makeUnary(s, UnaryOp::CeilFloat64)"),
("f64.floor", "makeUnary(s, UnaryOp::FloorFloat64)"),
("f64.trunc", "makeUnary(s, UnaryOp::TruncFloat64)"),
("f64.nearest", "makeUnary(s, UnaryOp::NearestFloat64)"),
("f64.sqrt", "makeUnary(s, UnaryOp::SqrtFloat64)"),
("f64.add", "makeBinary(s, BinaryOp::AddFloat64)"),
("f64.sub", "makeBinary(s, BinaryOp::SubFloat64)"),
("f64.mul", "makeBinary(s, BinaryOp::MulFloat64)"),
("f64.div", "makeBinary(s, BinaryOp::DivFloat64)"),
("f64.min", "makeBinary(s, BinaryOp::MinFloat64)"),
("f64.max", "makeBinary(s, BinaryOp::MaxFloat64)"),
("f64.copysign", "makeBinary(s, BinaryOp::CopySignFloat64)"),
("i32.wrap/i64", "makeUnary(s, UnaryOp::WrapInt64)"),
("i32.trunc_s/f32", "makeUnary(s, UnaryOp::TruncSFloat32ToInt32)"),
("i32.trunc_u/f32", "makeUnary(s, UnaryOp::TruncUFloat32ToInt32)"),
("i32.trunc_s/f64", "makeUnary(s, UnaryOp::TruncSFloat64ToInt32)"),
("i32.trunc_u/f64", "makeUnary(s, UnaryOp::TruncUFloat64ToInt32)"),
("i64.extend_s/i32", "makeUnary(s, UnaryOp::ExtendSInt32)"),
("i64.extend_u/i32", "makeUnary(s, UnaryOp::ExtendUInt32)"),
("i64.trunc_s/f32", "makeUnary(s, UnaryOp::TruncSFloat32ToInt64)"),
("i64.trunc_u/f32", "makeUnary(s, UnaryOp::TruncUFloat32ToInt64)"),
("i64.trunc_s/f64", "makeUnary(s, UnaryOp::TruncSFloat64ToInt64)"),
("i64.trunc_u/f64", "makeUnary(s, UnaryOp::TruncUFloat64ToInt64)"),
("f32.convert_s/i32", "makeUnary(s, UnaryOp::ConvertSInt32ToFloat32)"),
("f32.convert_u/i32", "makeUnary(s, UnaryOp::ConvertUInt32ToFloat32)"),
("f32.convert_s/i64", "makeUnary(s, UnaryOp::ConvertSInt64ToFloat32)"),
("f32.convert_u/i64", "makeUnary(s, UnaryOp::ConvertUInt64ToFloat32)"),
("f32.demote/f64", "makeUnary(s, UnaryOp::DemoteFloat64)"),
("f64.convert_s/i32", "makeUnary(s, UnaryOp::ConvertSInt32ToFloat64)"),
("f64.convert_u/i32", "makeUnary(s, UnaryOp::ConvertUInt32ToFloat64)"),
("f64.convert_s/i64", "makeUnary(s, UnaryOp::ConvertSInt64ToFloat64)"),
("f64.convert_u/i64", "makeUnary(s, UnaryOp::ConvertUInt64ToFloat64)"),
("f64.promote/f32", "makeUnary(s, UnaryOp::PromoteFloat32)"),
("i32.reinterpret/f32", "makeUnary(s, UnaryOp::ReinterpretFloat32)"),
("i64.reinterpret/f64", "makeUnary(s, UnaryOp::ReinterpretFloat64)"),
("f32.reinterpret/i32", "makeUnary(s, UnaryOp::ReinterpretInt32)"),
("f64.reinterpret/i64", "makeUnary(s, UnaryOp::ReinterpretInt64)"),
("i32.extend8_s", "makeUnary(s, UnaryOp::ExtendS8Int32)"),
("i32.extend16_s", "makeUnary(s, UnaryOp::ExtendS16Int32)"),
("i64.extend8_s", "makeUnary(s, UnaryOp::ExtendS8Int64)"),
("i64.extend16_s", "makeUnary(s, UnaryOp::ExtendS16Int64)"),
("i64.extend32_s", "makeUnary(s, UnaryOp::ExtendS32Int64)"),
# atomic instructions
("wake", "makeAtomicWake(s)"),
("i32.wait", "makeAtomicWait(s, i32)"),
("i64.wait", "makeAtomicWait(s, i64)"),
("i32.atomic.load8_u", "makeLoad(s, i32, /*isAtomic=*/true)"),
("i32.atomic.load16_u", "makeLoad(s, i32, /*isAtomic=*/true)"),
("i32.atomic.load", "makeLoad(s, i32, /*isAtomic=*/true)"),
("i64.atomic.load8_u", "makeLoad(s, i64, /*isAtomic=*/true)"),
("i64.atomic.load16_u", "makeLoad(s, i64, /*isAtomic=*/true)"),
("i64.atomic.load32_u", "makeLoad(s, i64, /*isAtomic=*/true)"),
("i64.atomic.load", "makeLoad(s, i64, /*isAtomic=*/true)"),
("i32.atomic.store8", "makeStore(s, i32, /*isAtomic=*/true)"),
("i32.atomic.store16", "makeStore(s, i32, /*isAtomic=*/true)"),
("i32.atomic.store", "makeStore(s, i32, /*isAtomic=*/true)"),
("i64.atomic.store8", "makeStore(s, i64, /*isAtomic=*/true)"),
("i64.atomic.store16", "makeStore(s, i64, /*isAtomic=*/true)"),
("i64.atomic.store32", "makeStore(s, i64, /*isAtomic=*/true)"),
("i64.atomic.store", "makeStore(s, i64, /*isAtomic=*/true)"),
("i32.atomic.rmw8_u.add", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.add", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.add", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.add", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.add", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.add", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.add", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i32.atomic.rmw8_u.sub", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.sub", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.sub", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.sub", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.sub", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.sub", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.sub", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i32.atomic.rmw8_u.and", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.and", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.and", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.and", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.and", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.and", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.and", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i32.atomic.rmw8_u.or", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.or", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.or", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.or", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.or", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.or", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.or", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i32.atomic.rmw8_u.xor", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.xor", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.xor", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.xor", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.xor", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.xor", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.xor", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i32.atomic.rmw8_u.xchg", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.xchg", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.xchg", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.xchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.xchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.xchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.xchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i32.atomic.rmw8_u.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw16_u.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i32.atomic.rmw.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i32)"),
("i64.atomic.rmw8_u.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw16_u.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw32_u.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
("i64.atomic.rmw.cmpxchg", "makeAtomicRMWOrCmpxchg(s, i64)"),
# nontrapping float-to-int instructions
("i32.trunc_s:sat/f32", "makeUnary(s, UnaryOp::TruncSatSFloat32ToInt32)"),
("i32.trunc_u:sat/f32", "makeUnary(s, UnaryOp::TruncSatUFloat32ToInt32)"),
("i32.trunc_s:sat/f64", "makeUnary(s, UnaryOp::TruncSatSFloat64ToInt32)"),
("i32.trunc_u:sat/f64", "makeUnary(s, UnaryOp::TruncSatUFloat64ToInt32)"),
("i64.trunc_s:sat/f32", "makeUnary(s, UnaryOp::TruncSatSFloat32ToInt64)"),
("i64.trunc_u:sat/f32", "makeUnary(s, UnaryOp::TruncSatUFloat32ToInt64)"),
("i64.trunc_s:sat/f64", "makeUnary(s, UnaryOp::TruncSatSFloat64ToInt64)"),
("i64.trunc_u:sat/f64", "makeUnary(s, UnaryOp::TruncSatUFloat64ToInt64)"),
]
class CodePrinter:
indents = 0
def __enter__(self):
CodePrinter.indents += 1
def __exit__(self, *args):
CodePrinter.indents -= 1
def indent(self):
# call in a 'with' statement
return self
def print_line(self, line):
print(" " * CodePrinter.indents + line)
class Node:
def __init__(self, expr=None, children=None, inst=None):
# the expression to return if this is the string has ended
self.expr = expr
# map unique strings to children nodes
self.children = children if children else {}
# full instruction leading to this node
self.inst = inst
def _common_prefix(a, b):
"""Return the common prefix of two strings."""
prefix = []
while a and b and a[0] == b[0]:
prefix.append(a[0])
a = a[1:]
b = b[1:]
return "".join(prefix)
def do_insert(self, full_inst, inst, expr):
if inst is "":
assert self.expr is None, "Repeated instruction"
self.expr = expr
self.inst = full_inst
return
# find key with shared prefix
prefix, key = "", None
for k in self.children:
prefix = Node._common_prefix(inst, k)
if prefix is not "":
key = k
break
if key is None:
# unique prefix, insert and stop
self.children[inst] = Node(expr, inst=full_inst)
return
key_remainder = key[len(prefix):]
if key_remainder is not "":
# split key and move everything after the prefix to a new node
child = self.children.pop(key)
self.children[prefix] = Node(children={key_remainder: child})
# update key for recursive insert
key = prefix
# chop off prefix and recurse
self.children[key].do_insert(full_inst, inst[len(key):], expr)
def insert(self, inst, expr):
self.do_insert(inst, inst, expr)
def instruction_parser():
"""Build a trie out of all the instructions, then emit it as C++ code."""
trie = Node()
inst_length = 0
for inst, expr in instructions:
inst_length = max(inst_length, len(inst))
trie.insert(inst, expr)
printer = CodePrinter()
printer.print_line("char op[{}] = {{'\\0'}};".format(inst_length + 1))
printer.print_line("strncpy(op, s[0]->c_str(), {});".format(inst_length))
def print_leaf(expr, inst):
printer.print_line("if (strcmp(op, \"{inst}\") == 0) return {expr};"
.format(inst=inst, expr=expr))
printer.print_line("goto parse_error;")
def emit(node, idx=0):
assert node.children
printer.print_line("switch (op[{}]) {{".format(idx))
with printer.indent():
if node.expr:
printer.print_line("case '\\0':")
with printer.indent():
print_leaf(node.expr, node.inst)
children = sorted(node.children.items(), key=lambda pair: pair[0])
for prefix, child in children:
if child.children:
printer.print_line("case '{}': {{".format(prefix[0]))
with printer.indent():
emit(child, idx + len(prefix))
printer.print_line("}")
else:
assert child.expr
printer.print_line("case '{}':".format(prefix[0]))
with printer.indent():
print_leaf(child.expr, child.inst)
printer.print_line("default: goto parse_error;")
printer.print_line("}")
emit(trie)
printer.print_line("parse_error:")
with printer.indent():
printer.print_line("throw ParseException(std::string(op));")
def print_header():
print("// DO NOT EDIT! This file generated by scripts/gen-s-parser.py\n")
def generate_with_guard(generator, guard):
print("#ifdef {}".format(guard))
print("#undef {}".format(guard))
generator()
print("#endif // {}".format(guard))
def main():
print_header()
generate_with_guard(instruction_parser, "INSTRUCTION_PARSER")
if __name__ == "__main__":
main()