artiq/artiq/py2llvm/types.py

175 lines
4.5 KiB
Python
Raw Normal View History

2015-05-29 14:53:24 +08:00
"""
The :mod:`types` module contains the classes describing the types
in :mod:`asttyped`.
"""
import string
def genalnum():
ident = ["a"]
while True:
yield "".join(ident)
pos = len(ident) - 1
while pos >= 0:
cur_n = string.ascii_lowercase.index(ident[pos])
if cur_n < 26:
ident[pos] = string.ascii_lowercase[cur_n + 1]
break
else:
ident[pos] = "a"
pos -= 1
if pos < 0:
ident = "a" + ident
class UnificationError(Exception):
def __init__(self, typea, typeb):
self.typea, self.typeb = typea, typeb
class Type(object):
pass
class TVar(Type):
"""
A type variable.
In effect, the classic union-find data structure is intrusively
folded into this class.
"""
def __init__(self):
self.parent = self
def find(self):
if self.parent is self:
return self
else:
root = self.parent.find()
self.parent = root # path compression
return root
def unify(self, other):
other = other.find()
if self.parent is self:
self.parent = other
else:
self.find().unify(other)
def __repr__(self):
if self.parent is self:
return "TVar(%d)" % id(self)
else:
return repr(self.find())
# __eq__ and __hash__ are not overridden and default to
# comparison by identity. Use .find() explicitly before
# any lookups or comparisons.
class TMono(Type):
"""A monomorphic type, possibly parametric."""
2015-06-13 18:50:56 +08:00
attributes = {}
2015-05-29 14:53:24 +08:00
def __init__(self, name, params={}):
self.name, self.params = name, params
def find(self):
return self
def unify(self, other):
if isinstance(other, TMono) and self.name == other.name:
assert self.params.keys() == other.params.keys()
for param in self.params:
self.params[param].unify(other.params[param])
2015-06-04 19:50:16 +08:00
elif isinstance(other, TVar):
other.unify(self)
2015-05-29 14:53:24 +08:00
else:
raise UnificationError(self, other)
def __repr__(self):
return "TMono(%s, %s)" % (repr(self.name), repr(self.params))
2015-06-04 19:12:41 +08:00
def __getitem__(self, param):
return self.params[param]
2015-05-29 14:53:24 +08:00
def __eq__(self, other):
return isinstance(other, TMono) and \
self.name == other.name and \
self.params == other.params
def __ne__(self, other):
return not (self == other)
class TValue(Type):
"""
A type-level value (such as the integer denoting width of
a generic integer type.
"""
def __init__(self, value):
self.value = value
def find(self):
return self
def unify(self, other):
2015-06-04 19:50:16 +08:00
if isinstance(other, TVar):
other.unify(self)
elif self != other:
2015-05-29 14:53:24 +08:00
raise UnificationError(self, other)
def __repr__(self):
return "TValue(%s)" % repr(self.value)
def __eq__(self, other):
return isinstance(other, TValue) and \
self.value == other.value
def __ne__(self, other):
return not (self == other)
2015-06-12 13:59:41 +08:00
def is_var(typ):
return isinstance(typ, TVar)
def is_mono(typ, name, **params):
2015-06-13 14:28:40 +08:00
params_match = True
for param in params:
params_match = params_match and typ.params[param] == params[param]
2015-06-12 13:59:41 +08:00
return isinstance(typ, TMono) and \
2015-06-13 14:28:40 +08:00
typ.name == name and params_match
2015-06-12 13:59:41 +08:00
2015-05-29 14:53:24 +08:00
class TypePrinter(object):
"""
A class that prints types using Python-like syntax and gives
type variables sequential alphabetic names.
"""
def __init__(self):
self.gen = genalnum()
self.map = {}
def name(self, typ):
typ = typ.find()
if isinstance(typ, TVar):
if typ not in self.map:
self.map[typ] = "'%s" % next(self.gen)
return self.map[typ]
elif isinstance(typ, TMono):
2015-06-06 20:12:40 +08:00
if typ.params == {}:
return typ.name
else:
return "%s(%s)" % (typ.name, ", ".join(
["%s=%s" % (k, self.name(typ.params[k])) for k in typ.params]))
2015-05-29 14:53:24 +08:00
elif isinstance(typ, TTuple):
if len(typ.elts) == 1:
return "(%s,)" % self.name(typ.elts[0])
else:
return "(%s)" % ", ".join(list(map(self.name, typ.elts)))
elif isinstance(typ, TValue):
return repr(typ.value)
else:
assert False