@@ -1222,7 +1222,7 @@ def mkIfExpr(site, cond, texpr, fexpr):
1222
1222
(texpr , fexpr , utype ) = usual_int_conv (
1223
1223
texpr , ttype , fexpr , ftype )
1224
1224
else :
1225
- if not compatible_types (ttype , ftype ):
1225
+ if not compatible_types_fuzzy (ttype , ftype ):
1226
1226
raise EBINOP (site , ':' , texpr , fexpr )
1227
1227
# TODO: in C, the rules are more complex,
1228
1228
# but our type system is too primitive to cover that
@@ -1396,7 +1396,7 @@ def make(cls, site, lh, rh):
1396
1396
if ((lhtype .is_arith and rhtype .is_arith )
1397
1397
or (isinstance (lhtype , (TPtr , TArray ))
1398
1398
and isinstance (rhtype , (TPtr , TArray ))
1399
- and compatible_types (lhtype .base , rhtype .base ))):
1399
+ and compatible_types_fuzzy (lhtype .base , rhtype .base ))):
1400
1400
return cls .make_simple (site , lh , rh )
1401
1401
raise EILLCOMP (site , lh , lhtype , rh , rhtype )
1402
1402
@@ -1601,7 +1601,7 @@ def make(cls, site, lh, rh):
1601
1601
if ((lhtype .is_arith and rhtype .is_arith )
1602
1602
or (isinstance (lhtype , (TPtr , TArray ))
1603
1603
and isinstance (rhtype , (TPtr , TArray ))
1604
- and compatible_types (lhtype , rhtype ))
1604
+ and compatible_types_fuzzy (lhtype , rhtype ))
1605
1605
or (isinstance (lhtype , TBool ) and isinstance (rhtype , TBool ))):
1606
1606
return Equals (site , lh , rh )
1607
1607
@@ -2932,8 +2932,8 @@ def mkInterfaceMethodRef(site, iface_node, indices, method_name):
2932
2932
if (not isinstance (ftype , TPtr )
2933
2933
or not isinstance (ftype .base , TFunction )
2934
2934
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 ):
2937
2937
# non-method members are not accessible
2938
2938
raise EMEMBER (site , struct_name , method_name )
2939
2939
@@ -4684,7 +4684,10 @@ class ArrayRef(LValue):
4684
4684
explicit_type = True
4685
4685
@auto_init
4686
4686
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 )
4688
4691
def __str__ (self ):
4689
4692
return '%s[%s]' % (self .expr , self .idx )
4690
4693
def read (self ):
@@ -4797,30 +4800,41 @@ def mkCast(site, expr, new_type):
4797
4800
raise ETEMPLATEUPCAST (site , "object" , new_type )
4798
4801
else :
4799
4802
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 ()
4800
4812
old_type = safe_realtype (expr .ctype ())
4801
4813
if (dml .globals .compat_dml12_int (site )
4802
4814
and (isinstance (old_type , (TStruct , TVector ))
4803
4815
or isinstance (real , (TStruct , TVector )))):
4804
4816
# these casts are permitted by C only if old and new are
4805
4817
# the same type, which is useless
4806
4818
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 )):
4814
4820
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 )):
4816
4830
raise ECAST (site , expr , new_type )
4817
4831
if isinstance (old_type , (TVoid , TStruct , TVector , TTraitList , TTrait )):
4818
4832
raise ECAST (site , expr , new_type )
4819
4833
if old_type .is_int and old_type .is_endian :
4820
4834
expr = as_int (expr )
4821
4835
old_type = safe_realtype (expr .ctype ())
4822
4836
if real .is_int and not real .is_endian :
4823
- if isinstance ( expr , IntegerConstant ) :
4837
+ if old_type . is_int and expr . constant :
4824
4838
value = truncate_int_bits (expr .value , real .signed , real .bits )
4825
4839
if dml .globals .compat_dml12_int (site ):
4826
4840
return IntegerConstant_dml12 (site , value , real )
@@ -4831,8 +4845,8 @@ def mkCast(site, expr, new_type):
4831
4845
# Shorten redundant chains of integer casts. Avoids insane C
4832
4846
# output for expressions like a+b+c+d.
4833
4847
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 ):
4836
4850
# (uint64)(int64)x -> (uint64)x
4837
4851
expr = expr .expr
4838
4852
old_type = safe_realtype (expr .ctype ())
@@ -4868,9 +4882,7 @@ def mkCast(site, expr, new_type):
4868
4882
return expr
4869
4883
elif real .is_int and real .is_endian :
4870
4884
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 ):
4874
4886
return mkApply (
4875
4887
expr .site ,
4876
4888
mkLit (expr .site , * real .get_store_fun ()),
@@ -4927,7 +4939,6 @@ def mkCast(site, expr, new_type):
4927
4939
class RValue (Expression ):
4928
4940
'''Wraps an lvalue to prohibit write. Useful when a composite
4929
4941
expression is reduced down to a single variable.'''
4930
- writable = False
4931
4942
@auto_init
4932
4943
def __init__ (self , site , expr ): pass
4933
4944
def __str__ (self ):
@@ -4936,11 +4947,22 @@ def ctype(self):
4936
4947
return self .expr .ctype ()
4937
4948
def read (self ):
4938
4949
return self .expr .read ()
4939
- def discard (self ): pass
4950
+ def discard (self ):
4951
+ return self .expr .discard ()
4940
4952
def incref (self ):
4941
4953
self .expr .incref ()
4942
4954
def decref (self ):
4943
4955
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
4944
4966
4945
4967
def mkRValue (expr ):
4946
4968
if isinstance (expr , LValue ) or expr .writable :
0 commit comments