forked from M-Labs/artiq
language.core: remove {int,round}64, implement int with device semantics.
This commit is contained in:
parent
786fde827a
commit
fd3c8a2830
@ -10,7 +10,7 @@ from functools import wraps
|
|||||||
from artiq.coredevice.runtime import source_loader
|
from artiq.coredevice.runtime import source_loader
|
||||||
|
|
||||||
|
|
||||||
__all__ = ["int64", "round64",
|
__all__ = ["host_int", "int",
|
||||||
"kernel", "portable", "syscall",
|
"kernel", "portable", "syscall",
|
||||||
"set_time_manager", "set_watchdog_factory",
|
"set_time_manager", "set_watchdog_factory",
|
||||||
"ARTIQException"]
|
"ARTIQException"]
|
||||||
@ -24,64 +24,135 @@ kernel_globals = (
|
|||||||
)
|
)
|
||||||
__all__.extend(kernel_globals)
|
__all__.extend(kernel_globals)
|
||||||
|
|
||||||
|
host_int = int
|
||||||
|
|
||||||
class int64(int):
|
class int:
|
||||||
"""64-bit integers for static compilation.
|
"""
|
||||||
|
Arbitrary-precision integers for static compilation.
|
||||||
|
|
||||||
When this class is used instead of Python's ``int``, the static compiler
|
The static compiler does not use unlimited-precision integers,
|
||||||
stores the corresponding variable on 64 bits instead of 32.
|
like Python normally does, because of their unbounded memory requirements.
|
||||||
|
Instead, it allows to choose a bit width (usually 32 or 64) at compile-time,
|
||||||
|
and all computations follow wrap-around semantics on overflow.
|
||||||
|
|
||||||
When used in the interpreter, it behaves as ``int`` and the results of
|
This class implements the same semantics on the host.
|
||||||
integer operations involving it are also ``int64`` (which matches the
|
|
||||||
size promotion rules of the static compiler). This way, it is possible to
|
|
||||||
specify 64-bit size annotations on constants that are passed to the
|
|
||||||
kernels.
|
|
||||||
|
|
||||||
Example:
|
For example:
|
||||||
|
|
||||||
>>> a = int64(1)
|
>>> a = int(1, width=64)
|
||||||
>>> b = int64(3) + 2
|
>>> b = int(3, width=64) + 2
|
||||||
>>> isinstance(a, int64)
|
>>> isinstance(a, int)
|
||||||
True
|
True
|
||||||
>>> isinstance(b, int64)
|
>>> isinstance(b, int)
|
||||||
True
|
True
|
||||||
>>> a + b
|
>>> a + b
|
||||||
6
|
int(6, width=64)
|
||||||
|
>>> int(10, width=32) + 0x7fffffff
|
||||||
|
int(9, width=32)
|
||||||
|
>>> int(0x80000000)
|
||||||
|
int(-2147483648, width=32)
|
||||||
"""
|
"""
|
||||||
pass
|
|
||||||
|
|
||||||
def _make_int64_op_method(int_method):
|
__slots__ = ['_value', '_width']
|
||||||
def method(self, *args):
|
|
||||||
r = int_method(self, *args)
|
|
||||||
if isinstance(r, int):
|
|
||||||
r = int64(r)
|
|
||||||
return r
|
|
||||||
return method
|
|
||||||
|
|
||||||
for _op_name in ("neg", "pos", "abs", "invert", "round",
|
def __new__(cls, value, width=32):
|
||||||
"add", "radd", "sub", "rsub", "mul", "rmul", "pow", "rpow",
|
if isinstance(value, int):
|
||||||
"lshift", "rlshift", "rshift", "rrshift",
|
return value
|
||||||
"and", "rand", "xor", "rxor", "or", "ror",
|
else:
|
||||||
"floordiv", "rfloordiv", "mod", "rmod"):
|
sign_bit = 2 ** (width - 1)
|
||||||
_method_name = "__" + _op_name + "__"
|
value = host_int(value)
|
||||||
_orig_method = getattr(int, _method_name)
|
if value & sign_bit:
|
||||||
setattr(int64, _method_name, _make_int64_op_method(_orig_method))
|
value = -1 & ~sign_bit + (value & (sign_bit - 1)) + 1
|
||||||
|
else:
|
||||||
|
value &= sign_bit - 1
|
||||||
|
|
||||||
for _op_name in ("add", "sub", "mul", "floordiv", "mod",
|
self = super().__new__(cls)
|
||||||
"pow", "lshift", "rshift", "lshift",
|
self._value = value
|
||||||
"and", "xor", "or"):
|
self._width = width
|
||||||
_op_method = getattr(int, "__" + _op_name + "__")
|
return self
|
||||||
setattr(int64, "__i" + _op_name + "__", _make_int64_op_method(_op_method))
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def width(width):
|
||||||
|
return width._width
|
||||||
|
|
||||||
def round64(x):
|
def __int__(self):
|
||||||
"""Rounds to a 64-bit integer.
|
return self._value
|
||||||
|
|
||||||
This function is equivalent to ``int64(round(x))`` but, when targeting
|
def __float__(self):
|
||||||
static compilation, prevents overflow when the rounded value is too large
|
return float(self._value)
|
||||||
to fit in a 32-bit integer.
|
|
||||||
"""
|
def __str__(self):
|
||||||
return int64(round(x))
|
return str(self._value)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "int({}, width={})".format(self._value, self._width)
|
||||||
|
|
||||||
|
def _unaryop(lower_fn):
|
||||||
|
def operator(self):
|
||||||
|
return int(lower_fn(self._value), self._width)
|
||||||
|
return operator
|
||||||
|
|
||||||
|
__neg__ = _unaryop(host_int.__neg__)
|
||||||
|
__pos__ = _unaryop(host_int.__pos__)
|
||||||
|
__abs__ = _unaryop(host_int.__abs__)
|
||||||
|
__invert__ = _unaryop(host_int.__invert__)
|
||||||
|
__round__ = _unaryop(host_int.__round__)
|
||||||
|
|
||||||
|
def _binaryop(lower_fn, rlower_fn=None):
|
||||||
|
def operator(self, other):
|
||||||
|
if isinstance(other, host_int):
|
||||||
|
return int(lower_fn(self._value, other), self._width)
|
||||||
|
elif isinstance(other, int):
|
||||||
|
width = self._width if self._width > other._width else other._width
|
||||||
|
return int(lower_fn(self._value, other._value), width)
|
||||||
|
elif rlower_fn:
|
||||||
|
return getattr(other, rlower_fn)(self._value)
|
||||||
|
else:
|
||||||
|
return NotImplemented
|
||||||
|
return operator
|
||||||
|
|
||||||
|
__add__ = __iadd__ = _binaryop(host_int.__add__, "__radd__")
|
||||||
|
__sub__ = __isub__ = _binaryop(host_int.__sub__, "__rsub__")
|
||||||
|
__mul__ = __imul__ = _binaryop(host_int.__mul__, "__rmul__")
|
||||||
|
__floordiv__ = __ifloordiv__ = _binaryop(host_int.__floordiv__, "__rfloordiv__")
|
||||||
|
__mod__ = __imod__ = _binaryop(host_int.__mod__, "__rmod__")
|
||||||
|
__pow__ = __ipow__ = _binaryop(host_int.__pow__, "__rpow__")
|
||||||
|
|
||||||
|
__radd__ = _binaryop(host_int.__radd__, "__add__")
|
||||||
|
__rsub__ = _binaryop(host_int.__rsub__, "__sub__")
|
||||||
|
__rmul__ = _binaryop(host_int.__rmul__, "__mul__")
|
||||||
|
__rfloordiv__ = _binaryop(host_int.__rfloordiv__, "__floordiv__")
|
||||||
|
__rmod__ = _binaryop(host_int.__rmod__, "__mod__")
|
||||||
|
__rpow__ = _binaryop(host_int.__rpow__, "__pow__")
|
||||||
|
|
||||||
|
__lshift__ = __ilshift__ = _binaryop(host_int.__lshift__)
|
||||||
|
__rshift__ = __irshift__ = _binaryop(host_int.__rshift__)
|
||||||
|
__and__ = __iand__ = _binaryop(host_int.__and__)
|
||||||
|
__or__ = __ior__ = _binaryop(host_int.__or__)
|
||||||
|
__xor__ = __ixor__ = _binaryop(host_int.__xor__)
|
||||||
|
|
||||||
|
__rlshift__ = _binaryop(host_int.__rlshift__)
|
||||||
|
__rrshift__ = _binaryop(host_int.__rrshift__)
|
||||||
|
__rand__ = _binaryop(host_int.__rand__)
|
||||||
|
__ror__ = _binaryop(host_int.__ror__)
|
||||||
|
__rxor__ = _binaryop(host_int.__rxor__)
|
||||||
|
|
||||||
|
def _compareop(lower_fn, rlower_fn):
|
||||||
|
def operator(self, other):
|
||||||
|
if isinstance(other, host_int):
|
||||||
|
return lower_fn(self._value, other)
|
||||||
|
elif isinstance(other, int):
|
||||||
|
return lower_fn(self._value, other._value)
|
||||||
|
else:
|
||||||
|
return getattr(other, rlower_fn)(self._value)
|
||||||
|
return operator
|
||||||
|
|
||||||
|
__eq__ = _compareop(host_int.__eq__, "__ne__")
|
||||||
|
__ne__ = _compareop(host_int.__ne__, "__eq__")
|
||||||
|
__gt__ = _compareop(host_int.__gt__, "__le__")
|
||||||
|
__ge__ = _compareop(host_int.__ge__, "__lt__")
|
||||||
|
__lt__ = _compareop(host_int.__lt__, "__ge__")
|
||||||
|
__le__ = _compareop(host_int.__le__, "__gt__")
|
||||||
|
|
||||||
|
|
||||||
_ARTIQEmbeddedInfo = namedtuple("_ARTIQEmbeddedInfo",
|
_ARTIQEmbeddedInfo = namedtuple("_ARTIQEmbeddedInfo",
|
||||||
|
44
artiq/test/language.py
Normal file
44
artiq/test/language.py
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import unittest
|
||||||
|
|
||||||
|
from artiq.language.core import *
|
||||||
|
|
||||||
|
|
||||||
|
class LanguageCoreTest(unittest.TestCase):
|
||||||
|
def test_unary(self):
|
||||||
|
self.assertEqual(int(10), +int(10))
|
||||||
|
self.assertEqual(int(-10), -int(10))
|
||||||
|
self.assertEqual(int(~10), ~int(10))
|
||||||
|
self.assertEqual(int(10), round(int(10)))
|
||||||
|
|
||||||
|
def test_arith(self):
|
||||||
|
self.assertEqual(int(9), int(4) + int(5))
|
||||||
|
self.assertEqual(int(9), int(4) + 5)
|
||||||
|
self.assertEqual(int(9), 5 + int(4))
|
||||||
|
|
||||||
|
self.assertEqual(9.0, int(4) + 5.0)
|
||||||
|
self.assertEqual(9.0, 5.0 + int(4))
|
||||||
|
|
||||||
|
a = int(5)
|
||||||
|
a += int(2)
|
||||||
|
a += 2
|
||||||
|
self.assertEqual(int(9), a)
|
||||||
|
|
||||||
|
def test_compare(self):
|
||||||
|
self.assertTrue(int(9) > int(8))
|
||||||
|
self.assertTrue(int(9) > 8)
|
||||||
|
self.assertTrue(int(9) > 8.0)
|
||||||
|
self.assertTrue(9 > int(8))
|
||||||
|
self.assertTrue(9.0 > int(8))
|
||||||
|
|
||||||
|
def test_bitwise(self):
|
||||||
|
self.assertEqual(int(0x100), int(0x10) << int(4))
|
||||||
|
self.assertEqual(int(0x100), int(0x10) << 4)
|
||||||
|
self.assertEqual(int(0x100), 0x10 << int(4))
|
||||||
|
|
||||||
|
def test_wraparound(self):
|
||||||
|
self.assertEqual(int(0xffffffff), int(-1))
|
||||||
|
self.assertTrue(int(0x7fffffff) > int(1))
|
||||||
|
self.assertTrue(int(0x80000000) < int(-1))
|
||||||
|
|
||||||
|
self.assertEqual(int(9), int(10) + int(0xffffffff))
|
||||||
|
self.assertEqual(-1.0, float(int(0xfffffffe) + int(1)))
|
Loading…
Reference in New Issue
Block a user