Skip to content

Commit 9ff55df

Browse files
committed
Type Equality/Compatibility Revamp -- SIMICS-9504
Taken from #213, but with some refinements
1 parent b933120 commit 9ff55df

File tree

8 files changed

+303
-124
lines changed

8 files changed

+303
-124
lines changed

py/dml/c_backend.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -805,13 +805,13 @@ def generate_implement_method(device, ifacestruct, meth, indices):
805805
# method in DML
806806
raise EMETH(meth.site, None, 'interface method is variadic')
807807
for ((mp, mt), it) in zip(meth.inp, iface_input_types):
808-
if safe_realtype(mt).cmp(safe_realtype(it)) != 0:
808+
if safe_realtype_unconst(mt).cmp(safe_realtype_unconst(it)) != 0:
809809
raise EARGT(meth.site, 'implement', meth.name,
810810
mt, mp, it, 'method')
811811
if iface_num_outputs and dml.globals.dml_version != (1, 2):
812812
[(_, mt)] = meth.outp
813-
if safe_realtype(mt).cmp(
814-
safe_realtype(func_type.output_type)) != 0:
813+
if safe_realtype_unconst(mt).cmp(
814+
safe_realtype_unconst(func_type.output_type)) != 0:
815815
raise EARGT(meth.site, 'implement', meth.name,
816816
mt, '<return value>', func_type.output_type,
817817
'method')

py/dml/codegen.py

+3-15
Original file line numberDiff line numberDiff line change
@@ -1306,18 +1306,7 @@ def expr_cast(tree, location, scope):
13061306
for (site, _) in struct_defs:
13071307
report(EANONSTRUCT(site, "'cast' expression"))
13081308

1309-
if (compat.dml12_misc in dml.globals.enabled_compat
1310-
and isinstance(expr, InterfaceMethodRef)):
1311-
# Workaround for SIMICS-9868
1312-
return mkLit(tree.site, "%s->%s" % (
1313-
expr.node_expr.read(), expr.method_name), type)
1314-
1315-
if isinstance(expr, NonValue) and (
1316-
not isinstance(expr, NodeRef)
1317-
or not isinstance(safe_realtype(type), TTrait)):
1318-
raise expr.exc()
1319-
else:
1320-
return mkCast(tree.site, expr, type)
1309+
return mkCast(tree.site, expr, type)
13211310

13221311
@expression_dispatcher
13231312
def expr_undefined(tree, location, scope):
@@ -2008,9 +1997,8 @@ def mk_sym(name, typ, mkunique=not dml.globals.debuggable):
20081997
for (name, typ) in decls:
20091998
sym = mk_sym(name, typ)
20101999
tgt_typ = safe_realtype_shallow(typ)
2011-
if tgt_typ.const:
2012-
nonconst_typ = tgt_typ.clone()
2013-
nonconst_typ.const = False
2000+
if shallow_const(tgt_typ):
2001+
nonconst_typ = safe_realtype_unconst(tgt_typ)
20142002
tgt_sym = mk_sym('_tmp_' + name, nonconst_typ, True)
20152003
sym.init = ExpressionInitializer(mkLocalVariable(stmt.site,
20162004
tgt_sym))

py/dml/ctree.py

+44-22
Original file line numberDiff line numberDiff line change
@@ -1222,7 +1222,7 @@ def mkIfExpr(site, cond, texpr, fexpr):
12221222
(texpr, fexpr, utype) = usual_int_conv(
12231223
texpr, ttype, fexpr, ftype)
12241224
else:
1225-
if not compatible_types(ttype, ftype):
1225+
if not compatible_types_fuzzy(ttype, ftype):
12261226
raise EBINOP(site, ':', texpr, fexpr)
12271227
# TODO: in C, the rules are more complex,
12281228
# but our type system is too primitive to cover that
@@ -1396,7 +1396,7 @@ def make(cls, site, lh, rh):
13961396
if ((lhtype.is_arith and rhtype.is_arith)
13971397
or (isinstance(lhtype, (TPtr, TArray))
13981398
and isinstance(rhtype, (TPtr, TArray))
1399-
and compatible_types(lhtype.base, rhtype.base))):
1399+
and compatible_types_fuzzy(lhtype.base, rhtype.base))):
14001400
return cls.make_simple(site, lh, rh)
14011401
raise EILLCOMP(site, lh, lhtype, rh, rhtype)
14021402

@@ -1601,7 +1601,7 @@ def make(cls, site, lh, rh):
16011601
if ((lhtype.is_arith and rhtype.is_arith)
16021602
or (isinstance(lhtype, (TPtr, TArray))
16031603
and isinstance(rhtype, (TPtr, TArray))
1604-
and compatible_types(lhtype, rhtype))
1604+
and compatible_types_fuzzy(lhtype, rhtype))
16051605
or (isinstance(lhtype, TBool) and isinstance(rhtype, TBool))):
16061606
return Equals(site, lh, rh)
16071607

@@ -2932,8 +2932,8 @@ def mkInterfaceMethodRef(site, iface_node, indices, method_name):
29322932
if (not isinstance(ftype, TPtr)
29332933
or not isinstance(ftype.base, TFunction)
29342934
or not ftype.base.input_types
2935-
or TPtr(safe_realtype(TNamed('conf_object_t'))).cmp(
2936-
safe_realtype(ftype.base.input_types[0])) != 0):
2935+
or TPtr(safe_realtype_unconst(TNamed('conf_object_t'))).cmp(
2936+
safe_realtype_unconst(ftype.base.input_types[0])) != 0):
29372937
# non-method members are not accessible
29382938
raise EMEMBER(site, struct_name, method_name)
29392939

@@ -4684,7 +4684,10 @@ class ArrayRef(LValue):
46844684
explicit_type = True
46854685
@auto_init
46864686
def __init__(self, site, expr, idx):
4687-
self.type = realtype_shallow(expr.ctype()).base
4687+
expr_type = realtype_shallow(expr.ctype())
4688+
self.type = conv_const(expr_type.const
4689+
and isinstance(expr_type, TArray),
4690+
expr_type.base)
46884691
def __str__(self):
46894692
return '%s[%s]' % (self.expr, self.idx)
46904693
def read(self):
@@ -4797,30 +4800,41 @@ def mkCast(site, expr, new_type):
47974800
raise ETEMPLATEUPCAST(site, "object", new_type)
47984801
else:
47994802
return mkTraitUpcast(site, expr, real.trait)
4803+
4804+
if (compat.dml12_misc in dml.globals.enabled_compat
4805+
and isinstance(expr, InterfaceMethodRef)):
4806+
# Workaround for SIMICS-9868
4807+
return mkLit(site, "%s->%s" % (
4808+
expr.node_expr.read(), expr.method_name), new_type)
4809+
4810+
if isinstance(expr, NonValue):
4811+
raise expr.exc()
48004812
old_type = safe_realtype(expr.ctype())
48014813
if (dml.globals.compat_dml12_int(site)
48024814
and (isinstance(old_type, (TStruct, TVector))
48034815
or isinstance(real, (TStruct, TVector)))):
48044816
# these casts are permitted by C only if old and new are
48054817
# the same type, which is useless
48064818
return Cast(site, expr, new_type)
4807-
if isinstance(real, TStruct):
4808-
if isinstance(old_type, TStruct) and old_type.label == real.label:
4809-
return expr
4810-
raise ECAST(site, expr, new_type)
4811-
if isinstance(real, TExternStruct):
4812-
if isinstance(old_type, TExternStruct) and old_type.id == real.id:
4813-
return expr
4819+
if isinstance(real, (TVoid, TArray, TFunction)):
48144820
raise ECAST(site, expr, new_type)
4815-
if isinstance(real, (TVoid, TArray, TVector, TTraitList, TFunction)):
4821+
if old_type.cmp(real) == 0:
4822+
if (old_type.is_int
4823+
and not old_type.is_endian
4824+
and dml.globals.compat_dml12_int(expr.site)):
4825+
# 1.2 integer expressions often lie about their actual type,
4826+
# and require a "redundant" cast! Why yes, this IS horrid!
4827+
return Cast(site, expr, new_type)
4828+
return mkRValue(expr)
4829+
if isinstance(real, (TStruct, TExternStruct, TVector, TTraitList)):
48164830
raise ECAST(site, expr, new_type)
48174831
if isinstance(old_type, (TVoid, TStruct, TVector, TTraitList, TTrait)):
48184832
raise ECAST(site, expr, new_type)
48194833
if old_type.is_int and old_type.is_endian:
48204834
expr = as_int(expr)
48214835
old_type = safe_realtype(expr.ctype())
48224836
if real.is_int and not real.is_endian:
4823-
if isinstance(expr, IntegerConstant):
4837+
if old_type.is_int and expr.constant:
48244838
value = truncate_int_bits(expr.value, real.signed, real.bits)
48254839
if dml.globals.compat_dml12_int(site):
48264840
return IntegerConstant_dml12(site, value, real)
@@ -4831,8 +4845,8 @@ def mkCast(site, expr, new_type):
48314845
# Shorten redundant chains of integer casts. Avoids insane C
48324846
# output for expressions like a+b+c+d.
48334847
if (isinstance(expr, Cast)
4834-
and isinstance(expr.type, TInt)
4835-
and expr.type.bits >= real.bits):
4848+
and isinstance(old_type, TInt)
4849+
and old_type.bits >= real.bits):
48364850
# (uint64)(int64)x -> (uint64)x
48374851
expr = expr.expr
48384852
old_type = safe_realtype(expr.ctype())
@@ -4868,9 +4882,7 @@ def mkCast(site, expr, new_type):
48684882
return expr
48694883
elif real.is_int and real.is_endian:
48704884
old_type = safe_realtype(expr.ctype())
4871-
if real.cmp(old_type) == 0:
4872-
return expr
4873-
elif old_type.is_arith or isinstance(old_type, TPtr):
4885+
if old_type.is_arith or isinstance(old_type, TPtr):
48744886
return mkApply(
48754887
expr.site,
48764888
mkLit(expr.site, *real.get_store_fun()),
@@ -4927,7 +4939,6 @@ def mkCast(site, expr, new_type):
49274939
class RValue(Expression):
49284940
'''Wraps an lvalue to prohibit write. Useful when a composite
49294941
expression is reduced down to a single variable.'''
4930-
writable = False
49314942
@auto_init
49324943
def __init__(self, site, expr): pass
49334944
def __str__(self):
@@ -4936,11 +4947,22 @@ def ctype(self):
49364947
return self.expr.ctype()
49374948
def read(self):
49384949
return self.expr.read()
4939-
def discard(self): pass
4950+
def discard(self):
4951+
return self.expr.discard()
49404952
def incref(self):
49414953
self.expr.incref()
49424954
def decref(self):
49434955
self.expr.decref()
4956+
@property
4957+
def explicit_type(self):
4958+
return self.expr.explicit_type
4959+
@property
4960+
def type(self):
4961+
assert self.explicit_type
4962+
return self.expr.type
4963+
@property
4964+
def is_pointer_to_stack_allocation(self):
4965+
return self.expr.is_pointer_to_stack_allocation
49444966

49454967
def mkRValue(expr):
49464968
if isinstance(expr, LValue) or expr.writable:

py/dml/ctree_test.py

+2
Original file line numberDiff line numberDiff line change
@@ -1416,6 +1416,8 @@ def const_types(self):
14161416
# abstract type
14171417
types.IntegerType,
14181418
# abstract type
1419+
types.ArchDependentIntegerType,
1420+
# abstract type
14191421
types.StructType,
14201422
# 1.2, weird
14211423
types.TUnknown,

py/dml/structure.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -737,7 +737,8 @@ def typecheck_method_override(m1, m2, location):
737737
# TODO move to caller
738738
(_, type1) = eval_type(t1, a1.site, location, global_scope)
739739
(_, type2) = eval_type(t2, a2.site, location, global_scope)
740-
if safe_realtype(type1).cmp(safe_realtype(type2)) != 0:
740+
if safe_realtype_unconst(type1).cmp(
741+
safe_realtype_unconst(type2)) != 0:
741742
raise EMETH(a1.site, a2.site,
742743
f"mismatching types in input argument {n1}")
743744

@@ -746,7 +747,8 @@ def typecheck_method_override(m1, m2, location):
746747
((n1, t1), (n2, t2)) = (a1.args, a2.args)
747748
(_, type1) = eval_type(t1, a1.site, location, global_scope)
748749
(_, type2) = eval_type(t2, a2.site, location, global_scope)
749-
if safe_realtype(type1).cmp(safe_realtype(type2)) != 0:
750+
if safe_realtype_unconst(type1).cmp(
751+
safe_realtype_unconst(type2)) != 0:
750752
msg = "mismatching types in return value"
751753
if len(outp1) > 1:
752754
msg += f" {i + 1}"

py/dml/traits.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -401,11 +401,11 @@ def typecheck_method_override(left, right):
401401
if throws0 != throws1:
402402
raise EMETH(site0, site1, "different nothrow annotations")
403403
for ((n, t0), (_, t1)) in zip(inp0, inp1):
404-
if realtype(t0).cmp(realtype(t1)) != 0:
404+
if safe_realtype_unconst(t0).cmp(safe_realtype_unconst(t1)) != 0:
405405
raise EMETH(site0, site1,
406406
"mismatching types in input argument %s" % (n,))
407407
for (i, ((_, t0), (_, t1))) in enumerate(zip(outp0, outp1)):
408-
if realtype(t0).cmp(realtype(t1)) != 0:
408+
if safe_realtype_unconst(t0).cmp(safe_realtype_unconst(t1)) != 0:
409409
raise EMETH(site0, site1,
410410
"mismatching types in output argument %d" % (i + 1,))
411411

0 commit comments

Comments
 (0)