blob: 1baed3964299eef077e3fe1040de0d6b99f1d943 [file] [log] [blame]
from __future__ import annotations
import unittest
from mypyc.codegen.emit import Emitter, EmitterContext
from mypyc.common import HAVE_IMMORTAL
from mypyc.ir.class_ir import ClassIR
from mypyc.ir.ops import BasicBlock, Register, Value
from mypyc.ir.rtypes import (
RInstance,
RTuple,
RUnion,
bool_rprimitive,
int_rprimitive,
list_rprimitive,
none_rprimitive,
object_rprimitive,
str_rprimitive,
)
from mypyc.irbuild.vtable import compute_vtable
from mypyc.namegen import NameGenerator
class TestEmitter(unittest.TestCase):
def setUp(self) -> None:
self.n = Register(int_rprimitive, "n")
self.context = EmitterContext(NameGenerator([["mod"]]))
self.emitter = Emitter(self.context, {})
ir = ClassIR("A", "mod")
compute_vtable(ir)
ir.mro = [ir]
self.instance_a = RInstance(ir)
def test_label(self) -> None:
assert self.emitter.label(BasicBlock(4)) == "CPyL4"
def test_reg(self) -> None:
names: dict[Value, str] = {self.n: "n"}
emitter = Emitter(self.context, names)
assert emitter.reg(self.n) == "cpy_r_n"
def test_object_annotation(self) -> None:
assert self.emitter.object_annotation("hello, world", "line;") == " /* 'hello, world' */"
assert (
self.emitter.object_annotation(list(range(30)), "line;")
== """\
/* [0, 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] */"""
)
def test_emit_line(self) -> None:
emitter = self.emitter
emitter.emit_line("line;")
emitter.emit_line("a {")
emitter.emit_line("f();")
emitter.emit_line("}")
assert emitter.fragments == ["line;\n", "a {\n", " f();\n", "}\n"]
emitter = Emitter(self.context, {})
emitter.emit_line("CPyStatics[0];", ann="hello, world")
emitter.emit_line("CPyStatics[1];", ann=list(range(30)))
assert emitter.fragments[0] == "CPyStatics[0]; /* 'hello, world' */\n"
assert (
emitter.fragments[1]
== """\
CPyStatics[1]; /* [0, 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] */\n"""
)
def test_emit_undefined_value_for_simple_type(self) -> None:
emitter = self.emitter
assert emitter.c_undefined_value(int_rprimitive) == "CPY_INT_TAG"
assert emitter.c_undefined_value(str_rprimitive) == "NULL"
assert emitter.c_undefined_value(bool_rprimitive) == "2"
def test_emit_undefined_value_for_tuple(self) -> None:
emitter = self.emitter
assert (
emitter.c_undefined_value(RTuple([str_rprimitive, int_rprimitive, bool_rprimitive]))
== "(tuple_T3OIC) { NULL, CPY_INT_TAG, 2 }"
)
assert emitter.c_undefined_value(RTuple([str_rprimitive])) == "(tuple_T1O) { NULL }"
assert (
emitter.c_undefined_value(RTuple([RTuple([str_rprimitive]), bool_rprimitive]))
== "(tuple_T2T1OC) { { NULL }, 2 }"
)
def test_emit_inc_ref_object(self) -> None:
self.emitter.emit_inc_ref("x", object_rprimitive)
self.assert_output("CPy_INCREF(x);\n")
def test_emit_inc_ref_int(self) -> None:
self.emitter.emit_inc_ref("x", int_rprimitive)
self.assert_output("CPyTagged_INCREF(x);\n")
def test_emit_inc_ref_rare(self) -> None:
self.emitter.emit_inc_ref("x", object_rprimitive, rare=True)
self.assert_output("CPy_INCREF(x);\n")
self.emitter.emit_inc_ref("x", int_rprimitive, rare=True)
self.assert_output("CPyTagged_IncRef(x);\n")
def test_emit_inc_ref_list(self) -> None:
self.emitter.emit_inc_ref("x", list_rprimitive)
if HAVE_IMMORTAL:
self.assert_output("CPy_INCREF_NO_IMM(x);\n")
else:
self.assert_output("CPy_INCREF(x);\n")
def test_emit_inc_ref_instance(self) -> None:
self.emitter.emit_inc_ref("x", self.instance_a)
if HAVE_IMMORTAL:
self.assert_output("CPy_INCREF_NO_IMM(x);\n")
else:
self.assert_output("CPy_INCREF(x);\n")
def test_emit_inc_ref_optional(self) -> None:
optional = RUnion([self.instance_a, none_rprimitive])
self.emitter.emit_inc_ref("o", optional)
self.assert_output("CPy_INCREF(o);\n")
def test_emit_dec_ref_object(self) -> None:
self.emitter.emit_dec_ref("x", object_rprimitive)
self.assert_output("CPy_DECREF(x);\n")
self.emitter.emit_dec_ref("x", object_rprimitive, is_xdec=True)
self.assert_output("CPy_XDECREF(x);\n")
def test_emit_dec_ref_int(self) -> None:
self.emitter.emit_dec_ref("x", int_rprimitive)
self.assert_output("CPyTagged_DECREF(x);\n")
self.emitter.emit_dec_ref("x", int_rprimitive, is_xdec=True)
self.assert_output("CPyTagged_XDECREF(x);\n")
def test_emit_dec_ref_rare(self) -> None:
self.emitter.emit_dec_ref("x", object_rprimitive, rare=True)
self.assert_output("CPy_DecRef(x);\n")
self.emitter.emit_dec_ref("x", int_rprimitive, rare=True)
self.assert_output("CPyTagged_DecRef(x);\n")
def test_emit_dec_ref_list(self) -> None:
self.emitter.emit_dec_ref("x", list_rprimitive)
if HAVE_IMMORTAL:
self.assert_output("CPy_DECREF_NO_IMM(x);\n")
else:
self.assert_output("CPy_DECREF(x);\n")
self.emitter.emit_dec_ref("x", list_rprimitive, is_xdec=True)
if HAVE_IMMORTAL:
self.assert_output("CPy_XDECREF_NO_IMM(x);\n")
else:
self.assert_output("CPy_XDECREF(x);\n")
def test_emit_dec_ref_instance(self) -> None:
self.emitter.emit_dec_ref("x", self.instance_a)
if HAVE_IMMORTAL:
self.assert_output("CPy_DECREF_NO_IMM(x);\n")
else:
self.assert_output("CPy_DECREF(x);\n")
self.emitter.emit_dec_ref("x", self.instance_a, is_xdec=True)
if HAVE_IMMORTAL:
self.assert_output("CPy_XDECREF_NO_IMM(x);\n")
else:
self.assert_output("CPy_XDECREF(x);\n")
def test_emit_dec_ref_optional(self) -> None:
optional = RUnion([self.instance_a, none_rprimitive])
self.emitter.emit_dec_ref("o", optional)
self.assert_output("CPy_DECREF(o);\n")
def assert_output(self, expected: str) -> None:
assert "".join(self.emitter.fragments) == expected
self.emitter.fragments = []