@@ -522,16 +522,21 @@ module Gvn {
522
522
523
523
/** Provides definitions related to type unification. */
524
524
module Unification {
525
- /** A type parameter that is compatible with any type. */
525
+ /** A type parameter that is compatible with any type except `ref struct` . */
526
526
class UnconstrainedTypeParameter extends TypeParameter {
527
- UnconstrainedTypeParameter ( ) { not exists ( getATypeConstraint ( this ) ) }
527
+ UnconstrainedTypeParameter ( ) {
528
+ not exists ( getATypeConstraint ( this ) ) and not exists ( getANegativeTypeConstraint ( this ) )
529
+ }
528
530
}
529
531
530
532
/** A type parameter that is constrained. */
531
533
class ConstrainedTypeParameter extends TypeParameter {
532
534
int constraintCount ;
533
535
534
- ConstrainedTypeParameter ( ) { constraintCount = strictcount ( getATypeConstraint ( this ) ) }
536
+ ConstrainedTypeParameter ( ) {
537
+ constraintCount = count ( getATypeConstraint ( this ) ) + count ( getANegativeTypeConstraint ( this ) ) and
538
+ constraintCount > 0
539
+ }
535
540
536
541
/**
537
542
* Holds if this type parameter is unifiable with type `t`.
@@ -559,29 +564,31 @@ module Unification {
559
564
bindingset [ this ]
560
565
pragma [ inline_late]
561
566
override predicate unifiable ( Type t ) {
562
- exists ( TTypeParameterConstraint ttc | ttc = getATypeConstraint ( this ) |
567
+ forall ( TTypeParameterConstraint ttc | ttc = getATypeConstraint ( this ) |
563
568
ttc = TRefTypeConstraint ( ) and
564
569
t .isRefType ( )
565
570
or
566
571
ttc = TValueTypeConstraint ( ) and
567
572
t .isValueType ( )
568
573
or
569
574
typeConstraintUnifiable ( ttc , t )
570
- )
575
+ ) and
576
+ ( t .isRefLikeType ( ) implies getANegativeTypeConstraint ( this ) = TAllowRefTypeConstraint ( ) )
571
577
}
572
578
573
579
bindingset [ this ]
574
580
pragma [ inline_late]
575
581
override predicate subsumes ( Type t ) {
576
- exists ( TTypeParameterConstraint ttc | ttc = getATypeConstraint ( this ) |
582
+ forall ( TTypeParameterConstraint ttc | ttc = getATypeConstraint ( this ) |
577
583
ttc = TRefTypeConstraint ( ) and
578
584
t .isRefType ( )
579
585
or
580
586
ttc = TValueTypeConstraint ( ) and
581
587
t .isValueType ( )
582
588
or
583
589
typeConstraintSubsumes ( ttc , t )
584
- )
590
+ ) and
591
+ ( t .isRefLikeType ( ) implies getANegativeTypeConstraint ( this ) = TAllowRefTypeConstraint ( ) )
585
592
}
586
593
}
587
594
@@ -603,7 +610,8 @@ module Unification {
603
610
t .isValueType ( )
604
611
or
605
612
typeConstraintUnifiable ( ttc , t )
606
- )
613
+ ) and
614
+ ( t .isRefLikeType ( ) implies getANegativeTypeConstraint ( this ) = TAllowRefTypeConstraint ( ) )
607
615
}
608
616
609
617
bindingset [ this ]
@@ -617,7 +625,8 @@ module Unification {
617
625
t .isValueType ( )
618
626
or
619
627
typeConstraintSubsumes ( ttc , t )
620
- )
628
+ ) and
629
+ ( t .isRefLikeType ( ) implies getANegativeTypeConstraint ( this ) = TAllowRefTypeConstraint ( ) )
621
630
}
622
631
}
623
632
@@ -632,6 +641,9 @@ module Unification {
632
641
not t instanceof TypeParameter
633
642
}
634
643
644
+ cached
645
+ newtype TTypeParameterNegativeConstraint = TAllowRefTypeConstraint ( )
646
+
635
647
cached
636
648
TTypeParameterConstraint getATypeConstraint ( TypeParameter tp ) {
637
649
exists ( TypeParameterConstraints tpc | tpc = tp .getConstraints ( ) |
@@ -650,6 +662,14 @@ module Unification {
650
662
)
651
663
}
652
664
665
+ cached
666
+ TTypeParameterNegativeConstraint getANegativeTypeConstraint ( TypeParameter tp ) {
667
+ exists ( TypeParameterConstraints tpc | tpc = tp .getConstraints ( ) |
668
+ tpc .hasAllowRefLikeTypeConstraint ( ) and
669
+ result = TAllowRefTypeConstraint ( )
670
+ )
671
+ }
672
+
653
673
cached
654
674
predicate typeConstraintUnifiable ( TTypeConstraint ttc , Type t ) {
655
675
exists ( Type t0 | ttc = TTypeConstraint ( t0 ) | implicitConversionRestricted ( t , t0 ) )
0 commit comments