diff options
author | Krzysztof Gogolewski <krzysztof.gogolewski@tweag.io> | 2021-08-06 19:35:24 +0200 |
---|---|---|
committer | Marge Bot <ben+marge-bot@smart-cactus.org> | 2021-09-17 09:35:53 -0400 |
commit | 4564f00fdeb5e072e8f91fec72a6393f0e3f0703 (patch) | |
tree | 9cbbc1791aac4c6cf07c08abf1c1c0f05f457e96 | |
parent | 885f17c89919d21815365da71eb7f9c489e5bfa3 (diff) | |
download | haskell-4564f00fdeb5e072e8f91fec72a6393f0e3f0703.tar.gz |
Improve pretty-printer defaulting logic (#19361)
When determining whether to default a RuntimeRep or Multiplicity
variable, use isMetaTyVar to distinguish between metavariables
(which can be hidden) and skolems (which cannot).
-rw-r--r-- | compiler/GHC/Iface/Type.hs | 79 | ||||
-rw-r--r-- | compiler/GHC/Tc/Utils/TcType.hs-boot | 4 | ||||
-rw-r--r-- | compiler/GHC/Types/Var.hs-boot | 1 | ||||
-rw-r--r-- | testsuite/tests/linear/should_fail/T19361.hs | 9 | ||||
-rw-r--r-- | testsuite/tests/linear/should_fail/T19361.stderr | 10 | ||||
-rw-r--r-- | testsuite/tests/linear/should_fail/all.T | 1 | ||||
-rw-r--r-- | testsuite/tests/typecheck/should_fail/T15883b.stderr | 9 | ||||
-rw-r--r-- | testsuite/tests/typecheck/should_fail/T15883c.stderr | 9 | ||||
-rw-r--r-- | testsuite/tests/typecheck/should_fail/T15883d.stderr | 10 | ||||
-rw-r--r-- | testsuite/tests/typecheck/should_fail/T15883e.stderr | 10 |
10 files changed, 92 insertions, 50 deletions
diff --git a/compiler/GHC/Iface/Type.hs b/compiler/GHC/Iface/Type.hs index 58410467d3..422091784a 100644 --- a/compiler/GHC/Iface/Type.hs +++ b/compiler/GHC/Iface/Type.hs @@ -87,6 +87,7 @@ import GHC.Utils.Outputable import GHC.Data.FastString import GHC.Utils.Misc import GHC.Utils.Panic +import {-# SOURCE #-} GHC.Tc.Utils.TcType ( isMetaTyVar, isTyConableTyVar ) import Data.Maybe( isJust ) import qualified Data.Semigroup as Semi @@ -1006,18 +1007,25 @@ This is done in a pass right before pretty-printing This applies to /quantified/ variables like 'w' above. What about variables that are /free/ in the type being printed, which certainly -happens in error messages. Suppose (#16074) we are reporting a -mismatch between two skolems +happens in error messages. Suppose (#16074, #19361) we are reporting a +mismatch between skolems (a :: RuntimeRep) ~ (b :: RuntimeRep) -We certainly don't want to say "Can't match LiftedRep ~ LiftedRep"! + or + (m :: Multiplicity) ~ Many +We certainly don't want to say "Can't match LiftedRep with LiftedRep" or +"Can't match Many with Many"! But if we are printing the type - (forall (a :: TYPE r). blah + (forall (a :: TYPE r). blah) we do want to turn that (free) r into LiftedRep, so it prints as (forall a. blah) -Conclusion: keep track of whether we are in the kind of a -binder; only if so, convert free RuntimeRep variables to LiftedRep. +We use isMetaTyVar to distinguish between those two situations: +metavariables are converted, skolem variables are not. + +There's one exception though: TyVarTv metavariables should not be defaulted, +as they appear during kind-checking of "newtype T :: TYPE r where..." +(test T18357a). Therefore, we additionally test for isTyConableTyVar. -} -- | Default 'RuntimeRep' variables to 'LiftedRep', and 'Multiplicity' @@ -1039,65 +1047,68 @@ binder; only if so, convert free RuntimeRep variables to LiftedRep. -- type signatures (e.g. ($)). See Note [Defaulting RuntimeRep variables] -- and #11549 for further discussion. defaultNonStandardVars :: Bool -> Bool -> IfaceType -> IfaceType -defaultNonStandardVars do_runtimereps do_multiplicities ty = go False emptyFsEnv ty +defaultNonStandardVars do_runtimereps do_multiplicities ty = go emptyFsEnv ty where - go :: Bool -- True <=> Inside the kind of a binder - -> FastStringEnv IfaceType -- Set of enclosing forall-ed RuntimeRep/Multiplicity variables + go :: FastStringEnv IfaceType -- Set of enclosing forall-ed RuntimeRep/Multiplicity variables -> IfaceType -> IfaceType - go ink subs (IfaceForAllTy (Bndr (IfaceTvBndr (var, var_kind)) argf) ty) + go subs (IfaceForAllTy (Bndr (IfaceTvBndr (var, var_kind)) argf) ty) | isInvisibleArgFlag argf -- Don't default *visible* quantification -- or we get the mess in #13963 , Just substituted_ty <- check_substitution var_kind = let subs' = extendFsEnv subs var substituted_ty -- Record that we should replace it with LiftedRep, -- and recurse, discarding the forall - in go ink subs' ty + in go subs' ty - go ink subs (IfaceForAllTy bndr ty) - = IfaceForAllTy (go_ifacebndr subs bndr) (go ink subs ty) + go subs (IfaceForAllTy bndr ty) + = IfaceForAllTy (go_ifacebndr subs bndr) (go subs ty) - go _ subs ty@(IfaceTyVar tv) = case lookupFsEnv subs tv of + go subs ty@(IfaceTyVar tv) = case lookupFsEnv subs tv of Just s -> s Nothing -> ty - go in_kind _ ty@(IfaceFreeTyVar tv) + go _ ty@(IfaceFreeTyVar tv) -- See Note [Defaulting RuntimeRep variables], about free vars - | in_kind && do_runtimereps && GHC.Core.Type.isRuntimeRepTy (tyVarKind tv) + | do_runtimereps && GHC.Core.Type.isRuntimeRepTy (tyVarKind tv) + , isMetaTyVar tv + , isTyConableTyVar tv = liftedRep_ty | do_multiplicities && GHC.Core.Type.isMultiplicityTy (tyVarKind tv) + , isMetaTyVar tv + , isTyConableTyVar tv = many_ty | otherwise = ty - go ink subs (IfaceTyConApp tc tc_args) - = IfaceTyConApp tc (go_args ink subs tc_args) + go subs (IfaceTyConApp tc tc_args) + = IfaceTyConApp tc (go_args subs tc_args) - go ink subs (IfaceTupleTy sort is_prom tc_args) - = IfaceTupleTy sort is_prom (go_args ink subs tc_args) + go subs (IfaceTupleTy sort is_prom tc_args) + = IfaceTupleTy sort is_prom (go_args subs tc_args) - go ink subs (IfaceFunTy af w arg res) - = IfaceFunTy af (go ink subs w) (go ink subs arg) (go ink subs res) + go subs (IfaceFunTy af w arg res) + = IfaceFunTy af (go subs w) (go subs arg) (go subs res) - go ink subs (IfaceAppTy t ts) - = IfaceAppTy (go ink subs t) (go_args ink subs ts) + go subs (IfaceAppTy t ts) + = IfaceAppTy (go subs t) (go_args subs ts) - go ink subs (IfaceCastTy x co) - = IfaceCastTy (go ink subs x) co + go subs (IfaceCastTy x co) + = IfaceCastTy (go subs x) co - go _ _ ty@(IfaceLitTy {}) = ty - go _ _ ty@(IfaceCoercionTy {}) = ty + go _ ty@(IfaceLitTy {}) = ty + go _ ty@(IfaceCoercionTy {}) = ty go_ifacebndr :: FastStringEnv IfaceType -> IfaceForAllBndr -> IfaceForAllBndr go_ifacebndr subs (Bndr (IfaceIdBndr (w, n, t)) argf) - = Bndr (IfaceIdBndr (w, n, go True subs t)) argf + = Bndr (IfaceIdBndr (w, n, go subs t)) argf go_ifacebndr subs (Bndr (IfaceTvBndr (n, t)) argf) - = Bndr (IfaceTvBndr (n, go True subs t)) argf + = Bndr (IfaceTvBndr (n, go subs t)) argf - go_args :: Bool -> FastStringEnv IfaceType -> IfaceAppArgs -> IfaceAppArgs - go_args _ _ IA_Nil = IA_Nil - go_args ink subs (IA_Arg ty argf args) - = IA_Arg (go ink subs ty) argf (go_args ink subs args) + go_args :: FastStringEnv IfaceType -> IfaceAppArgs -> IfaceAppArgs + go_args _ IA_Nil = IA_Nil + go_args subs (IA_Arg ty argf args) + = IA_Arg (go subs ty) argf (go_args subs args) check_substitution :: IfaceType -> Maybe IfaceType check_substitution (IfaceTyConApp tc _) diff --git a/compiler/GHC/Tc/Utils/TcType.hs-boot b/compiler/GHC/Tc/Utils/TcType.hs-boot index dc5f4cf73f..6b808dd7ab 100644 --- a/compiler/GHC/Tc/Utils/TcType.hs-boot +++ b/compiler/GHC/Tc/Utils/TcType.hs-boot @@ -1,8 +1,12 @@ module GHC.Tc.Utils.TcType where import GHC.Utils.Outputable( SDoc ) +import GHC.Prelude ( Bool ) +import {-# SOURCE #-} GHC.Types.Var ( TcTyVar ) data MetaDetails data TcTyVarDetails pprTcTyVarDetails :: TcTyVarDetails -> SDoc vanillaSkolemTv :: TcTyVarDetails +isMetaTyVar :: TcTyVar -> Bool +isTyConableTyVar :: TcTyVar -> Bool diff --git a/compiler/GHC/Types/Var.hs-boot b/compiler/GHC/Types/Var.hs-boot index f96157540a..1882a86d33 100644 --- a/compiler/GHC/Types/Var.hs-boot +++ b/compiler/GHC/Types/Var.hs-boot @@ -18,4 +18,5 @@ data Specificity type TyVar = Var type Id = Var type TyCoVar = Id +type TcTyVar = Var type InvisTVBinder = VarBndr TyVar Specificity diff --git a/testsuite/tests/linear/should_fail/T19361.hs b/testsuite/tests/linear/should_fail/T19361.hs new file mode 100644 index 0000000000..503b299a0b --- /dev/null +++ b/testsuite/tests/linear/should_fail/T19361.hs @@ -0,0 +1,9 @@ +{-# LANGUAGE LinearTypes #-} + +module T19361 where + +f :: a %m -> a +f x = g x + +g :: a -> a +g x = x diff --git a/testsuite/tests/linear/should_fail/T19361.stderr b/testsuite/tests/linear/should_fail/T19361.stderr new file mode 100644 index 0000000000..1d7746786a --- /dev/null +++ b/testsuite/tests/linear/should_fail/T19361.stderr @@ -0,0 +1,10 @@ + +T19361.hs:6:3: error: + • Couldn't match type ‘m’ with ‘'Many’ + arising from multiplicity of ‘x’ + ‘m’ is a rigid type variable bound by + the type signature for: + f :: forall a. a -> a + at T19361.hs:5:1-14 + • In an equation for ‘f’: f x = g x + • Relevant bindings include f :: a %m -> a (bound at T19361.hs:6:1) diff --git a/testsuite/tests/linear/should_fail/all.T b/testsuite/tests/linear/should_fail/all.T index 89363cba85..4d8eec398e 100644 --- a/testsuite/tests/linear/should_fail/all.T +++ b/testsuite/tests/linear/should_fail/all.T @@ -39,3 +39,4 @@ test('T18888', normal, compile_fail, ['']) test('T18888_datakinds', normal, compile_fail, ['']) test('T19120', normal, compile_fail, ['']) test('T20083', normal, compile_fail, ['-XLinearTypes']) +test('T19361', normal, compile_fail, ['']) diff --git a/testsuite/tests/typecheck/should_fail/T15883b.stderr b/testsuite/tests/typecheck/should_fail/T15883b.stderr index 21b9305315..b3efbc1b41 100644 --- a/testsuite/tests/typecheck/should_fail/T15883b.stderr +++ b/testsuite/tests/typecheck/should_fail/T15883b.stderr @@ -1,6 +1,7 @@ -T15883b.hs:14:1: - Can't make a derived instance of + +T15883b.hs:14:1: error: + • Can't make a derived instance of ‘Eq (Foo ('BoxedRep 'Lifted))’ with the stock strategy: - Don't know how to derive ‘Eq’ for type ‘forall a. a’ - In the stand-alone deriving instance for + Don't know how to derive ‘Eq’ for type ‘forall (a :: TYPE rep). a’ + • In the stand-alone deriving instance for ‘Eq (Foo (BoxedRep Lifted))’ diff --git a/testsuite/tests/typecheck/should_fail/T15883c.stderr b/testsuite/tests/typecheck/should_fail/T15883c.stderr index 60678c4fcb..2aa1049fa5 100644 --- a/testsuite/tests/typecheck/should_fail/T15883c.stderr +++ b/testsuite/tests/typecheck/should_fail/T15883c.stderr @@ -1,6 +1,7 @@ -T15883c.hs:14:1: - Can't make a derived instance of + +T15883c.hs:14:1: error: + • Can't make a derived instance of ‘Ord (Foo ('BoxedRep 'Lifted))’ with the stock strategy: - Don't know how to derive ‘Ord’ for type ‘forall a. a’ - In the stand-alone deriving instance for + Don't know how to derive ‘Ord’ for type ‘forall (a :: TYPE rep). a’ + • In the stand-alone deriving instance for ‘Ord (Foo (BoxedRep Lifted))’ diff --git a/testsuite/tests/typecheck/should_fail/T15883d.stderr b/testsuite/tests/typecheck/should_fail/T15883d.stderr index 162b31072e..96a294bc9e 100644 --- a/testsuite/tests/typecheck/should_fail/T15883d.stderr +++ b/testsuite/tests/typecheck/should_fail/T15883d.stderr @@ -1,6 +1,8 @@ -T15883d.hs:14:1: - Can't make a derived instance of + +T15883d.hs:14:1: error: + • Can't make a derived instance of ‘Show (Foo ('BoxedRep 'Lifted))’ with the stock strategy: - Don't know how to derive ‘Show’ for type ‘forall a. a’ - In the stand-alone deriving instance for + Don't know how to derive ‘Show’ + for type ‘forall (a :: TYPE rep). a’ + • In the stand-alone deriving instance for ‘Show (Foo (BoxedRep Lifted))’ diff --git a/testsuite/tests/typecheck/should_fail/T15883e.stderr b/testsuite/tests/typecheck/should_fail/T15883e.stderr index a20b3f5d43..c7006fb790 100644 --- a/testsuite/tests/typecheck/should_fail/T15883e.stderr +++ b/testsuite/tests/typecheck/should_fail/T15883e.stderr @@ -1,6 +1,8 @@ -T15883e.hs:16:1: - Can't make a derived instance of + +T15883e.hs:16:1: error: + • Can't make a derived instance of ‘Data (Foo ('BoxedRep 'Lifted))’ with the stock strategy: - Don't know how to derive ‘Data’ for type ‘forall a. a’ - In the stand-alone deriving instance for + Don't know how to derive ‘Data’ + for type ‘forall (a :: TYPE rep). a’ + • In the stand-alone deriving instance for ‘Data (Foo (BoxedRep Lifted))’ |