{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE Trustworthy #-}
{-# LANGUAGE TupleSections #-}

-- | Facilities for type-checking Futhark terms.  Checking a term
-- requires a little more context to track uniqueness and such.
--
-- Type inference is implemented through a variation of
-- Hindley-Milner.  The main complication is supporting the rich
-- number of built-in language constructs, as well as uniqueness
-- types.  This is mostly done in an ad hoc way, and many programs
-- will require the programmer to fall back on type annotations.
module Language.Futhark.TypeChecker.Terms
  ( checkOneExp,
    checkFunDef,
  )
where

import Control.Monad.Except
import Control.Monad.Reader
import Control.Monad.State
import Data.Bifunctor
import Data.Bitraversable
import Data.Char (isAscii)
import Data.Either
import Data.List (find, foldl', isPrefixOf, sort)
import qualified Data.List.NonEmpty as NE
import qualified Data.Map.Strict as M
import Data.Maybe
import qualified Data.Set as S
import Futhark.IR.Primitive (intByteSize)
import Futhark.Util (nubOrd)
import Futhark.Util.Pretty hiding (bool, group, space)
import Language.Futhark
import Language.Futhark.Semantic (includeToFilePath)
import Language.Futhark.Traversals
import Language.Futhark.TypeChecker.Match
import Language.Futhark.TypeChecker.Monad hiding (BoundV)
import qualified Language.Futhark.TypeChecker.Monad as TypeM
import Language.Futhark.TypeChecker.Types hiding (checkTypeDecl)
import qualified Language.Futhark.TypeChecker.Types as Types
import Language.Futhark.TypeChecker.Unify hiding (Usage)
import Prelude hiding (mod)

--- Uniqueness

data Usage
  = Consumed SrcLoc
  | Observed SrcLoc
  deriving (Usage -> Usage -> Bool
(Usage -> Usage -> Bool) -> (Usage -> Usage -> Bool) -> Eq Usage
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: Usage -> Usage -> Bool
$c/= :: Usage -> Usage -> Bool
== :: Usage -> Usage -> Bool
$c== :: Usage -> Usage -> Bool
Eq, Eq Usage
Eq Usage
-> (Usage -> Usage -> Ordering)
-> (Usage -> Usage -> Bool)
-> (Usage -> Usage -> Bool)
-> (Usage -> Usage -> Bool)
-> (Usage -> Usage -> Bool)
-> (Usage -> Usage -> Usage)
-> (Usage -> Usage -> Usage)
-> Ord Usage
Usage -> Usage -> Bool
Usage -> Usage -> Ordering
Usage -> Usage -> Usage
forall a.
Eq a
-> (a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
min :: Usage -> Usage -> Usage
$cmin :: Usage -> Usage -> Usage
max :: Usage -> Usage -> Usage
$cmax :: Usage -> Usage -> Usage
>= :: Usage -> Usage -> Bool
$c>= :: Usage -> Usage -> Bool
> :: Usage -> Usage -> Bool
$c> :: Usage -> Usage -> Bool
<= :: Usage -> Usage -> Bool
$c<= :: Usage -> Usage -> Bool
< :: Usage -> Usage -> Bool
$c< :: Usage -> Usage -> Bool
compare :: Usage -> Usage -> Ordering
$ccompare :: Usage -> Usage -> Ordering
$cp1Ord :: Eq Usage
Ord, Int -> Usage -> ShowS
[Usage] -> ShowS
Usage -> String
(Int -> Usage -> ShowS)
-> (Usage -> String) -> ([Usage] -> ShowS) -> Show Usage
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Usage] -> ShowS
$cshowList :: [Usage] -> ShowS
show :: Usage -> String
$cshow :: Usage -> String
showsPrec :: Int -> Usage -> ShowS
$cshowsPrec :: Int -> Usage -> ShowS
Show)

type Names = S.Set VName

-- | The consumption set is a Maybe so we can distinguish whether a
-- consumption took place, but the variable went out of scope since,
-- or no consumption at all took place.
data Occurence = Occurence
  { Occurence -> Names
observed :: Names,
    Occurence -> Maybe Names
consumed :: Maybe Names,
    Occurence -> SrcLoc
location :: SrcLoc
  }
  deriving (Occurence -> Occurence -> Bool
(Occurence -> Occurence -> Bool)
-> (Occurence -> Occurence -> Bool) -> Eq Occurence
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: Occurence -> Occurence -> Bool
$c/= :: Occurence -> Occurence -> Bool
== :: Occurence -> Occurence -> Bool
$c== :: Occurence -> Occurence -> Bool
Eq, Int -> Occurence -> ShowS
[Occurence] -> ShowS
Occurence -> String
(Int -> Occurence -> ShowS)
-> (Occurence -> String)
-> ([Occurence] -> ShowS)
-> Show Occurence
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Occurence] -> ShowS
$cshowList :: [Occurence] -> ShowS
show :: Occurence -> String
$cshow :: Occurence -> String
showsPrec :: Int -> Occurence -> ShowS
$cshowsPrec :: Int -> Occurence -> ShowS
Show)

instance Located Occurence where
  locOf :: Occurence -> Loc
locOf = SrcLoc -> Loc
forall a. Located a => a -> Loc
locOf (SrcLoc -> Loc) -> (Occurence -> SrcLoc) -> Occurence -> Loc
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Occurence -> SrcLoc
location

observation :: Aliasing -> SrcLoc -> Occurence
observation :: Aliasing -> SrcLoc -> Occurence
observation = (Names -> Maybe Names -> SrcLoc -> Occurence)
-> Maybe Names -> Names -> SrcLoc -> Occurence
forall a b c. (a -> b -> c) -> b -> a -> c
flip Names -> Maybe Names -> SrcLoc -> Occurence
Occurence Maybe Names
forall a. Maybe a
Nothing (Names -> SrcLoc -> Occurence)
-> (Aliasing -> Names) -> Aliasing -> SrcLoc -> Occurence
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Alias -> VName) -> Aliasing -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> VName
aliasVar

consumption :: Aliasing -> SrcLoc -> Occurence
consumption :: Aliasing -> SrcLoc -> Occurence
consumption = Names -> Maybe Names -> SrcLoc -> Occurence
Occurence Names
forall a. Set a
S.empty (Maybe Names -> SrcLoc -> Occurence)
-> (Aliasing -> Maybe Names) -> Aliasing -> SrcLoc -> Occurence
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Names -> Maybe Names
forall a. a -> Maybe a
Just (Names -> Maybe Names)
-> (Aliasing -> Names) -> Aliasing -> Maybe Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Alias -> VName) -> Aliasing -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> VName
aliasVar

-- | A null occurence is one that we can remove without affecting
-- anything.
nullOccurence :: Occurence -> Bool
nullOccurence :: Occurence -> Bool
nullOccurence Occurence
occ = Names -> Bool
forall a. Set a -> Bool
S.null (Occurence -> Names
observed Occurence
occ) Bool -> Bool -> Bool
&& Maybe Names -> Bool
forall a. Maybe a -> Bool
isNothing (Occurence -> Maybe Names
consumed Occurence
occ)

-- | A seminull occurence is one that does not contain references to
-- any variables in scope.  The big difference is that a seminull
-- occurence may denote a consumption, as long as the array that was
-- consumed is now out of scope.
seminullOccurence :: Occurence -> Bool
seminullOccurence :: Occurence -> Bool
seminullOccurence Occurence
occ = Names -> Bool
forall a. Set a -> Bool
S.null (Occurence -> Names
observed Occurence
occ) Bool -> Bool -> Bool
&& Bool -> (Names -> Bool) -> Maybe Names -> Bool
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Bool
True Names -> Bool
forall a. Set a -> Bool
S.null (Occurence -> Maybe Names
consumed Occurence
occ)

type Occurences = [Occurence]

type UsageMap = M.Map VName [Usage]

usageMap :: Occurences -> UsageMap
usageMap :: [Occurence] -> UsageMap
usageMap = (UsageMap -> Occurence -> UsageMap)
-> UsageMap -> [Occurence] -> UsageMap
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl UsageMap -> Occurence -> UsageMap
comb UsageMap
forall k a. Map k a
M.empty
  where
    comb :: UsageMap -> Occurence -> UsageMap
comb UsageMap
m (Occurence Names
obs Maybe Names
cons SrcLoc
loc) =
      let m' :: UsageMap
m' = (UsageMap -> VName -> UsageMap) -> UsageMap -> Names -> UsageMap
forall a b. (a -> b -> a) -> a -> Set b -> a
S.foldl' (Usage -> UsageMap -> VName -> UsageMap
forall k a. Ord k => a -> Map k [a] -> k -> Map k [a]
ins (Usage -> UsageMap -> VName -> UsageMap)
-> Usage -> UsageMap -> VName -> UsageMap
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
Observed SrcLoc
loc) UsageMap
m Names
obs
       in (UsageMap -> VName -> UsageMap) -> UsageMap -> Names -> UsageMap
forall a b. (a -> b -> a) -> a -> Set b -> a
S.foldl' (Usage -> UsageMap -> VName -> UsageMap
forall k a. Ord k => a -> Map k [a] -> k -> Map k [a]
ins (Usage -> UsageMap -> VName -> UsageMap)
-> Usage -> UsageMap -> VName -> UsageMap
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
Consumed SrcLoc
loc) UsageMap
m' (Names -> UsageMap) -> Names -> UsageMap
forall a b. (a -> b) -> a -> b
$ Names -> Maybe Names -> Names
forall a. a -> Maybe a -> a
fromMaybe Names
forall a. Monoid a => a
mempty Maybe Names
cons
    ins :: a -> Map k [a] -> k -> Map k [a]
ins a
v Map k [a]
m k
k = ([a] -> [a] -> [a]) -> k -> [a] -> Map k [a] -> Map k [a]
forall k a. Ord k => (a -> a -> a) -> k -> a -> Map k a -> Map k a
M.insertWith [a] -> [a] -> [a]
forall a. [a] -> [a] -> [a]
(++) k
k [a
v] Map k [a]
m

combineOccurences :: VName -> Usage -> Usage -> TermTypeM Usage
combineOccurences :: VName -> Usage -> Usage -> TermTypeM Usage
combineOccurences VName
_ (Observed SrcLoc
loc) (Observed SrcLoc
_) = Usage -> TermTypeM Usage
forall (m :: * -> *) a. Monad m => a -> m a
return (Usage -> TermTypeM Usage) -> Usage -> TermTypeM Usage
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
Observed SrcLoc
loc
combineOccurences VName
name (Consumed SrcLoc
wloc) (Observed SrcLoc
rloc) =
  VName -> SrcLoc -> SrcLoc -> TermTypeM Usage
forall a. VName -> SrcLoc -> SrcLoc -> TermTypeM a
useAfterConsume VName
name SrcLoc
rloc SrcLoc
wloc
combineOccurences VName
name (Observed SrcLoc
rloc) (Consumed SrcLoc
wloc) =
  VName -> SrcLoc -> SrcLoc -> TermTypeM Usage
forall a. VName -> SrcLoc -> SrcLoc -> TermTypeM a
useAfterConsume VName
name SrcLoc
rloc SrcLoc
wloc
combineOccurences VName
name (Consumed SrcLoc
loc1) (Consumed SrcLoc
loc2) =
  VName -> SrcLoc -> SrcLoc -> TermTypeM Usage
forall a. VName -> SrcLoc -> SrcLoc -> TermTypeM a
useAfterConsume VName
name (SrcLoc -> SrcLoc -> SrcLoc
forall a. Ord a => a -> a -> a
max SrcLoc
loc1 SrcLoc
loc2) (SrcLoc -> SrcLoc -> SrcLoc
forall a. Ord a => a -> a -> a
min SrcLoc
loc1 SrcLoc
loc2)

checkOccurences :: Occurences -> TermTypeM ()
checkOccurences :: [Occurence] -> TermTypeM ()
checkOccurences = TermTypeM (Map VName ()) -> TermTypeM ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (TermTypeM (Map VName ()) -> TermTypeM ())
-> ([Occurence] -> TermTypeM (Map VName ()))
-> [Occurence]
-> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (VName -> [Usage] -> TermTypeM ())
-> UsageMap -> TermTypeM (Map VName ())
forall (t :: * -> *) k a b.
Applicative t =>
(k -> a -> t b) -> Map k a -> t (Map k b)
M.traverseWithKey VName -> [Usage] -> TermTypeM ()
comb (UsageMap -> TermTypeM (Map VName ()))
-> ([Occurence] -> UsageMap)
-> [Occurence]
-> TermTypeM (Map VName ())
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Occurence] -> UsageMap
usageMap
  where
    comb :: VName -> [Usage] -> TermTypeM ()
comb VName
_ [] = () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
    comb VName
name (Usage
u : [Usage]
us) = (Usage -> Usage -> TermTypeM Usage)
-> Usage -> [Usage] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m ()
foldM_ (VName -> Usage -> Usage -> TermTypeM Usage
combineOccurences VName
name) Usage
u [Usage]
us

allObserved :: Occurences -> Names
allObserved :: [Occurence] -> Names
allObserved = [Names] -> Names
forall (f :: * -> *) a. (Foldable f, Ord a) => f (Set a) -> Set a
S.unions ([Names] -> Names)
-> ([Occurence] -> [Names]) -> [Occurence] -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Occurence -> Names) -> [Occurence] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map Occurence -> Names
observed

allConsumed :: Occurences -> Names
allConsumed :: [Occurence] -> Names
allConsumed = [Names] -> Names
forall (f :: * -> *) a. (Foldable f, Ord a) => f (Set a) -> Set a
S.unions ([Names] -> Names)
-> ([Occurence] -> [Names]) -> [Occurence] -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Occurence -> Names) -> [Occurence] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map (Names -> Maybe Names -> Names
forall a. a -> Maybe a -> a
fromMaybe Names
forall a. Monoid a => a
mempty (Maybe Names -> Names)
-> (Occurence -> Maybe Names) -> Occurence -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Occurence -> Maybe Names
consumed)

allOccuring :: Occurences -> Names
allOccuring :: [Occurence] -> Names
allOccuring [Occurence]
occs = [Occurence] -> Names
allConsumed [Occurence]
occs Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> [Occurence] -> Names
allObserved [Occurence]
occs

anyConsumption :: Occurences -> Maybe Occurence
anyConsumption :: [Occurence] -> Maybe Occurence
anyConsumption = (Occurence -> Bool) -> [Occurence] -> Maybe Occurence
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Maybe a
find (Maybe Names -> Bool
forall a. Maybe a -> Bool
isJust (Maybe Names -> Bool)
-> (Occurence -> Maybe Names) -> Occurence -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Occurence -> Maybe Names
consumed)

seqOccurences :: Occurences -> Occurences -> Occurences
seqOccurences :: [Occurence] -> [Occurence] -> [Occurence]
seqOccurences [Occurence]
occurs1 [Occurence]
occurs2 =
  (Occurence -> Bool) -> [Occurence] -> [Occurence]
forall a. (a -> Bool) -> [a] -> [a]
filter (Bool -> Bool
not (Bool -> Bool) -> (Occurence -> Bool) -> Occurence -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Occurence -> Bool
nullOccurence) ([Occurence] -> [Occurence]) -> [Occurence] -> [Occurence]
forall a b. (a -> b) -> a -> b
$ (Occurence -> Occurence) -> [Occurence] -> [Occurence]
forall a b. (a -> b) -> [a] -> [b]
map Occurence -> Occurence
filt [Occurence]
occurs1 [Occurence] -> [Occurence] -> [Occurence]
forall a. [a] -> [a] -> [a]
++ [Occurence]
occurs2
  where
    filt :: Occurence -> Occurence
filt Occurence
occ =
      Occurence
occ {observed :: Names
observed = Occurence -> Names
observed Occurence
occ Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` Names
postcons}
    postcons :: Names
postcons = [Occurence] -> Names
allConsumed [Occurence]
occurs2

altOccurences :: Occurences -> Occurences -> Occurences
altOccurences :: [Occurence] -> [Occurence] -> [Occurence]
altOccurences [Occurence]
occurs1 [Occurence]
occurs2 =
  (Occurence -> Bool) -> [Occurence] -> [Occurence]
forall a. (a -> Bool) -> [a] -> [a]
filter (Bool -> Bool
not (Bool -> Bool) -> (Occurence -> Bool) -> Occurence -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Occurence -> Bool
nullOccurence) ([Occurence] -> [Occurence]) -> [Occurence] -> [Occurence]
forall a b. (a -> b) -> a -> b
$ (Occurence -> Occurence) -> [Occurence] -> [Occurence]
forall a b. (a -> b) -> [a] -> [b]
map Occurence -> Occurence
filt1 [Occurence]
occurs1 [Occurence] -> [Occurence] -> [Occurence]
forall a. [a] -> [a] -> [a]
++ (Occurence -> Occurence) -> [Occurence] -> [Occurence]
forall a b. (a -> b) -> [a] -> [b]
map Occurence -> Occurence
filt2 [Occurence]
occurs2
  where
    filt1 :: Occurence -> Occurence
filt1 Occurence
occ =
      Occurence
occ
        { consumed :: Maybe Names
consumed = Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
S.difference (Names -> Names -> Names) -> Maybe Names -> Maybe (Names -> Names)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Occurence -> Maybe Names
consumed Occurence
occ Maybe (Names -> Names) -> Maybe Names -> Maybe Names
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> Names -> Maybe Names
forall (f :: * -> *) a. Applicative f => a -> f a
pure Names
cons2,
          observed :: Names
observed = Occurence -> Names
observed Occurence
occ Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` Names
cons2
        }
    filt2 :: Occurence -> Occurence
filt2 Occurence
occ =
      Occurence
occ
        { consumed :: Maybe Names
consumed = Occurence -> Maybe Names
consumed Occurence
occ,
          observed :: Names
observed = Occurence -> Names
observed Occurence
occ Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` Names
cons1
        }
    cons1 :: Names
cons1 = [Occurence] -> Names
allConsumed [Occurence]
occurs1
    cons2 :: Names
cons2 = [Occurence] -> Names
allConsumed [Occurence]
occurs2

--- Scope management

data Checking
  = CheckingApply (Maybe (QualName VName)) Exp StructType StructType
  | CheckingReturn StructType StructType
  | CheckingAscription StructType StructType
  | CheckingLetGeneralise Name
  | CheckingParams (Maybe Name)
  | CheckingPat UncheckedPat InferredType
  | CheckingLoopBody StructType StructType
  | CheckingLoopInitial StructType StructType
  | CheckingRecordUpdate [Name] StructType StructType
  | CheckingRequired [StructType] StructType
  | CheckingBranches StructType StructType

instance Pretty Checking where
  ppr :: Checking -> Doc
ppr (CheckingApply Maybe (QualName VName)
f Exp
e StructType
expected StructType
actual) =
    Doc
header
      Doc -> Doc -> Doc
</> Doc
"Expected:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected)
      Doc -> Doc -> Doc
</> Doc
"Actual:  " Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
    where
      header :: Doc
header =
        case Maybe (QualName VName)
f of
          Maybe (QualName VName)
Nothing ->
            Doc
"Cannot apply function to"
              Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (String -> Doc
forall a. Pretty a => a -> Doc
shorten (String -> Doc) -> String -> Doc
forall a b. (a -> b) -> a -> b
$ Doc -> String
forall a. Pretty a => a -> String
pretty (Doc -> String) -> Doc -> String
forall a b. (a -> b) -> a -> b
$ Doc -> Doc
flatten (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ Exp -> Doc
forall a. Pretty a => a -> Doc
ppr Exp
e) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
" (invalid type)."
          Just QualName VName
fname ->
            Doc
"Cannot apply" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (QualName VName -> Doc
forall a. Pretty a => a -> Doc
ppr QualName VName
fname) Doc -> Doc -> Doc
<+> Doc
"to"
              Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (String -> Doc
forall a. Pretty a => a -> Doc
shorten (String -> Doc) -> String -> Doc
forall a b. (a -> b) -> a -> b
$ Doc -> String
forall a. Pretty a => a -> String
pretty (Doc -> String) -> Doc -> String
forall a b. (a -> b) -> a -> b
$ Doc -> Doc
flatten (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ Exp -> Doc
forall a. Pretty a => a -> Doc
ppr Exp
e) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
" (invalid type)."
  ppr (CheckingReturn StructType
expected StructType
actual) =
    Doc
"Function body does not have expected type."
      Doc -> Doc -> Doc
</> Doc
"Expected:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected)
      Doc -> Doc -> Doc
</> Doc
"Actual:  " Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
  ppr (CheckingAscription StructType
expected StructType
actual) =
    Doc
"Expression does not have expected type from explicit ascription."
      Doc -> Doc -> Doc
</> Doc
"Expected:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected)
      Doc -> Doc -> Doc
</> Doc
"Actual:  " Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
  ppr (CheckingLetGeneralise Name
fname) =
    Doc
"Cannot generalise type of" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Name -> Doc
forall a. Pretty a => a -> Doc
ppr Name
fname) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
  ppr (CheckingParams Maybe Name
fname) =
    Doc
"Invalid use of parameters in" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote Doc
fname' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
    where
      fname' :: Doc
fname' = Doc -> (Name -> Doc) -> Maybe Name -> Doc
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Doc
"anonymous function" Name -> Doc
forall a. Pretty a => a -> Doc
ppr Maybe Name
fname
  ppr (CheckingPat UncheckedPat
pat InferredType
NoneInferred) =
    Doc
"Invalid pattern" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (UncheckedPat -> Doc
forall a. Pretty a => a -> Doc
ppr UncheckedPat
pat) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
  ppr (CheckingPat UncheckedPat
pat (Ascribed PatType
t)) =
    Doc
"Pat" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (UncheckedPat -> Doc
forall a. Pretty a => a -> Doc
ppr UncheckedPat
pat)
      Doc -> Doc -> Doc
<+> Doc
"cannot match value of type"
      Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
t)
  ppr (CheckingLoopBody StructType
expected StructType
actual) =
    Doc
"Loop body does not have expected type."
      Doc -> Doc -> Doc
</> Doc
"Expected:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected)
      Doc -> Doc -> Doc
</> Doc
"Actual:  " Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
  ppr (CheckingLoopInitial StructType
expected StructType
actual) =
    Doc
"Initial loop values do not have expected type."
      Doc -> Doc -> Doc
</> Doc
"Expected:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected)
      Doc -> Doc -> Doc
</> Doc
"Actual:  " Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
  ppr (CheckingRecordUpdate [Name]
fs StructType
expected StructType
actual) =
    Doc
"Type mismatch when updating record field" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote Doc
fs' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
      Doc -> Doc -> Doc
</> Doc
"Existing:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected)
      Doc -> Doc -> Doc
</> Doc
"New:     " Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
    where
      fs' :: Doc
fs' = [Doc] -> Doc
forall a. Monoid a => [a] -> a
mconcat ([Doc] -> Doc) -> [Doc] -> Doc
forall a b. (a -> b) -> a -> b
$ Doc -> [Doc] -> [Doc]
punctuate Doc
"." ([Doc] -> [Doc]) -> [Doc] -> [Doc]
forall a b. (a -> b) -> a -> b
$ (Name -> Doc) -> [Name] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map Name -> Doc
forall a. Pretty a => a -> Doc
ppr [Name]
fs
  ppr (CheckingRequired [StructType
expected] StructType
actual) =
    Doc
"Expression must must have type" Doc -> Doc -> Doc
<+> StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
expected Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
      Doc -> Doc -> Doc
</> Doc
"Actual type:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
  ppr (CheckingRequired [StructType]
expected StructType
actual) =
    Doc
"Type of expression must must be one of " Doc -> Doc -> Doc
<+> Doc
expected' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
      Doc -> Doc -> Doc
</> Doc
"Actual type:" Doc -> Doc -> Doc
<+> Doc -> Doc
align (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
actual)
    where
      expected' :: Doc
expected' = [Doc] -> Doc
commasep ((StructType -> Doc) -> [StructType] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map StructType -> Doc
forall a. Pretty a => a -> Doc
ppr [StructType]
expected)
  ppr (CheckingBranches StructType
t1 StructType
t2) =
    Doc
"Conditional branches differ in type."
      Doc -> Doc -> Doc
</> Doc
"Former:" Doc -> Doc -> Doc
<+> StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
t1
      Doc -> Doc -> Doc
</> Doc
"Latter:" Doc -> Doc -> Doc
<+> StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
t2

-- | Whether something is a global or a local variable.
data Locality = Local | Global
  deriving (Int -> Locality -> ShowS
[Locality] -> ShowS
Locality -> String
(Int -> Locality -> ShowS)
-> (Locality -> String) -> ([Locality] -> ShowS) -> Show Locality
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Locality] -> ShowS
$cshowList :: [Locality] -> ShowS
show :: Locality -> String
$cshow :: Locality -> String
showsPrec :: Int -> Locality -> ShowS
$cshowsPrec :: Int -> Locality -> ShowS
Show)

data ValBinding
  = -- | Aliases in parameters indicate the lexical
    -- closure.
    BoundV Locality [TypeParam] PatType
  | OverloadedF [PrimType] [Maybe PrimType] (Maybe PrimType)
  | EqualityF
  | WasConsumed SrcLoc
  deriving (Int -> ValBinding -> ShowS
[ValBinding] -> ShowS
ValBinding -> String
(Int -> ValBinding -> ShowS)
-> (ValBinding -> String)
-> ([ValBinding] -> ShowS)
-> Show ValBinding
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [ValBinding] -> ShowS
$cshowList :: [ValBinding] -> ShowS
show :: ValBinding -> String
$cshow :: ValBinding -> String
showsPrec :: Int -> ValBinding -> ShowS
$cshowsPrec :: Int -> ValBinding -> ShowS
Show)

-- | Type checking happens with access to this environment.  The
-- 'TermScope' will be extended during type-checking as bindings come into
-- scope.
data TermEnv = TermEnv
  { TermEnv -> TermScope
termScope :: TermScope,
    TermEnv -> Maybe Checking
termChecking :: Maybe Checking,
    TermEnv -> Int
termLevel :: Level
  }

data TermScope = TermScope
  { TermScope -> Map VName ValBinding
scopeVtable :: M.Map VName ValBinding,
    TermScope -> Map VName TypeBinding
scopeTypeTable :: M.Map VName TypeBinding,
    TermScope -> Map VName Mod
scopeModTable :: M.Map VName Mod,
    TermScope -> NameMap
scopeNameMap :: NameMap
  }
  deriving (Int -> TermScope -> ShowS
[TermScope] -> ShowS
TermScope -> String
(Int -> TermScope -> ShowS)
-> (TermScope -> String)
-> ([TermScope] -> ShowS)
-> Show TermScope
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [TermScope] -> ShowS
$cshowList :: [TermScope] -> ShowS
show :: TermScope -> String
$cshow :: TermScope -> String
showsPrec :: Int -> TermScope -> ShowS
$cshowsPrec :: Int -> TermScope -> ShowS
Show)

instance Semigroup TermScope where
  TermScope Map VName ValBinding
vt1 Map VName TypeBinding
tt1 Map VName Mod
mt1 NameMap
nt1 <> :: TermScope -> TermScope -> TermScope
<> TermScope Map VName ValBinding
vt2 Map VName TypeBinding
tt2 Map VName Mod
mt2 NameMap
nt2 =
    Map VName ValBinding
-> Map VName TypeBinding -> Map VName Mod -> NameMap -> TermScope
TermScope (Map VName ValBinding
vt2 Map VName ValBinding
-> Map VName ValBinding -> Map VName ValBinding
forall k a. Ord k => Map k a -> Map k a -> Map k a
`M.union` Map VName ValBinding
vt1) (Map VName TypeBinding
tt2 Map VName TypeBinding
-> Map VName TypeBinding -> Map VName TypeBinding
forall k a. Ord k => Map k a -> Map k a -> Map k a
`M.union` Map VName TypeBinding
tt1) (Map VName Mod
mt1 Map VName Mod -> Map VName Mod -> Map VName Mod
forall k a. Ord k => Map k a -> Map k a -> Map k a
`M.union` Map VName Mod
mt2) (NameMap
nt2 NameMap -> NameMap -> NameMap
forall k a. Ord k => Map k a -> Map k a -> Map k a
`M.union` NameMap
nt1)

envToTermScope :: Env -> TermScope
envToTermScope :: Env -> TermScope
envToTermScope Env
env =
  TermScope :: Map VName ValBinding
-> Map VName TypeBinding -> Map VName Mod -> NameMap -> TermScope
TermScope
    { scopeVtable :: Map VName ValBinding
scopeVtable = Map VName ValBinding
vtable,
      scopeTypeTable :: Map VName TypeBinding
scopeTypeTable = Env -> Map VName TypeBinding
envTypeTable Env
env,
      scopeNameMap :: NameMap
scopeNameMap = Env -> NameMap
envNameMap Env
env,
      scopeModTable :: Map VName Mod
scopeModTable = Env -> Map VName Mod
envModTable Env
env
    }
  where
    vtable :: Map VName ValBinding
vtable = (VName -> BoundV -> ValBinding)
-> Map VName BoundV -> Map VName ValBinding
forall k a b. (k -> a -> b) -> Map k a -> Map k b
M.mapWithKey VName -> BoundV -> ValBinding
valBinding (Map VName BoundV -> Map VName ValBinding)
-> Map VName BoundV -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ Env -> Map VName BoundV
envVtable Env
env
    valBinding :: VName -> BoundV -> ValBinding
valBinding VName
k (TypeM.BoundV [TypeParam]
tps StructType
v) =
      Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
Global [TypeParam]
tps (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$
        StructType
v
          StructType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` (if StructType -> Int
forall dim as. TypeBase dim as -> Int
arrayRank StructType
v Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0 then Alias -> Aliasing
forall a. a -> Set a
S.singleton (VName -> Alias
AliasBound VName
k) else Aliasing
forall a. Monoid a => a
mempty)

withEnv :: TermEnv -> Env -> TermEnv
withEnv :: TermEnv -> Env -> TermEnv
withEnv TermEnv
tenv Env
env = TermEnv
tenv {termScope :: TermScope
termScope = TermEnv -> TermScope
termScope TermEnv
tenv TermScope -> TermScope -> TermScope
forall a. Semigroup a => a -> a -> a
<> Env -> TermScope
envToTermScope Env
env}

overloadedTypeVars :: Constraints -> Names
overloadedTypeVars :: Constraints -> Names
overloadedTypeVars = [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ([Names] -> Names)
-> (Constraints -> [Names]) -> Constraints -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((Int, Constraint) -> Names) -> [(Int, Constraint)] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map (Int, Constraint) -> Names
forall a. (a, Constraint) -> Names
f ([(Int, Constraint)] -> [Names])
-> (Constraints -> [(Int, Constraint)]) -> Constraints -> [Names]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Constraints -> [(Int, Constraint)]
forall k a. Map k a -> [a]
M.elems
  where
    f :: (a, Constraint) -> Names
f (a
_, HasFields Map Name StructType
fs Usage
_) = [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ([Names] -> Names) -> [Names] -> Names
forall a b. (a -> b) -> a -> b
$ (StructType -> Names) -> [StructType] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map StructType -> Names
forall as dim. Monoid as => TypeBase dim as -> Names
typeVars ([StructType] -> [Names]) -> [StructType] -> [Names]
forall a b. (a -> b) -> a -> b
$ Map Name StructType -> [StructType]
forall k a. Map k a -> [a]
M.elems Map Name StructType
fs
    f (a, Constraint)
_ = Names
forall a. Monoid a => a
mempty

-- | Get the type of an expression, with top level type variables
-- substituted.  Never call 'typeOf' directly (except in a few
-- carefully inspected locations)!
expType :: Exp -> TermTypeM PatType
expType :: Exp -> TermTypeM PatType
expType = PatType -> TermTypeM PatType
forall (m :: * -> *). MonadUnify m => PatType -> m PatType
normPatType (PatType -> TermTypeM PatType)
-> (Exp -> PatType) -> Exp -> TermTypeM PatType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> PatType
typeOf

-- | Get the type of an expression, with all type variables
-- substituted.  Slower than 'expType', but sometimes necessary.
-- Never call 'typeOf' directly (except in a few carefully inspected
-- locations)!
expTypeFully :: Exp -> TermTypeM PatType
expTypeFully :: Exp -> TermTypeM PatType
expTypeFully = PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully (PatType -> TermTypeM PatType)
-> (Exp -> PatType) -> Exp -> TermTypeM PatType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> PatType
typeOf

-- Wrap a function name to give it a vacuous Eq instance for SizeSource.
newtype FName = FName (Maybe (QualName VName))
  deriving (Int -> FName -> ShowS
[FName] -> ShowS
FName -> String
(Int -> FName -> ShowS)
-> (FName -> String) -> ([FName] -> ShowS) -> Show FName
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [FName] -> ShowS
$cshowList :: [FName] -> ShowS
show :: FName -> String
$cshow :: FName -> String
showsPrec :: Int -> FName -> ShowS
$cshowsPrec :: Int -> FName -> ShowS
Show)

instance Eq FName where
  FName
_ == :: FName -> FName -> Bool
== FName
_ = Bool
True

instance Ord FName where
  compare :: FName -> FName -> Ordering
compare FName
_ FName
_ = Ordering
EQ

-- | What was the source of some existential size?  This is used for
-- using the same existential variable if the same source is
-- encountered in multiple locations.
data SizeSource
  = SourceArg FName (ExpBase NoInfo VName)
  | SourceBound (ExpBase NoInfo VName)
  | SourceSlice
      (Maybe (DimDecl VName))
      (Maybe (ExpBase NoInfo VName))
      (Maybe (ExpBase NoInfo VName))
      (Maybe (ExpBase NoInfo VName))
  deriving (SizeSource -> SizeSource -> Bool
(SizeSource -> SizeSource -> Bool)
-> (SizeSource -> SizeSource -> Bool) -> Eq SizeSource
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: SizeSource -> SizeSource -> Bool
$c/= :: SizeSource -> SizeSource -> Bool
== :: SizeSource -> SizeSource -> Bool
$c== :: SizeSource -> SizeSource -> Bool
Eq, Eq SizeSource
Eq SizeSource
-> (SizeSource -> SizeSource -> Ordering)
-> (SizeSource -> SizeSource -> Bool)
-> (SizeSource -> SizeSource -> Bool)
-> (SizeSource -> SizeSource -> Bool)
-> (SizeSource -> SizeSource -> Bool)
-> (SizeSource -> SizeSource -> SizeSource)
-> (SizeSource -> SizeSource -> SizeSource)
-> Ord SizeSource
SizeSource -> SizeSource -> Bool
SizeSource -> SizeSource -> Ordering
SizeSource -> SizeSource -> SizeSource
forall a.
Eq a
-> (a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
min :: SizeSource -> SizeSource -> SizeSource
$cmin :: SizeSource -> SizeSource -> SizeSource
max :: SizeSource -> SizeSource -> SizeSource
$cmax :: SizeSource -> SizeSource -> SizeSource
>= :: SizeSource -> SizeSource -> Bool
$c>= :: SizeSource -> SizeSource -> Bool
> :: SizeSource -> SizeSource -> Bool
$c> :: SizeSource -> SizeSource -> Bool
<= :: SizeSource -> SizeSource -> Bool
$c<= :: SizeSource -> SizeSource -> Bool
< :: SizeSource -> SizeSource -> Bool
$c< :: SizeSource -> SizeSource -> Bool
compare :: SizeSource -> SizeSource -> Ordering
$ccompare :: SizeSource -> SizeSource -> Ordering
$cp1Ord :: Eq SizeSource
Ord, Int -> SizeSource -> ShowS
[SizeSource] -> ShowS
SizeSource -> String
(Int -> SizeSource -> ShowS)
-> (SizeSource -> String)
-> ([SizeSource] -> ShowS)
-> Show SizeSource
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [SizeSource] -> ShowS
$cshowList :: [SizeSource] -> ShowS
show :: SizeSource -> String
$cshow :: SizeSource -> String
showsPrec :: Int -> SizeSource -> ShowS
$cshowsPrec :: Int -> SizeSource -> ShowS
Show)

-- | A description of where an artificial compiler-generated
-- intermediate name came from.
data NameReason
  = -- | Name is the result of a function application.
    NameAppRes (Maybe (QualName VName)) SrcLoc

nameReason :: SrcLoc -> NameReason -> Doc
nameReason :: SrcLoc -> NameReason -> Doc
nameReason SrcLoc
loc (NameAppRes Maybe (QualName VName)
Nothing SrcLoc
apploc) =
  Doc
"result of application at" Doc -> Doc -> Doc
<+> String -> Doc
text (SrcLoc -> SrcLoc -> String
forall a b. (Located a, Located b) => a -> b -> String
locStrRel SrcLoc
loc SrcLoc
apploc)
nameReason SrcLoc
loc (NameAppRes Maybe (QualName VName)
fname SrcLoc
apploc) =
  Doc
"result of applying" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Maybe (QualName VName) -> Doc
forall a. Pretty a => a -> Doc
ppr Maybe (QualName VName)
fname)
    Doc -> Doc -> Doc
<+> Doc -> Doc
parens (Doc
"at" Doc -> Doc -> Doc
<+> String -> Doc
text (SrcLoc -> SrcLoc -> String
forall a b. (Located a, Located b) => a -> b -> String
locStrRel SrcLoc
loc SrcLoc
apploc))

-- | The state is a set of constraints and a counter for generating
-- type names.  This is distinct from the usual counter we use for
-- generating unique names, as these will be user-visible.
data TermTypeState = TermTypeState
  { TermTypeState -> Constraints
stateConstraints :: Constraints,
    TermTypeState -> Int
stateCounter :: !Int,
    -- | Mapping function arguments encountered to
    -- the sizes they ended up generating (when
    -- they could not be substituted directly).
    -- This happens for function arguments that are
    -- not constants or names.
    TermTypeState -> Map SizeSource VName
stateDimTable :: M.Map SizeSource VName,
    TermTypeState -> Map VName NameReason
stateNames :: M.Map VName NameReason,
    TermTypeState -> [Occurence]
stateOccs :: Occurences
  }

newtype TermTypeM a
  = TermTypeM (ReaderT TermEnv (StateT TermTypeState TypeM) a)
  deriving
    ( Applicative TermTypeM
a -> TermTypeM a
Applicative TermTypeM
-> (forall a b. TermTypeM a -> (a -> TermTypeM b) -> TermTypeM b)
-> (forall a b. TermTypeM a -> TermTypeM b -> TermTypeM b)
-> (forall a. a -> TermTypeM a)
-> Monad TermTypeM
TermTypeM a -> (a -> TermTypeM b) -> TermTypeM b
TermTypeM a -> TermTypeM b -> TermTypeM b
forall a. a -> TermTypeM a
forall a b. TermTypeM a -> TermTypeM b -> TermTypeM b
forall a b. TermTypeM a -> (a -> TermTypeM b) -> TermTypeM b
forall (m :: * -> *).
Applicative m
-> (forall a b. m a -> (a -> m b) -> m b)
-> (forall a b. m a -> m b -> m b)
-> (forall a. a -> m a)
-> Monad m
return :: a -> TermTypeM a
$creturn :: forall a. a -> TermTypeM a
>> :: TermTypeM a -> TermTypeM b -> TermTypeM b
$c>> :: forall a b. TermTypeM a -> TermTypeM b -> TermTypeM b
>>= :: TermTypeM a -> (a -> TermTypeM b) -> TermTypeM b
$c>>= :: forall a b. TermTypeM a -> (a -> TermTypeM b) -> TermTypeM b
$cp1Monad :: Applicative TermTypeM
Monad,
      a -> TermTypeM b -> TermTypeM a
(a -> b) -> TermTypeM a -> TermTypeM b
(forall a b. (a -> b) -> TermTypeM a -> TermTypeM b)
-> (forall a b. a -> TermTypeM b -> TermTypeM a)
-> Functor TermTypeM
forall a b. a -> TermTypeM b -> TermTypeM a
forall a b. (a -> b) -> TermTypeM a -> TermTypeM b
forall (f :: * -> *).
(forall a b. (a -> b) -> f a -> f b)
-> (forall a b. a -> f b -> f a) -> Functor f
<$ :: a -> TermTypeM b -> TermTypeM a
$c<$ :: forall a b. a -> TermTypeM b -> TermTypeM a
fmap :: (a -> b) -> TermTypeM a -> TermTypeM b
$cfmap :: forall a b. (a -> b) -> TermTypeM a -> TermTypeM b
Functor,
      Functor TermTypeM
a -> TermTypeM a
Functor TermTypeM
-> (forall a. a -> TermTypeM a)
-> (forall a b. TermTypeM (a -> b) -> TermTypeM a -> TermTypeM b)
-> (forall a b c.
    (a -> b -> c) -> TermTypeM a -> TermTypeM b -> TermTypeM c)
-> (forall a b. TermTypeM a -> TermTypeM b -> TermTypeM b)
-> (forall a b. TermTypeM a -> TermTypeM b -> TermTypeM a)
-> Applicative TermTypeM
TermTypeM a -> TermTypeM b -> TermTypeM b
TermTypeM a -> TermTypeM b -> TermTypeM a
TermTypeM (a -> b) -> TermTypeM a -> TermTypeM b
(a -> b -> c) -> TermTypeM a -> TermTypeM b -> TermTypeM c
forall a. a -> TermTypeM a
forall a b. TermTypeM a -> TermTypeM b -> TermTypeM a
forall a b. TermTypeM a -> TermTypeM b -> TermTypeM b
forall a b. TermTypeM (a -> b) -> TermTypeM a -> TermTypeM b
forall a b c.
(a -> b -> c) -> TermTypeM a -> TermTypeM b -> TermTypeM c
forall (f :: * -> *).
Functor f
-> (forall a. a -> f a)
-> (forall a b. f (a -> b) -> f a -> f b)
-> (forall a b c. (a -> b -> c) -> f a -> f b -> f c)
-> (forall a b. f a -> f b -> f b)
-> (forall a b. f a -> f b -> f a)
-> Applicative f
<* :: TermTypeM a -> TermTypeM b -> TermTypeM a
$c<* :: forall a b. TermTypeM a -> TermTypeM b -> TermTypeM a
*> :: TermTypeM a -> TermTypeM b -> TermTypeM b
$c*> :: forall a b. TermTypeM a -> TermTypeM b -> TermTypeM b
liftA2 :: (a -> b -> c) -> TermTypeM a -> TermTypeM b -> TermTypeM c
$cliftA2 :: forall a b c.
(a -> b -> c) -> TermTypeM a -> TermTypeM b -> TermTypeM c
<*> :: TermTypeM (a -> b) -> TermTypeM a -> TermTypeM b
$c<*> :: forall a b. TermTypeM (a -> b) -> TermTypeM a -> TermTypeM b
pure :: a -> TermTypeM a
$cpure :: forall a. a -> TermTypeM a
$cp1Applicative :: Functor TermTypeM
Applicative,
      MonadReader TermEnv,
      MonadState TermTypeState,
      MonadError TypeError
    )

instance MonadUnify TermTypeM where
  getConstraints :: TermTypeM Constraints
getConstraints = (TermTypeState -> Constraints) -> TermTypeM Constraints
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets TermTypeState -> Constraints
stateConstraints
  putConstraints :: Constraints -> TermTypeM ()
putConstraints Constraints
x = (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateConstraints :: Constraints
stateConstraints = Constraints
x}

  newTypeVar :: SrcLoc -> String -> TermTypeM (TypeBase dim als)
newTypeVar SrcLoc
loc String
desc = do
    Int
i <- TermTypeM Int
incCounter
    VName
v <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID (Name -> TermTypeM VName) -> Name -> TermTypeM VName
forall a b. (a -> b) -> a -> b
$ String -> Int -> Name
mkTypeVarName String
desc Int
i
    VName -> Constraint -> TermTypeM ()
constrain VName
v (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Liftedness -> Usage -> Constraint
NoConstraint Liftedness
Lifted (Usage -> Constraint) -> Usage -> Constraint
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
mkUsage' SrcLoc
loc
    TypeBase dim als -> TermTypeM (TypeBase dim als)
forall (m :: * -> *) a. Monad m => a -> m a
return (TypeBase dim als -> TermTypeM (TypeBase dim als))
-> TypeBase dim als -> TermTypeM (TypeBase dim als)
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase dim als -> TypeBase dim als
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim als -> TypeBase dim als)
-> ScalarTypeBase dim als -> TypeBase dim als
forall a b. (a -> b) -> a -> b
$ als
-> Uniqueness
-> TypeName
-> [TypeArg dim]
-> ScalarTypeBase dim als
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar als
forall a. Monoid a => a
mempty Uniqueness
Nonunique (VName -> TypeName
typeName VName
v) []

  curLevel :: TermTypeM Int
curLevel = (TermEnv -> Int) -> TermTypeM Int
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks TermEnv -> Int
termLevel

  newDimVar :: SrcLoc -> Rigidity -> String -> TermTypeM VName
newDimVar SrcLoc
loc Rigidity
rigidity String
name = do
    Int
i <- TermTypeM Int
incCounter
    VName
dim <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID (Name -> TermTypeM VName) -> Name -> TermTypeM VName
forall a b. (a -> b) -> a -> b
$ String -> Int -> Name
mkTypeVarName String
name Int
i
    case Rigidity
rigidity of
      Rigid RigidSource
rsrc -> VName -> Constraint -> TermTypeM ()
constrain VName
dim (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ SrcLoc -> RigidSource -> Constraint
UnknowableSize SrcLoc
loc RigidSource
rsrc
      Rigidity
Nonrigid -> VName -> Constraint -> TermTypeM ()
constrain VName
dim (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Maybe (DimDecl VName) -> Usage -> Constraint
Size Maybe (DimDecl VName)
forall a. Maybe a
Nothing (Usage -> Constraint) -> Usage -> Constraint
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
mkUsage' SrcLoc
loc
    VName -> TermTypeM VName
forall (m :: * -> *) a. Monad m => a -> m a
return VName
dim

  unifyError :: loc -> Notes -> BreadCrumbs -> Doc -> TermTypeM a
unifyError loc
loc Notes
notes BreadCrumbs
bcs Doc
doc = do
    Maybe Checking
checking <- (TermEnv -> Maybe Checking) -> TermTypeM (Maybe Checking)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks TermEnv -> Maybe Checking
termChecking
    case Maybe Checking
checking of
      Just Checking
checking' ->
        TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
          SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes (Doc -> TypeError) -> Doc -> TypeError
forall a b. (a -> b) -> a -> b
$
            Checking -> Doc
forall a. Pretty a => a -> Doc
ppr Checking
checking' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
line Doc -> Doc -> Doc
</> Doc
doc Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> BreadCrumbs -> Doc
forall a. Pretty a => a -> Doc
ppr BreadCrumbs
bcs
      Maybe Checking
Nothing ->
        TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes (Doc -> TypeError) -> Doc -> TypeError
forall a b. (a -> b) -> a -> b
$ Doc
doc Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> BreadCrumbs -> Doc
forall a. Pretty a => a -> Doc
ppr BreadCrumbs
bcs

  matchError :: loc
-> Notes -> BreadCrumbs -> StructType -> StructType -> TermTypeM a
matchError loc
loc Notes
notes BreadCrumbs
bcs StructType
t1 StructType
t2 = do
    Maybe Checking
checking <- (TermEnv -> Maybe Checking) -> TermTypeM (Maybe Checking)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks TermEnv -> Maybe Checking
termChecking
    case Maybe Checking
checking of
      Just Checking
checking'
        | BreadCrumbs -> Bool
hasNoBreadCrumbs BreadCrumbs
bcs ->
          TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
            SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes (Doc -> TypeError) -> Doc -> TypeError
forall a b. (a -> b) -> a -> b
$
              Checking -> Doc
forall a. Pretty a => a -> Doc
ppr Checking
checking'
        | Bool
otherwise ->
          TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
            SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes (Doc -> TypeError) -> Doc -> TypeError
forall a b. (a -> b) -> a -> b
$
              Checking -> Doc
forall a. Pretty a => a -> Doc
ppr Checking
checking' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
line Doc -> Doc -> Doc
</> Doc
doc Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> BreadCrumbs -> Doc
forall a. Pretty a => a -> Doc
ppr BreadCrumbs
bcs
      Maybe Checking
Nothing ->
        TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes (Doc -> TypeError) -> Doc -> TypeError
forall a b. (a -> b) -> a -> b
$ Doc
doc Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> BreadCrumbs -> Doc
forall a. Pretty a => a -> Doc
ppr BreadCrumbs
bcs
    where
      doc :: Doc
doc =
        Doc
"Types"
          Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
t1)
          Doc -> Doc -> Doc
</> Doc
"and"
          Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
t2)
          Doc -> Doc -> Doc
</> Doc
"do not match."

onFailure :: Checking -> TermTypeM a -> TermTypeM a
onFailure :: Checking -> TermTypeM a -> TermTypeM a
onFailure Checking
c = (TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a
forall r (m :: * -> *) a. MonadReader r m => (r -> r) -> m a -> m a
local ((TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a)
-> (TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \TermEnv
env -> TermEnv
env {termChecking :: Maybe Checking
termChecking = Checking -> Maybe Checking
forall a. a -> Maybe a
Just Checking
c}

runTermTypeM :: TermTypeM a -> TypeM (a, Occurences)
runTermTypeM :: TermTypeM a -> TypeM (a, [Occurence])
runTermTypeM (TermTypeM ReaderT TermEnv (StateT TermTypeState TypeM) a
m) = do
  TermScope
initial_scope <- (TermScope
initialTermScope TermScope -> TermScope -> TermScope
forall a. Semigroup a => a -> a -> a
<>) (TermScope -> TermScope) -> (Env -> TermScope) -> Env -> TermScope
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Env -> TermScope
envToTermScope (Env -> TermScope) -> TypeM Env -> TypeM TermScope
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> TypeM Env
askEnv
  let initial_tenv :: TermEnv
initial_tenv =
        TermEnv :: TermScope -> Maybe Checking -> Int -> TermEnv
TermEnv
          { termScope :: TermScope
termScope = TermScope
initial_scope,
            termChecking :: Maybe Checking
termChecking = Maybe Checking
forall a. Maybe a
Nothing,
            termLevel :: Int
termLevel = Int
0
          }
  (TermTypeState -> [Occurence])
-> (a, TermTypeState) -> (a, [Occurence])
forall (p :: * -> * -> *) b c a.
Bifunctor p =>
(b -> c) -> p a b -> p a c
second TermTypeState -> [Occurence]
stateOccs
    ((a, TermTypeState) -> (a, [Occurence]))
-> TypeM (a, TermTypeState) -> TypeM (a, [Occurence])
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> StateT TermTypeState TypeM a
-> TermTypeState -> TypeM (a, TermTypeState)
forall s (m :: * -> *) a. StateT s m a -> s -> m (a, s)
runStateT
      (ReaderT TermEnv (StateT TermTypeState TypeM) a
-> TermEnv -> StateT TermTypeState TypeM a
forall r (m :: * -> *) a. ReaderT r m a -> r -> m a
runReaderT ReaderT TermEnv (StateT TermTypeState TypeM) a
m TermEnv
initial_tenv)
      (Constraints
-> Int
-> Map SizeSource VName
-> Map VName NameReason
-> [Occurence]
-> TermTypeState
TermTypeState Constraints
forall a. Monoid a => a
mempty Int
0 Map SizeSource VName
forall a. Monoid a => a
mempty Map VName NameReason
forall a. Monoid a => a
mempty [Occurence]
forall a. Monoid a => a
mempty)

liftTypeM :: TypeM a -> TermTypeM a
liftTypeM :: TypeM a -> TermTypeM a
liftTypeM = ReaderT TermEnv (StateT TermTypeState TypeM) a -> TermTypeM a
forall a.
ReaderT TermEnv (StateT TermTypeState TypeM) a -> TermTypeM a
TermTypeM (ReaderT TermEnv (StateT TermTypeState TypeM) a -> TermTypeM a)
-> (TypeM a -> ReaderT TermEnv (StateT TermTypeState TypeM) a)
-> TypeM a
-> TermTypeM a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. StateT TermTypeState TypeM a
-> ReaderT TermEnv (StateT TermTypeState TypeM) a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (StateT TermTypeState TypeM a
 -> ReaderT TermEnv (StateT TermTypeState TypeM) a)
-> (TypeM a -> StateT TermTypeState TypeM a)
-> TypeM a
-> ReaderT TermEnv (StateT TermTypeState TypeM) a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TypeM a -> StateT TermTypeState TypeM a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift

localScope :: (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope :: (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope TermScope -> TermScope
f = (TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a
forall r (m :: * -> *) a. MonadReader r m => (r -> r) -> m a -> m a
local ((TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a)
-> (TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \TermEnv
tenv -> TermEnv
tenv {termScope :: TermScope
termScope = TermScope -> TermScope
f (TermScope -> TermScope) -> TermScope -> TermScope
forall a b. (a -> b) -> a -> b
$ TermEnv -> TermScope
termScope TermEnv
tenv}

incCounter :: TermTypeM Int
incCounter :: TermTypeM Int
incCounter = do
  TermTypeState
s <- TermTypeM TermTypeState
forall s (m :: * -> *). MonadState s m => m s
get
  TermTypeState -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => s -> m ()
put TermTypeState
s {stateCounter :: Int
stateCounter = TermTypeState -> Int
stateCounter TermTypeState
s Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1}
  Int -> TermTypeM Int
forall (m :: * -> *) a. Monad m => a -> m a
return (Int -> TermTypeM Int) -> Int -> TermTypeM Int
forall a b. (a -> b) -> a -> b
$ TermTypeState -> Int
stateCounter TermTypeState
s

extSize :: SrcLoc -> SizeSource -> TermTypeM (DimDecl VName, Maybe VName)
extSize :: SrcLoc -> SizeSource -> TermTypeM (DimDecl VName, Maybe VName)
extSize SrcLoc
loc SizeSource
e = do
  Maybe VName
prev <- (TermTypeState -> Maybe VName) -> TermTypeM (Maybe VName)
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets ((TermTypeState -> Maybe VName) -> TermTypeM (Maybe VName))
-> (TermTypeState -> Maybe VName) -> TermTypeM (Maybe VName)
forall a b. (a -> b) -> a -> b
$ SizeSource -> Map SizeSource VName -> Maybe VName
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup SizeSource
e (Map SizeSource VName -> Maybe VName)
-> (TermTypeState -> Map SizeSource VName)
-> TermTypeState
-> Maybe VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeState -> Map SizeSource VName
stateDimTable
  case Maybe VName
prev of
    Maybe VName
Nothing -> do
      let rsrc :: RigidSource
rsrc = case SizeSource
e of
            SourceArg (FName Maybe (QualName VName)
fname) ExpBase NoInfo VName
e' ->
              Maybe (QualName VName) -> String -> RigidSource
RigidArg Maybe (QualName VName)
fname (String -> RigidSource) -> String -> RigidSource
forall a b. (a -> b) -> a -> b
$ ExpBase NoInfo VName -> String
forall a. Pretty a => a -> String
prettyOneLine ExpBase NoInfo VName
e'
            SourceBound ExpBase NoInfo VName
e' ->
              String -> RigidSource
RigidBound (String -> RigidSource) -> String -> RigidSource
forall a b. (a -> b) -> a -> b
$ ExpBase NoInfo VName -> String
forall a. Pretty a => a -> String
prettyOneLine ExpBase NoInfo VName
e'
            SourceSlice Maybe (DimDecl VName)
d Maybe (ExpBase NoInfo VName)
i Maybe (ExpBase NoInfo VName)
j Maybe (ExpBase NoInfo VName)
s ->
              Maybe (DimDecl VName) -> String -> RigidSource
RigidSlice Maybe (DimDecl VName)
d (String -> RigidSource) -> String -> RigidSource
forall a b. (a -> b) -> a -> b
$ DimIndexBase NoInfo VName -> String
forall a. Pretty a => a -> String
prettyOneLine (DimIndexBase NoInfo VName -> String)
-> DimIndexBase NoInfo VName -> String
forall a b. (a -> b) -> a -> b
$ Maybe (ExpBase NoInfo VName)
-> Maybe (ExpBase NoInfo VName)
-> Maybe (ExpBase NoInfo VName)
-> DimIndexBase NoInfo VName
forall (f :: * -> *) vn.
Maybe (ExpBase f vn)
-> Maybe (ExpBase f vn)
-> Maybe (ExpBase f vn)
-> DimIndexBase f vn
DimSlice Maybe (ExpBase NoInfo VName)
i Maybe (ExpBase NoInfo VName)
j Maybe (ExpBase NoInfo VName)
s
      VName
d <- SrcLoc -> Rigidity -> String -> TermTypeM VName
forall (m :: * -> *).
MonadUnify m =>
SrcLoc -> Rigidity -> String -> m VName
newDimVar SrcLoc
loc (RigidSource -> Rigidity
Rigid RigidSource
rsrc) String
"n"
      (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateDimTable :: Map SizeSource VName
stateDimTable = SizeSource -> VName -> Map SizeSource VName -> Map SizeSource VName
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert SizeSource
e VName
d (Map SizeSource VName -> Map SizeSource VName)
-> Map SizeSource VName -> Map SizeSource VName
forall a b. (a -> b) -> a -> b
$ TermTypeState -> Map SizeSource VName
stateDimTable TermTypeState
s}
      (DimDecl VName, Maybe VName)
-> TermTypeM (DimDecl VName, Maybe VName)
forall (m :: * -> *) a. Monad m => a -> m a
return
        ( QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
d,
          VName -> Maybe VName
forall a. a -> Maybe a
Just VName
d
        )
    Just VName
d ->
      (DimDecl VName, Maybe VName)
-> TermTypeM (DimDecl VName, Maybe VName)
forall (m :: * -> *) a. Monad m => a -> m a
return
        ( QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
d,
          VName -> Maybe VName
forall a. a -> Maybe a
Just VName
d
        )

-- Any argument sizes created with 'extSize' inside the given action
-- will be removed once the action finishes.  This is to ensure that
-- just because e.g. @n+1@ appears as a size in one branch of a
-- conditional, that doesn't mean it's also available in the other branch.
noSizeEscape :: TermTypeM a -> TermTypeM a
noSizeEscape :: TermTypeM a -> TermTypeM a
noSizeEscape TermTypeM a
m = do
  Map SizeSource VName
dimtable <- (TermTypeState -> Map SizeSource VName)
-> TermTypeM (Map SizeSource VName)
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets TermTypeState -> Map SizeSource VName
stateDimTable
  a
x <- TermTypeM a
m
  (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateDimTable :: Map SizeSource VName
stateDimTable = Map SizeSource VName
dimtable}
  a -> TermTypeM a
forall (m :: * -> *) a. Monad m => a -> m a
return a
x

constrain :: VName -> Constraint -> TermTypeM ()
constrain :: VName -> Constraint -> TermTypeM ()
constrain VName
v Constraint
c = do
  Int
lvl <- TermTypeM Int
forall (m :: * -> *). MonadUnify m => m Int
curLevel
  (Constraints -> Constraints) -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
(Constraints -> Constraints) -> m ()
modifyConstraints ((Constraints -> Constraints) -> TermTypeM ())
-> (Constraints -> Constraints) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ VName -> (Int, Constraint) -> Constraints -> Constraints
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
v (Int
lvl, Constraint
c)

incLevel :: TermTypeM a -> TermTypeM a
incLevel :: TermTypeM a -> TermTypeM a
incLevel = (TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a
forall r (m :: * -> *) a. MonadReader r m => (r -> r) -> m a -> m a
local ((TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a)
-> (TermEnv -> TermEnv) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \TermEnv
env -> TermEnv
env {termLevel :: Int
termLevel = TermEnv -> Int
termLevel TermEnv
env Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1}

initialTermScope :: TermScope
initialTermScope :: TermScope
initialTermScope =
  TermScope :: Map VName ValBinding
-> Map VName TypeBinding -> Map VName Mod -> NameMap -> TermScope
TermScope
    { scopeVtable :: Map VName ValBinding
scopeVtable = Map VName ValBinding
initialVtable,
      scopeTypeTable :: Map VName TypeBinding
scopeTypeTable = Map VName TypeBinding
forall a. Monoid a => a
mempty,
      scopeNameMap :: NameMap
scopeNameMap = NameMap
topLevelNameMap,
      scopeModTable :: Map VName Mod
scopeModTable = Map VName Mod
forall a. Monoid a => a
mempty
    }
  where
    initialVtable :: Map VName ValBinding
initialVtable = [(VName, ValBinding)] -> Map VName ValBinding
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([(VName, ValBinding)] -> Map VName ValBinding)
-> [(VName, ValBinding)] -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ ((VName, Intrinsic) -> Maybe (VName, ValBinding))
-> [(VName, Intrinsic)] -> [(VName, ValBinding)]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe (VName, Intrinsic) -> Maybe (VName, ValBinding)
forall a. (a, Intrinsic) -> Maybe (a, ValBinding)
addIntrinsicF ([(VName, Intrinsic)] -> [(VName, ValBinding)])
-> [(VName, Intrinsic)] -> [(VName, ValBinding)]
forall a b. (a -> b) -> a -> b
$ Map VName Intrinsic -> [(VName, Intrinsic)]
forall k a. Map k a -> [(k, a)]
M.toList Map VName Intrinsic
intrinsics

    prim :: PrimType -> TypeBase dim as
prim = ScalarTypeBase dim as -> TypeBase dim as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim as -> TypeBase dim as)
-> (PrimType -> ScalarTypeBase dim as)
-> PrimType
-> TypeBase dim as
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PrimType -> ScalarTypeBase dim as
forall dim as. PrimType -> ScalarTypeBase dim as
Prim
    arrow :: TypeBase dim as -> TypeBase dim as -> TypeBase dim as
arrow TypeBase dim as
x TypeBase dim as
y = ScalarTypeBase dim as -> TypeBase dim as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim as -> TypeBase dim as)
-> ScalarTypeBase dim as -> TypeBase dim as
forall a b. (a -> b) -> a -> b
$ as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow as
forall a. Monoid a => a
mempty PName
Unnamed TypeBase dim as
x TypeBase dim as
y

    addIntrinsicF :: (a, Intrinsic) -> Maybe (a, ValBinding)
addIntrinsicF (a
name, IntrinsicMonoFun [PrimType]
pts PrimType
t) =
      (a, ValBinding) -> Maybe (a, ValBinding)
forall a. a -> Maybe a
Just (a
name, Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
Global [] (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$ PatType -> PatType -> PatType
forall as dim.
Monoid as =>
TypeBase dim as -> TypeBase dim as -> TypeBase dim as
arrow PatType
forall dim as. TypeBase dim as
pts' (PatType -> PatType) -> PatType -> PatType
forall a b. (a -> b) -> a -> b
$ PrimType -> PatType
forall dim as. PrimType -> TypeBase dim as
prim PrimType
t)
      where
        pts' :: TypeBase dim as
pts' = case [PrimType]
pts of
          [PrimType
pt] -> PrimType -> TypeBase dim as
forall dim as. PrimType -> TypeBase dim as
prim PrimType
pt
          [PrimType]
_ -> [TypeBase dim as] -> TypeBase dim as
forall dim as. [TypeBase dim as] -> TypeBase dim as
tupleRecord ([TypeBase dim as] -> TypeBase dim as)
-> [TypeBase dim as] -> TypeBase dim as
forall a b. (a -> b) -> a -> b
$ (PrimType -> TypeBase dim as) -> [PrimType] -> [TypeBase dim as]
forall a b. (a -> b) -> [a] -> [b]
map PrimType -> TypeBase dim as
forall dim as. PrimType -> TypeBase dim as
prim [PrimType]
pts
    addIntrinsicF (a
name, IntrinsicOverloadedFun [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rts) =
      (a, ValBinding) -> Maybe (a, ValBinding)
forall a. a -> Maybe a
Just (a
name, [PrimType] -> [Maybe PrimType] -> Maybe PrimType -> ValBinding
OverloadedF [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rts)
    addIntrinsicF (a
name, IntrinsicPolyFun [TypeParam]
tvs [StructType]
pts StructType
rt) =
      (a, ValBinding) -> Maybe (a, ValBinding)
forall a. a -> Maybe a
Just
        ( a
name,
          Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
Global [TypeParam]
tvs (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$
            StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct (StructType -> PatType) -> StructType -> PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ ()
-> PName
-> StructType
-> StructType
-> ScalarTypeBase (DimDecl VName) ()
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow ()
forall a. Monoid a => a
mempty PName
Unnamed StructType
pts' StructType
rt
        )
      where
        pts' :: StructType
pts' = case [StructType]
pts of
          [StructType
pt] -> StructType
pt
          [StructType]
_ -> [StructType] -> StructType
forall dim as. [TypeBase dim as] -> TypeBase dim as
tupleRecord [StructType]
pts
    addIntrinsicF (a
name, Intrinsic
IntrinsicEquality) =
      (a, ValBinding) -> Maybe (a, ValBinding)
forall a. a -> Maybe a
Just (a
name, ValBinding
EqualityF)
    addIntrinsicF (a, Intrinsic)
_ = Maybe (a, ValBinding)
forall a. Maybe a
Nothing

instance MonadTypeChecker TermTypeM where
  warn :: loc -> Doc -> TermTypeM ()
warn loc
loc Doc
problem = TypeM () -> TermTypeM ()
forall a. TypeM a -> TermTypeM a
liftTypeM (TypeM () -> TermTypeM ()) -> TypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ loc -> Doc -> TypeM ()
forall (m :: * -> *) loc.
(MonadTypeChecker m, Located loc) =>
loc -> Doc -> m ()
warn loc
loc Doc
problem
  newName :: VName -> TermTypeM VName
newName = TypeM VName -> TermTypeM VName
forall a. TypeM a -> TermTypeM a
liftTypeM (TypeM VName -> TermTypeM VName)
-> (VName -> TypeM VName) -> VName -> TermTypeM VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> TypeM VName
forall (m :: * -> *). MonadTypeChecker m => VName -> m VName
newName
  newID :: Name -> TermTypeM VName
newID = TypeM VName -> TermTypeM VName
forall a. TypeM a -> TermTypeM a
liftTypeM (TypeM VName -> TermTypeM VName)
-> (Name -> TypeM VName) -> Name -> TermTypeM VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Name -> TypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID

  checkQualName :: Namespace -> QualName Name -> SrcLoc -> TermTypeM (QualName VName)
checkQualName Namespace
space QualName Name
name SrcLoc
loc = (TermScope, QualName VName) -> QualName VName
forall a b. (a, b) -> b
snd ((TermScope, QualName VName) -> QualName VName)
-> TermTypeM (TermScope, QualName VName)
-> TermTypeM (QualName VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkQualNameWithEnv Namespace
space QualName Name
name SrcLoc
loc

  bindNameMap :: NameMap -> TermTypeM a -> TermTypeM a
bindNameMap NameMap
m = (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope ((TermScope -> TermScope) -> TermTypeM a -> TermTypeM a)
-> (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \TermScope
scope ->
    TermScope
scope {scopeNameMap :: NameMap
scopeNameMap = NameMap
m NameMap -> NameMap -> NameMap
forall a. Semigroup a => a -> a -> a
<> TermScope -> NameMap
scopeNameMap TermScope
scope}

  bindVal :: VName -> BoundV -> TermTypeM a -> TermTypeM a
bindVal VName
v (TypeM.BoundV [TypeParam]
tps StructType
t) = (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope ((TermScope -> TermScope) -> TermTypeM a -> TermTypeM a)
-> (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \TermScope
scope ->
    TermScope
scope {scopeVtable :: Map VName ValBinding
scopeVtable = VName -> ValBinding -> Map VName ValBinding -> Map VName ValBinding
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
v ValBinding
vb (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable TermScope
scope}
    where
      vb :: ValBinding
vb = Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
Local [TypeParam]
tps (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t

  lookupType :: SrcLoc
-> QualName Name
-> TermTypeM (QualName VName, [TypeParam], StructType, Liftedness)
lookupType SrcLoc
loc QualName Name
qn = do
    Env
outer_env <- TypeM Env -> TermTypeM Env
forall a. TypeM a -> TermTypeM a
liftTypeM TypeM Env
askEnv
    (TermScope
scope, qn' :: QualName VName
qn'@(QualName [VName]
qs VName
name)) <- Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkQualNameWithEnv Namespace
Type QualName Name
qn SrcLoc
loc
    case VName -> Map VName TypeBinding -> Maybe TypeBinding
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
name (Map VName TypeBinding -> Maybe TypeBinding)
-> Map VName TypeBinding -> Maybe TypeBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName TypeBinding
scopeTypeTable TermScope
scope of
      Maybe TypeBinding
Nothing -> SrcLoc
-> QualName Name
-> TermTypeM (QualName VName, [TypeParam], StructType, Liftedness)
forall (m :: * -> *) a.
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m a
unknownType SrcLoc
loc QualName Name
qn
      Just (TypeAbbr Liftedness
l [TypeParam]
ps StructType
def) ->
        (QualName VName, [TypeParam], StructType, Liftedness)
-> TermTypeM (QualName VName, [TypeParam], StructType, Liftedness)
forall (m :: * -> *) a. Monad m => a -> m a
return (QualName VName
qn', [TypeParam]
ps, Env -> [VName] -> [VName] -> StructType -> StructType
forall as.
Env
-> [VName]
-> [VName]
-> TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
qualifyTypeVars Env
outer_env ((TypeParam -> VName) -> [TypeParam] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName [TypeParam]
ps) [VName]
qs StructType
def, Liftedness
l)

  lookupMod :: SrcLoc -> QualName Name -> TermTypeM (QualName VName, Mod)
lookupMod SrcLoc
loc QualName Name
qn = do
    (TermScope
scope, qn' :: QualName VName
qn'@(QualName [VName]
_ VName
name)) <- Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkQualNameWithEnv Namespace
Term QualName Name
qn SrcLoc
loc
    case VName -> Map VName Mod -> Maybe Mod
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
name (Map VName Mod -> Maybe Mod) -> Map VName Mod -> Maybe Mod
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName Mod
scopeModTable TermScope
scope of
      Maybe Mod
Nothing -> Namespace
-> QualName Name -> SrcLoc -> TermTypeM (QualName VName, Mod)
forall (m :: * -> *) a.
MonadTypeChecker m =>
Namespace -> QualName Name -> SrcLoc -> m a
unknownVariable Namespace
Term QualName Name
qn SrcLoc
loc
      Just Mod
m -> (QualName VName, Mod) -> TermTypeM (QualName VName, Mod)
forall (m :: * -> *) a. Monad m => a -> m a
return (QualName VName
qn', Mod
m)

  lookupVar :: SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
lookupVar SrcLoc
loc QualName Name
qn = do
    Env
outer_env <- TypeM Env -> TermTypeM Env
forall a. TypeM a -> TermTypeM a
liftTypeM TypeM Env
askEnv
    (TermScope
scope, qn' :: QualName VName
qn'@(QualName [VName]
qs VName
name)) <- Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkQualNameWithEnv Namespace
Term QualName Name
qn SrcLoc
loc
    let usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage SrcLoc
loc (String -> Usage) -> String -> Usage
forall a b. (a -> b) -> a -> b
$ String
"use of " String -> ShowS
forall a. [a] -> [a] -> [a]
++ ShowS
quote (QualName Name -> String
forall a. Pretty a => a -> String
pretty QualName Name
qn)

    PatType
t <- case VName -> Map VName ValBinding -> Maybe ValBinding
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
name (Map VName ValBinding -> Maybe ValBinding)
-> Map VName ValBinding -> Maybe ValBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable TermScope
scope of
      Maybe ValBinding
Nothing ->
        SrcLoc -> Notes -> Doc -> TermTypeM PatType
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM PatType) -> Doc -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$
          Doc
"Unknown variable" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (QualName Name -> Doc
forall a. Pretty a => a -> Doc
ppr QualName Name
qn) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
      Just (WasConsumed SrcLoc
wloc) -> VName -> SrcLoc -> SrcLoc -> TermTypeM PatType
forall a. VName -> SrcLoc -> SrcLoc -> TermTypeM a
useAfterConsume VName
name SrcLoc
loc SrcLoc
wloc
      Just (BoundV Locality
_ [TypeParam]
tparams PatType
t)
        | String
"_" String -> String -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf` VName -> String
baseString VName
name -> SrcLoc -> QualName Name -> TermTypeM PatType
forall (m :: * -> *) a.
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m a
underscoreUse SrcLoc
loc QualName Name
qn
        | Bool
otherwise -> do
          ([VName]
tnames, PatType
t') <- SrcLoc -> [TypeParam] -> PatType -> TermTypeM ([VName], PatType)
instantiateTypeScheme SrcLoc
loc [TypeParam]
tparams PatType
t
          PatType -> TermTypeM PatType
forall (m :: * -> *) a. Monad m => a -> m a
return (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ Env -> [VName] -> [VName] -> PatType -> PatType
forall as.
Env
-> [VName]
-> [VName]
-> TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
qualifyTypeVars Env
outer_env [VName]
tnames [VName]
qs PatType
t'
      Just ValBinding
EqualityF -> do
        PatType
argtype <- SrcLoc -> String -> TermTypeM PatType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
        Usage -> PatType -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> TypeBase dim as -> m ()
equalityType Usage
usage PatType
argtype
        PatType -> TermTypeM PatType
forall (m :: * -> *) a. Monad m => a -> m a
return (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$
          ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$
            Aliasing
-> PName
-> PatType
-> PatType
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow Aliasing
forall a. Monoid a => a
mempty PName
Unnamed PatType
argtype (PatType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$
              ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> PName
-> PatType
-> PatType
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow Aliasing
forall a. Monoid a => a
mempty PName
Unnamed PatType
argtype (PatType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. PrimType -> ScalarTypeBase dim as
Prim PrimType
Bool
      Just (OverloadedF [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rt) -> do
        StructType
argtype <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
        [PrimType] -> Usage -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
[PrimType] -> Usage -> StructType -> m ()
mustBeOneOf [PrimType]
ts Usage
usage StructType
argtype
        let ([StructType]
pts', StructType
rt') = StructType
-> [Maybe PrimType] -> Maybe PrimType -> ([StructType], StructType)
forall dim as.
TypeBase dim as
-> [Maybe PrimType]
-> Maybe PrimType
-> ([TypeBase dim ()], TypeBase dim ())
instOverloaded StructType
argtype [Maybe PrimType]
pts Maybe PrimType
rt
            arrow :: TypeBase dim as -> TypeBase dim as -> TypeBase dim as
arrow TypeBase dim as
xt TypeBase dim as
yt = ScalarTypeBase dim as -> TypeBase dim as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim as -> TypeBase dim as)
-> ScalarTypeBase dim as -> TypeBase dim as
forall a b. (a -> b) -> a -> b
$ as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow as
forall a. Monoid a => a
mempty PName
Unnamed TypeBase dim as
xt TypeBase dim as
yt
        PatType -> TermTypeM PatType
forall (m :: * -> *) a. Monad m => a -> m a
return (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct (StructType -> PatType) -> StructType -> PatType
forall a b. (a -> b) -> a -> b
$ (StructType -> StructType -> StructType)
-> StructType -> [StructType] -> StructType
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr StructType -> StructType -> StructType
forall as dim.
Monoid as =>
TypeBase dim as -> TypeBase dim as -> TypeBase dim as
arrow StructType
rt' [StructType]
pts'

    Ident -> TermTypeM ()
observe (Ident -> TermTypeM ()) -> Ident -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> Ident
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident VName
name (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc
    (QualName VName, PatType) -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *) a. Monad m => a -> m a
return (QualName VName
qn', PatType
t)
    where
      instOverloaded :: TypeBase dim as
-> [Maybe PrimType]
-> Maybe PrimType
-> ([TypeBase dim ()], TypeBase dim ())
instOverloaded TypeBase dim as
argtype [Maybe PrimType]
pts Maybe PrimType
rt =
        ( (Maybe PrimType -> TypeBase dim ())
-> [Maybe PrimType] -> [TypeBase dim ()]
forall a b. (a -> b) -> [a] -> [b]
map (TypeBase dim ()
-> (PrimType -> TypeBase dim ())
-> Maybe PrimType
-> TypeBase dim ()
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (TypeBase dim as -> TypeBase dim ()
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase dim as
argtype) (ScalarTypeBase dim () -> TypeBase dim ()
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim () -> TypeBase dim ())
-> (PrimType -> ScalarTypeBase dim ())
-> PrimType
-> TypeBase dim ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PrimType -> ScalarTypeBase dim ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim)) [Maybe PrimType]
pts,
          TypeBase dim ()
-> (PrimType -> TypeBase dim ())
-> Maybe PrimType
-> TypeBase dim ()
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (TypeBase dim as -> TypeBase dim ()
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase dim as
argtype) (ScalarTypeBase dim () -> TypeBase dim ()
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim () -> TypeBase dim ())
-> (PrimType -> ScalarTypeBase dim ())
-> PrimType
-> TypeBase dim ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PrimType -> ScalarTypeBase dim ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim) Maybe PrimType
rt
        )

  checkNamedDim :: SrcLoc -> QualName Name -> TermTypeM (QualName VName)
checkNamedDim SrcLoc
loc QualName Name
v = do
    (QualName VName
v', PatType
t) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
loc QualName Name
v
    Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure ([StructType] -> StructType -> Checking
CheckingRequired [ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) ())
-> PrimType -> ScalarTypeBase (DimDecl VName) ()
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int64] (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t)) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
      Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"use as array size") (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t) (StructType -> TermTypeM ()) -> StructType -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) ())
-> PrimType -> ScalarTypeBase (DimDecl VName) ()
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int64
    QualName VName -> TermTypeM (QualName VName)
forall (m :: * -> *) a. Monad m => a -> m a
return QualName VName
v'

  typeError :: loc -> Notes -> Doc -> TermTypeM a
typeError loc
loc Notes
notes Doc
s = do
    Maybe Checking
checking <- (TermEnv -> Maybe Checking) -> TermTypeM (Maybe Checking)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks TermEnv -> Maybe Checking
termChecking
    case Maybe Checking
checking of
      Just Checking
checking' ->
        TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes (Checking -> Doc
forall a. Pretty a => a -> Doc
ppr Checking
checking' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
line Doc -> Doc -> Doc
</> Doc
s)
      Maybe Checking
Nothing ->
        TypeError -> TermTypeM a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TypeError -> TermTypeM a) -> TypeError -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Notes -> Doc -> TypeError
TypeError (loc -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf loc
loc) Notes
notes Doc
s

checkQualNameWithEnv :: Namespace -> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkQualNameWithEnv :: Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkQualNameWithEnv Namespace
space qn :: QualName Name
qn@(QualName [Name]
quals Name
name) SrcLoc
loc = do
  TermScope
scope <- (TermEnv -> TermScope) -> TermTypeM TermScope
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks TermEnv -> TermScope
termScope
  TermScope -> [Name] -> TermTypeM (TermScope, QualName VName)
descend TermScope
scope [Name]
quals
  where
    descend :: TermScope -> [Name] -> TermTypeM (TermScope, QualName VName)
descend TermScope
scope []
      | Just QualName VName
name' <- (Namespace, Name) -> NameMap -> Maybe (QualName VName)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup (Namespace
space, Name
name) (NameMap -> Maybe (QualName VName))
-> NameMap -> Maybe (QualName VName)
forall a b. (a -> b) -> a -> b
$ TermScope -> NameMap
scopeNameMap TermScope
scope =
        (TermScope, QualName VName)
-> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (TermScope
scope, QualName VName
name')
      | Bool
otherwise =
        Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a.
MonadTypeChecker m =>
Namespace -> QualName Name -> SrcLoc -> m a
unknownVariable Namespace
space QualName Name
qn SrcLoc
loc
    descend TermScope
scope (Name
q : [Name]
qs)
      | Just (QualName [VName]
_ VName
q') <- (Namespace, Name) -> NameMap -> Maybe (QualName VName)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup (Namespace
Term, Name
q) (NameMap -> Maybe (QualName VName))
-> NameMap -> Maybe (QualName VName)
forall a b. (a -> b) -> a -> b
$ TermScope -> NameMap
scopeNameMap TermScope
scope,
        Just Mod
res <- VName -> Map VName Mod -> Maybe Mod
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
q' (Map VName Mod -> Maybe Mod) -> Map VName Mod -> Maybe Mod
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName Mod
scopeModTable TermScope
scope =
        case Mod
res of
          -- Check if we are referring to the magical intrinsics
          -- module.
          Mod
_
            | VName -> Int
baseTag VName
q' Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
maxIntrinsicTag ->
              Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkIntrinsic Namespace
space QualName Name
qn SrcLoc
loc
          ModEnv Env
q_scope -> do
            (TermScope
scope', QualName [VName]
qs' VName
name') <- TermScope -> [Name] -> TermTypeM (TermScope, QualName VName)
descend (Env -> TermScope
envToTermScope Env
q_scope) [Name]
qs
            (TermScope, QualName VName)
-> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (TermScope
scope', [VName] -> VName -> QualName VName
forall vn. [vn] -> vn -> QualName vn
QualName (VName
q' VName -> [VName] -> [VName]
forall a. a -> [a] -> [a]
: [VName]
qs') VName
name')
          ModFun {} -> SrcLoc -> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a. MonadTypeChecker m => SrcLoc -> m a
unappliedFunctor SrcLoc
loc
      | Bool
otherwise =
        Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a.
MonadTypeChecker m =>
Namespace -> QualName Name -> SrcLoc -> m a
unknownVariable Namespace
space QualName Name
qn SrcLoc
loc

checkIntrinsic :: Namespace -> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkIntrinsic :: Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
checkIntrinsic Namespace
space qn :: QualName Name
qn@(QualName [Name]
_ Name
name) SrcLoc
loc
  | Just QualName VName
v <- (Namespace, Name) -> NameMap -> Maybe (QualName VName)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup (Namespace
space, Name
name) NameMap
intrinsicsNameMap = do
    ImportName
me <- TypeM ImportName -> TermTypeM ImportName
forall a. TypeM a -> TermTypeM a
liftTypeM TypeM ImportName
askImportName
    Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (String
"/prelude" String -> String -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf` ImportName -> String
includeToFilePath ImportName
me) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
      SrcLoc -> Doc -> TermTypeM ()
forall (m :: * -> *) loc.
(MonadTypeChecker m, Located loc) =>
loc -> Doc -> m ()
warn SrcLoc
loc Doc
"Using intrinsic functions directly can easily crash the compiler or result in wrong code generation."
    TermScope
scope <- (TermEnv -> TermScope) -> TermTypeM TermScope
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks TermEnv -> TermScope
termScope
    (TermScope, QualName VName)
-> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (TermScope
scope, QualName VName
v)
  | Bool
otherwise =
    Namespace
-> QualName Name -> SrcLoc -> TermTypeM (TermScope, QualName VName)
forall (m :: * -> *) a.
MonadTypeChecker m =>
Namespace -> QualName Name -> SrcLoc -> m a
unknownVariable Namespace
space QualName Name
qn SrcLoc
loc

-- | Wrap 'Types.checkTypeDecl' to also perform an observation of
-- every size in the type.
checkTypeDecl :: TypeDeclBase NoInfo Name -> TermTypeM (TypeDeclBase Info VName)
checkTypeDecl :: TypeDeclBase NoInfo Name -> TermTypeM (TypeDeclBase Info VName)
checkTypeDecl TypeDeclBase NoInfo Name
tdecl = do
  (TypeDeclBase Info VName
tdecl', Liftedness
_) <- TypeDeclBase NoInfo Name
-> TermTypeM (TypeDeclBase Info VName, Liftedness)
forall (m :: * -> *).
MonadTypeChecker m =>
TypeDeclBase NoInfo Name -> m (TypeDeclBase Info VName, Liftedness)
Types.checkTypeDecl TypeDeclBase NoInfo Name
tdecl
  (DimDecl VName -> TermTypeM ()) -> [DimDecl VName] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ DimDecl VName -> TermTypeM ()
observeDim ([DimDecl VName] -> TermTypeM ())
-> [DimDecl VName] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ StructType -> [DimDecl VName]
forall as. TypeBase (DimDecl VName) as -> [DimDecl VName]
nestedDims (StructType -> [DimDecl VName]) -> StructType -> [DimDecl VName]
forall a b. (a -> b) -> a -> b
$ Info StructType -> StructType
forall a. Info a -> a
unInfo (Info StructType -> StructType) -> Info StructType -> StructType
forall a b. (a -> b) -> a -> b
$ TypeDeclBase Info VName -> Info StructType
forall (f :: * -> *) vn. TypeDeclBase f vn -> f StructType
expandedType TypeDeclBase Info VName
tdecl'
  TypeDeclBase Info VName -> TermTypeM (TypeDeclBase Info VName)
forall (m :: * -> *) a. Monad m => a -> m a
return TypeDeclBase Info VName
tdecl'
  where
    observeDim :: DimDecl VName -> TermTypeM ()
observeDim (NamedDim QualName VName
v) =
      Ident -> TermTypeM ()
observe (Ident -> TermTypeM ()) -> Ident -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> Ident
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident (QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
v) (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int64) SrcLoc
forall a. Monoid a => a
mempty
    observeDim DimDecl VName
_ = () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

-- | Instantiate a type scheme with fresh type variables for its type
-- parameters. Returns the names of the fresh type variables, the
-- instance list, and the instantiated type.
instantiateTypeScheme ::
  SrcLoc ->
  [TypeParam] ->
  PatType ->
  TermTypeM ([VName], PatType)
instantiateTypeScheme :: SrcLoc -> [TypeParam] -> PatType -> TermTypeM ([VName], PatType)
instantiateTypeScheme SrcLoc
loc [TypeParam]
tparams PatType
t = do
  let tnames :: [VName]
tnames = (TypeParam -> VName) -> [TypeParam] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName [TypeParam]
tparams
  ([VName]
tparam_names, [Subst StructType]
tparam_substs) <- [(VName, Subst StructType)] -> ([VName], [Subst StructType])
forall a b. [(a, b)] -> ([a], [b])
unzip ([(VName, Subst StructType)] -> ([VName], [Subst StructType]))
-> TermTypeM [(VName, Subst StructType)]
-> TermTypeM ([VName], [Subst StructType])
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (TypeParam -> TermTypeM (VName, Subst StructType))
-> [TypeParam] -> TermTypeM [(VName, Subst StructType)]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM (SrcLoc -> TypeParam -> TermTypeM (VName, Subst StructType)
forall as dim.
Monoid as =>
SrcLoc -> TypeParam -> TermTypeM (VName, Subst (TypeBase dim as))
instantiateTypeParam SrcLoc
loc) [TypeParam]
tparams
  let substs :: Map VName (Subst StructType)
substs = [(VName, Subst StructType)] -> Map VName (Subst StructType)
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([(VName, Subst StructType)] -> Map VName (Subst StructType))
-> [(VName, Subst StructType)] -> Map VName (Subst StructType)
forall a b. (a -> b) -> a -> b
$ [VName] -> [Subst StructType] -> [(VName, Subst StructType)]
forall a b. [a] -> [b] -> [(a, b)]
zip [VName]
tnames [Subst StructType]
tparam_substs
      t' :: PatType
t' = TypeSubs -> PatType -> PatType
forall a. Substitutable a => TypeSubs -> a -> a
applySubst (VName -> Map VName (Subst StructType) -> Maybe (Subst StructType)
forall k a. Ord k => k -> Map k a -> Maybe a
`M.lookup` Map VName (Subst StructType)
substs) PatType
t
  ([VName], PatType) -> TermTypeM ([VName], PatType)
forall (m :: * -> *) a. Monad m => a -> m a
return ([VName]
tparam_names, PatType
t')

-- | Create a new type name and insert it (unconstrained) in the
-- substitution map.
instantiateTypeParam :: Monoid as => SrcLoc -> TypeParam -> TermTypeM (VName, Subst (TypeBase dim as))
instantiateTypeParam :: SrcLoc -> TypeParam -> TermTypeM (VName, Subst (TypeBase dim as))
instantiateTypeParam SrcLoc
loc TypeParam
tparam = do
  Int
i <- TermTypeM Int
incCounter
  VName
v <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID (Name -> TermTypeM VName) -> Name -> TermTypeM VName
forall a b. (a -> b) -> a -> b
$ String -> Int -> Name
mkTypeVarName ((Char -> Bool) -> ShowS
forall a. (a -> Bool) -> [a] -> [a]
takeWhile Char -> Bool
isAscii (VName -> String
baseString (TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName TypeParam
tparam))) Int
i
  case TypeParam
tparam of
    TypeParamType Liftedness
x VName
_ SrcLoc
_ -> do
      VName -> Constraint -> TermTypeM ()
constrain VName
v (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Liftedness -> Usage -> Constraint
NoConstraint Liftedness
x (Usage -> Constraint) -> Usage -> Constraint
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
mkUsage' SrcLoc
loc
      (VName, Subst (TypeBase dim as))
-> TermTypeM (VName, Subst (TypeBase dim as))
forall (m :: * -> *) a. Monad m => a -> m a
return (VName
v, [TypeParam] -> TypeBase dim as -> Subst (TypeBase dim as)
forall t. [TypeParam] -> t -> Subst t
Subst [] (TypeBase dim as -> Subst (TypeBase dim as))
-> TypeBase dim as -> Subst (TypeBase dim as)
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase dim as -> TypeBase dim as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim as -> TypeBase dim as)
-> ScalarTypeBase dim as -> TypeBase dim as
forall a b. (a -> b) -> a -> b
$ as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar as
forall a. Monoid a => a
mempty Uniqueness
Nonunique (VName -> TypeName
typeName VName
v) [])
    TypeParamDim {} -> do
      VName -> Constraint -> TermTypeM ()
constrain VName
v (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Maybe (DimDecl VName) -> Usage -> Constraint
Size Maybe (DimDecl VName)
forall a. Maybe a
Nothing (Usage -> Constraint) -> Usage -> Constraint
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
mkUsage' SrcLoc
loc
      (VName, Subst (TypeBase dim as))
-> TermTypeM (VName, Subst (TypeBase dim as))
forall (m :: * -> *) a. Monad m => a -> m a
return (VName
v, DimDecl VName -> Subst (TypeBase dim as)
forall t. DimDecl VName -> Subst t
SizeSubst (DimDecl VName -> Subst (TypeBase dim as))
-> DimDecl VName -> Subst (TypeBase dim as)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
v)

newArrayType :: SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType :: SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType SrcLoc
loc String
desc Int
r = do
  VName
v <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID (Name -> TermTypeM VName) -> Name -> TermTypeM VName
forall a b. (a -> b) -> a -> b
$ String -> Name
nameFromString String
desc
  VName -> Constraint -> TermTypeM ()
constrain VName
v (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Liftedness -> Usage -> Constraint
NoConstraint Liftedness
Unlifted (Usage -> Constraint) -> Usage -> Constraint
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
mkUsage' SrcLoc
loc
  [VName]
dims <- Int -> TermTypeM VName -> TermTypeM [VName]
forall (m :: * -> *) a. Applicative m => Int -> m a -> m [a]
replicateM Int
r (TermTypeM VName -> TermTypeM [VName])
-> TermTypeM VName -> TermTypeM [VName]
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Rigidity -> String -> TermTypeM VName
forall (m :: * -> *).
MonadUnify m =>
SrcLoc -> Rigidity -> String -> m VName
newDimVar SrcLoc
loc Rigidity
Nonrigid String
"dim"
  let rowt :: ScalarTypeBase dim ()
rowt = ()
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim ()
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar () Uniqueness
Nonunique (VName -> TypeName
typeName VName
v) []
  (StructType, StructType) -> TermTypeM (StructType, StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return
    ( ()
-> Uniqueness
-> ScalarTypeBase (DimDecl VName) ()
-> ShapeDecl (DimDecl VName)
-> StructType
forall dim as.
as
-> Uniqueness
-> ScalarTypeBase dim ()
-> ShapeDecl dim
-> TypeBase dim as
Array () Uniqueness
Nonunique ScalarTypeBase (DimDecl VName) ()
forall dim. ScalarTypeBase dim ()
rowt ([DimDecl VName] -> ShapeDecl (DimDecl VName)
forall dim. [dim] -> ShapeDecl dim
ShapeDecl ([DimDecl VName] -> ShapeDecl (DimDecl VName))
-> [DimDecl VName] -> ShapeDecl (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ (VName -> DimDecl VName) -> [VName] -> [DimDecl VName]
forall a b. (a -> b) -> [a] -> [b]
map (QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> (VName -> QualName VName) -> VName -> DimDecl VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> QualName VName
forall v. v -> QualName v
qualName) [VName]
dims),
      ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar ScalarTypeBase (DimDecl VName) ()
forall dim. ScalarTypeBase dim ()
rowt
    )

--- Errors

useAfterConsume :: VName -> SrcLoc -> SrcLoc -> TermTypeM a
useAfterConsume :: VName -> SrcLoc -> SrcLoc -> TermTypeM a
useAfterConsume VName
name SrcLoc
rloc SrcLoc
wloc = do
  Doc
name' <- SrcLoc -> VName -> TermTypeM Doc
describeVar SrcLoc
rloc VName
name
  SrcLoc -> Notes -> Doc -> TermTypeM a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
rloc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM a) -> (Doc -> Doc) -> Doc -> TermTypeM a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Doc -> Doc -> Doc
withIndexLink Doc
"use-after-consume" (Doc -> TermTypeM a) -> Doc -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
    Doc
"Using" Doc -> Doc -> Doc
<+> Doc
name' Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
", but this was consumed at"
      Doc -> Doc -> Doc
<+> String -> Doc
text (SrcLoc -> SrcLoc -> String
forall a b. (Located a, Located b) => a -> b -> String
locStrRel SrcLoc
rloc SrcLoc
wloc) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
".  (Possibly through aliasing.)"

badLetWithValue :: (Pretty arr, Pretty src) => arr -> src -> SrcLoc -> TermTypeM a
badLetWithValue :: arr -> src -> SrcLoc -> TermTypeM a
badLetWithValue arr
arre src
vale SrcLoc
loc =
  SrcLoc -> Notes -> Doc -> TermTypeM a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM a) -> Doc -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
    Doc
"Source array for in-place update"
      Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (arr -> Doc
forall a. Pretty a => a -> Doc
ppr arr
arre)
      Doc -> Doc -> Doc
</> Doc
"might alias update value"
      Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (src -> Doc
forall a. Pretty a => a -> Doc
ppr src
vale)
      Doc -> Doc -> Doc
</> Doc
"Hint: use" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote Doc
"copy" Doc -> Doc -> Doc
<+> Doc
"to remove aliases from the value."

returnAliased :: Name -> Name -> SrcLoc -> TermTypeM ()
returnAliased :: Name -> Name -> SrcLoc -> TermTypeM ()
returnAliased Name
fname Name
name SrcLoc
loc =
  SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> (Doc -> Doc) -> Doc -> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Doc -> Doc -> Doc
withIndexLink Doc
"return-aliased" (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    Doc
"Unique-typed return value of" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Name -> Doc
forall v. IsName v => v -> Doc
pprName Name
fname)
      Doc -> Doc -> Doc
<+> Doc
"is aliased to"
      Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Name -> Doc
forall v. IsName v => v -> Doc
pprName Name
name) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
", which is not consumable."

uniqueReturnAliased :: Name -> SrcLoc -> TermTypeM a
uniqueReturnAliased :: Name -> SrcLoc -> TermTypeM a
uniqueReturnAliased Name
fname SrcLoc
loc =
  SrcLoc -> Notes -> Doc -> TermTypeM a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM a) -> (Doc -> Doc) -> Doc -> TermTypeM a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Doc -> Doc -> Doc
withIndexLink Doc
"unique-return-aliased" (Doc -> TermTypeM a) -> Doc -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
    Doc
"A unique-typed component of the return value of"
      Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Name -> Doc
forall v. IsName v => v -> Doc
pprName Name
fname)
      Doc -> Doc -> Doc
<+> Doc
"is aliased to some other component."

unexpectedType :: MonadTypeChecker m => SrcLoc -> StructType -> [StructType] -> m a
unexpectedType :: SrcLoc -> StructType -> [StructType] -> m a
unexpectedType SrcLoc
loc StructType
_ [] =
  SrcLoc -> Notes -> Doc -> m a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m a) -> Doc -> m a
forall a b. (a -> b) -> a -> b
$
    Doc
"Type of expression at" Doc -> Doc -> Doc
<+> String -> Doc
text (SrcLoc -> String
forall a. Located a => a -> String
locStr SrcLoc
loc)
      Doc -> Doc -> Doc
<+> Doc
"cannot have any type - possibly a bug in the type checker."
unexpectedType SrcLoc
loc StructType
t [StructType]
ts =
  SrcLoc -> Notes -> Doc -> m a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m a) -> Doc -> m a
forall a b. (a -> b) -> a -> b
$
    Doc
"Type of expression at" Doc -> Doc -> Doc
<+> String -> Doc
text (SrcLoc -> String
forall a. Located a => a -> String
locStr SrcLoc
loc) Doc -> Doc -> Doc
<+> Doc
"must be one of"
      Doc -> Doc -> Doc
<+> [Doc] -> Doc
commasep ((StructType -> Doc) -> [StructType] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map StructType -> Doc
forall a. Pretty a => a -> Doc
ppr [StructType]
ts) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
", but is"
      Doc -> Doc -> Doc
<+> StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
t Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."

notConsumable :: MonadTypeChecker m => SrcLoc -> Doc -> m b
notConsumable :: SrcLoc -> Doc -> m b
notConsumable SrcLoc
loc Doc
v =
  SrcLoc -> Notes -> Doc -> m b
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m b) -> (Doc -> Doc) -> Doc -> m b
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Doc -> Doc -> Doc
withIndexLink Doc
"not-consumable" (Doc -> m b) -> Doc -> m b
forall a b. (a -> b) -> a -> b
$
    Doc
"Would consume" Doc -> Doc -> Doc
<+> Doc
v Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
", which is not consumable."

unusedSize :: (MonadTypeChecker m) => SizeBinder VName -> m a
unusedSize :: SizeBinder VName -> m a
unusedSize SizeBinder VName
p =
  SizeBinder VName -> Notes -> Doc -> m a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SizeBinder VName
p Notes
forall a. Monoid a => a
mempty (Doc -> m a) -> (Doc -> Doc) -> Doc -> m a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Doc -> Doc -> Doc
withIndexLink Doc
"unused-size" (Doc -> m a) -> Doc -> m a
forall a b. (a -> b) -> a -> b
$
    Doc
"Size" Doc -> Doc -> Doc
<+> SizeBinder VName -> Doc
forall a. Pretty a => a -> Doc
ppr SizeBinder VName
p Doc -> Doc -> Doc
<+> Doc
"unused in pattern."

--- Basic checking

-- | Determine if the two types of identical, ignoring uniqueness.
-- Mismatched dimensions are turned into fresh rigid type variables.
-- Causes a 'TypeError' if they fail to match, and otherwise returns
-- one of them.
unifyBranchTypes :: SrcLoc -> PatType -> PatType -> TermTypeM (PatType, [VName])
unifyBranchTypes :: SrcLoc -> PatType -> PatType -> TermTypeM (PatType, [VName])
unifyBranchTypes SrcLoc
loc PatType
t1 PatType
t2 =
  Checking
-> TermTypeM (PatType, [VName]) -> TermTypeM (PatType, [VName])
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (StructType -> StructType -> Checking
CheckingBranches (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t1) (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t2)) (TermTypeM (PatType, [VName]) -> TermTypeM (PatType, [VName]))
-> TermTypeM (PatType, [VName]) -> TermTypeM (PatType, [VName])
forall a b. (a -> b) -> a -> b
$
    Usage -> PatType -> PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *).
MonadUnify m =>
Usage -> PatType -> PatType -> m (PatType, [VName])
unifyMostCommon (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"unification of branch results") PatType
t1 PatType
t2

unifyBranches :: SrcLoc -> Exp -> Exp -> TermTypeM (PatType, [VName])
unifyBranches :: SrcLoc -> Exp -> Exp -> TermTypeM (PatType, [VName])
unifyBranches SrcLoc
loc Exp
e1 Exp
e2 = do
  PatType
e1_t <- Exp -> TermTypeM PatType
expTypeFully Exp
e1
  PatType
e2_t <- Exp -> TermTypeM PatType
expTypeFully Exp
e2
  SrcLoc -> PatType -> PatType -> TermTypeM (PatType, [VName])
unifyBranchTypes SrcLoc
loc PatType
e1_t PatType
e2_t

--- General binding.

doNotShadow :: [String]
doNotShadow :: [String]
doNotShadow = [String
"&&", String
"||"]

data InferredType
  = NoneInferred
  | Ascribed PatType

-- All this complexity is just so we can handle un-suffixed numeric
-- literals in patterns.
patLitMkType :: PatLit -> SrcLoc -> TermTypeM StructType
patLitMkType :: PatLit -> SrcLoc -> TermTypeM StructType
patLitMkType (PatLitInt Integer
_) SrcLoc
loc = do
  StructType
t <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  [PrimType] -> Usage -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
[PrimType] -> Usage -> StructType -> m ()
mustBeOneOf [PrimType]
anyNumberType (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"integer literal") StructType
t
  StructType -> TermTypeM StructType
forall (m :: * -> *) a. Monad m => a -> m a
return StructType
t
patLitMkType (PatLitFloat Double
_) SrcLoc
loc = do
  StructType
t <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  [PrimType] -> Usage -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
[PrimType] -> Usage -> StructType -> m ()
mustBeOneOf [PrimType]
anyFloatType (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"float literal") StructType
t
  StructType -> TermTypeM StructType
forall (m :: * -> *) a. Monad m => a -> m a
return StructType
t
patLitMkType (PatLitPrim PrimValue
v) SrcLoc
_ =
  StructType -> TermTypeM StructType
forall (f :: * -> *) a. Applicative f => a -> f a
pure (StructType -> TermTypeM StructType)
-> StructType -> TermTypeM StructType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) ())
-> PrimType -> ScalarTypeBase (DimDecl VName) ()
forall a b. (a -> b) -> a -> b
$ PrimValue -> PrimType
primValueType PrimValue
v

nonrigidFor :: [SizeBinder VName] -> StructType -> TermTypeM StructType
nonrigidFor :: [SizeBinder VName] -> StructType -> TermTypeM StructType
nonrigidFor [] StructType
t = StructType -> TermTypeM StructType
forall (f :: * -> *) a. Applicative f => a -> f a
pure StructType
t -- Minor optimisation.
nonrigidFor [SizeBinder VName]
sizes StructType
t = StateT [(VName, VName)] TermTypeM StructType
-> [(VName, VName)] -> TermTypeM StructType
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m a
evalStateT ((DimDecl VName
 -> StateT [(VName, VName)] TermTypeM (DimDecl VName))
-> (() -> StateT [(VName, VName)] TermTypeM ())
-> StructType
-> StateT [(VName, VName)] TermTypeM StructType
forall (t :: * -> * -> *) (f :: * -> *) a c b d.
(Bitraversable t, Applicative f) =>
(a -> f c) -> (b -> f d) -> t a b -> f (t c d)
bitraverse DimDecl VName -> StateT [(VName, VName)] TermTypeM (DimDecl VName)
forall (t :: (* -> *) -> * -> *).
(MonadState [(VName, VName)] (t TermTypeM), MonadTrans t) =>
DimDecl VName -> t TermTypeM (DimDecl VName)
onDim () -> StateT [(VName, VName)] TermTypeM ()
forall (f :: * -> *) a. Applicative f => a -> f a
pure StructType
t) [(VName, VName)]
forall a. Monoid a => a
mempty
  where
    onDim :: DimDecl VName -> t TermTypeM (DimDecl VName)
onDim (NamedDim (QualName [VName]
_ VName
v))
      | Just SizeBinder VName
size <- (SizeBinder VName -> Bool)
-> [SizeBinder VName] -> Maybe (SizeBinder VName)
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Maybe a
find ((VName -> VName -> Bool
forall a. Eq a => a -> a -> Bool
== VName
v) (VName -> Bool)
-> (SizeBinder VName -> VName) -> SizeBinder VName -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SizeBinder VName -> VName
forall vn. SizeBinder vn -> vn
sizeName) [SizeBinder VName]
sizes = do
        Maybe VName
prev <- ([(VName, VName)] -> Maybe VName) -> t TermTypeM (Maybe VName)
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets (([(VName, VName)] -> Maybe VName) -> t TermTypeM (Maybe VName))
-> ([(VName, VName)] -> Maybe VName) -> t TermTypeM (Maybe VName)
forall a b. (a -> b) -> a -> b
$ VName -> [(VName, VName)] -> Maybe VName
forall a b. Eq a => a -> [(a, b)] -> Maybe b
lookup VName
v
        case Maybe VName
prev of
          Maybe VName
Nothing -> do
            VName
v' <- TermTypeM VName -> t TermTypeM VName
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (TermTypeM VName -> t TermTypeM VName)
-> TermTypeM VName -> t TermTypeM VName
forall a b. (a -> b) -> a -> b
$ Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID (Name -> TermTypeM VName) -> Name -> TermTypeM VName
forall a b. (a -> b) -> a -> b
$ VName -> Name
baseName VName
v
            TermTypeM () -> t TermTypeM ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (TermTypeM () -> t TermTypeM ()) -> TermTypeM () -> t TermTypeM ()
forall a b. (a -> b) -> a -> b
$ VName -> Constraint -> TermTypeM ()
constrain VName
v' (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Maybe (DimDecl VName) -> Usage -> Constraint
Size Maybe (DimDecl VName)
forall a. Maybe a
Nothing (Usage -> Constraint) -> Usage -> Constraint
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Usage
mkUsage' (SrcLoc -> Usage) -> SrcLoc -> Usage
forall a b. (a -> b) -> a -> b
$ SizeBinder VName -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf SizeBinder VName
size
            ([(VName, VName)] -> [(VName, VName)]) -> t TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((VName
v, VName
v') (VName, VName) -> [(VName, VName)] -> [(VName, VName)]
forall a. a -> [a] -> [a]
:)
            DimDecl VName -> t TermTypeM (DimDecl VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (DimDecl VName -> t TermTypeM (DimDecl VName))
-> DimDecl VName -> t TermTypeM (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
v'
          Just VName
v' ->
            DimDecl VName -> t TermTypeM (DimDecl VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (DimDecl VName -> t TermTypeM (DimDecl VName))
-> DimDecl VName -> t TermTypeM (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
v'
    onDim DimDecl VName
d = DimDecl VName -> t TermTypeM (DimDecl VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure DimDecl VName
d

checkPat' ::
  [SizeBinder VName] ->
  UncheckedPat ->
  InferredType ->
  TermTypeM Pat
checkPat' :: [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes (PatParens UncheckedPat
p SrcLoc
loc) InferredType
t =
  Pat -> SrcLoc -> Pat
forall (f :: * -> *) vn. PatBase f vn -> SrcLoc -> PatBase f vn
PatParens (Pat -> SrcLoc -> Pat)
-> TermTypeM Pat -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p InferredType
t TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
checkPat' [SizeBinder VName]
_ (Id Name
name NoInfo PatType
_ SrcLoc
loc) InferredType
_
  | String
name' String -> [String] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [String]
doNotShadow =
    SrcLoc -> Notes -> Doc -> TermTypeM Pat
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM Pat) -> Doc -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ Doc
"The" Doc -> Doc -> Doc
<+> String -> Doc
text String
name' Doc -> Doc -> Doc
<+> Doc
"operator may not be redefined."
  where
    name' :: String
name' = Name -> String
nameToString Name
name
checkPat' [SizeBinder VName]
_ (Id Name
name NoInfo PatType
NoInfo SrcLoc
loc) (Ascribed PatType
t) = do
  VName
name' <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID Name
name
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. vn -> f PatType -> SrcLoc -> PatBase f vn
Id VName
name' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc
checkPat' [SizeBinder VName]
_ (Id Name
name NoInfo PatType
NoInfo SrcLoc
loc) InferredType
NoneInferred = do
  VName
name' <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID Name
name
  PatType
t <- SrcLoc -> String -> TermTypeM PatType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. vn -> f PatType -> SrcLoc -> PatBase f vn
Id VName
name' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc
checkPat' [SizeBinder VName]
_ (Wildcard NoInfo PatType
_ SrcLoc
loc) (Ascribed PatType
t) =
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. f PatType -> SrcLoc -> PatBase f vn
Wildcard (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ PatType
t PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique) SrcLoc
loc
checkPat' [SizeBinder VName]
_ (Wildcard NoInfo PatType
NoInfo SrcLoc
loc) InferredType
NoneInferred = do
  PatType
t <- SrcLoc -> String -> TermTypeM PatType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. f PatType -> SrcLoc -> PatBase f vn
Wildcard (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc
checkPat' [SizeBinder VName]
sizes (TuplePat [UncheckedPat]
ps SrcLoc
loc) (Ascribed PatType
t)
  | Just [PatType]
ts <- PatType -> Maybe [PatType]
forall dim as. TypeBase dim as -> Maybe [TypeBase dim as]
isTupleRecord PatType
t,
    [PatType] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [PatType]
ts Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== [UncheckedPat] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [UncheckedPat]
ps =
    [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn. [PatBase f vn] -> SrcLoc -> PatBase f vn
TuplePat
      ([Pat] -> SrcLoc -> Pat)
-> TermTypeM [Pat] -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (UncheckedPat -> InferredType -> TermTypeM Pat)
-> [UncheckedPat] -> [InferredType] -> TermTypeM [Pat]
forall (m :: * -> *) a b c.
Applicative m =>
(a -> b -> m c) -> [a] -> [b] -> m [c]
zipWithM ([SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes) [UncheckedPat]
ps ((PatType -> InferredType) -> [PatType] -> [InferredType]
forall a b. (a -> b) -> [a] -> [b]
map PatType -> InferredType
Ascribed [PatType]
ts)
      TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
checkPat' [SizeBinder VName]
sizes p :: UncheckedPat
p@(TuplePat [UncheckedPat]
ps SrcLoc
loc) (Ascribed PatType
t) = do
  [StructType]
ps_t <- Int -> TermTypeM StructType -> TermTypeM [StructType]
forall (m :: * -> *) a. Applicative m => Int -> m a -> m [a]
replicateM ([UncheckedPat] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [UncheckedPat]
ps) (SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t")
  Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"matching a tuple pattern") ([StructType] -> StructType
forall dim as. [TypeBase dim as] -> TypeBase dim as
tupleRecord [StructType]
ps_t) (StructType -> TermTypeM ()) -> StructType -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t
  PatType
t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
t
  [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p (InferredType -> TermTypeM Pat) -> InferredType -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ PatType -> InferredType
Ascribed PatType
t'
checkPat' [SizeBinder VName]
sizes (TuplePat [UncheckedPat]
ps SrcLoc
loc) InferredType
NoneInferred =
  [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn. [PatBase f vn] -> SrcLoc -> PatBase f vn
TuplePat ([Pat] -> SrcLoc -> Pat)
-> TermTypeM [Pat] -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (UncheckedPat -> TermTypeM Pat)
-> [UncheckedPat] -> TermTypeM [Pat]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM (\UncheckedPat
p -> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p InferredType
NoneInferred) [UncheckedPat]
ps TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
checkPat' [SizeBinder VName]
_ (RecordPat [(Name, UncheckedPat)]
p_fs SrcLoc
_) InferredType
_
  | Just (Name
f, UncheckedPat
fp) <- ((Name, UncheckedPat) -> Bool)
-> [(Name, UncheckedPat)] -> Maybe (Name, UncheckedPat)
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Maybe a
find ((String
"_" String -> String -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf`) (String -> Bool)
-> ((Name, UncheckedPat) -> String) -> (Name, UncheckedPat) -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Name -> String
nameToString (Name -> String)
-> ((Name, UncheckedPat) -> Name) -> (Name, UncheckedPat) -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Name, UncheckedPat) -> Name
forall a b. (a, b) -> a
fst) [(Name, UncheckedPat)]
p_fs =
    UncheckedPat -> Notes -> Doc -> TermTypeM Pat
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError UncheckedPat
fp Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM Pat) -> Doc -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$
      Doc
"Underscore-prefixed fields are not allowed."
        Doc -> Doc -> Doc
</> Doc
"Did you mean" Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc -> Doc
dquotes (String -> Doc
text (Int -> ShowS
forall a. Int -> [a] -> [a]
drop Int
1 (Name -> String
nameToString Name
f)) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"=_") Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"?"
checkPat' [SizeBinder VName]
sizes (RecordPat [(Name, UncheckedPat)]
p_fs SrcLoc
loc) (Ascribed (Scalar (Record Map Name PatType
t_fs)))
  | [Name] -> [Name]
forall a. Ord a => [a] -> [a]
sort (((Name, UncheckedPat) -> Name) -> [(Name, UncheckedPat)] -> [Name]
forall a b. (a -> b) -> [a] -> [b]
map (Name, UncheckedPat) -> Name
forall a b. (a, b) -> a
fst [(Name, UncheckedPat)]
p_fs) [Name] -> [Name] -> Bool
forall a. Eq a => a -> a -> Bool
== [Name] -> [Name]
forall a. Ord a => [a] -> [a]
sort (Map Name PatType -> [Name]
forall k a. Map k a -> [k]
M.keys Map Name PatType
t_fs) =
    [(Name, Pat)] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
[(Name, PatBase f vn)] -> SrcLoc -> PatBase f vn
RecordPat ([(Name, Pat)] -> SrcLoc -> Pat)
-> (Map Name Pat -> [(Name, Pat)]) -> Map Name Pat -> SrcLoc -> Pat
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Name Pat -> [(Name, Pat)]
forall k a. Map k a -> [(k, a)]
M.toList (Map Name Pat -> SrcLoc -> Pat)
-> TermTypeM (Map Name Pat) -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> TermTypeM (Map Name Pat)
check TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
  where
    check :: TermTypeM (Map Name Pat)
check =
      ((UncheckedPat, InferredType) -> TermTypeM Pat)
-> Map Name (UncheckedPat, InferredType)
-> TermTypeM (Map Name Pat)
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse ((UncheckedPat -> InferredType -> TermTypeM Pat)
-> (UncheckedPat, InferredType) -> TermTypeM Pat
forall a b c. (a -> b -> c) -> (a, b) -> c
uncurry ([SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes)) (Map Name (UncheckedPat, InferredType) -> TermTypeM (Map Name Pat))
-> Map Name (UncheckedPat, InferredType)
-> TermTypeM (Map Name Pat)
forall a b. (a -> b) -> a -> b
$
        (UncheckedPat -> InferredType -> (UncheckedPat, InferredType))
-> Map Name UncheckedPat
-> Map Name InferredType
-> Map Name (UncheckedPat, InferredType)
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
M.intersectionWith (,) ([(Name, UncheckedPat)] -> Map Name UncheckedPat
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList [(Name, UncheckedPat)]
p_fs) ((PatType -> InferredType)
-> Map Name PatType -> Map Name InferredType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap PatType -> InferredType
Ascribed Map Name PatType
t_fs)
checkPat' [SizeBinder VName]
sizes p :: UncheckedPat
p@(RecordPat [(Name, UncheckedPat)]
fields SrcLoc
loc) (Ascribed PatType
t) = do
  Map Name StructType
fields' <- (UncheckedPat -> TermTypeM StructType)
-> Map Name UncheckedPat -> TermTypeM (Map Name StructType)
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse (TermTypeM StructType -> UncheckedPat -> TermTypeM StructType
forall a b. a -> b -> a
const (TermTypeM StructType -> UncheckedPat -> TermTypeM StructType)
-> TermTypeM StructType -> UncheckedPat -> TermTypeM StructType
forall a b. (a -> b) -> a -> b
$ SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t") (Map Name UncheckedPat -> TermTypeM (Map Name StructType))
-> Map Name UncheckedPat -> TermTypeM (Map Name StructType)
forall a b. (a -> b) -> a -> b
$ [(Name, UncheckedPat)] -> Map Name UncheckedPat
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList [(Name, UncheckedPat)]
fields

  Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when ([Name] -> [Name]
forall a. Ord a => [a] -> [a]
sort (Map Name StructType -> [Name]
forall k a. Map k a -> [k]
M.keys Map Name StructType
fields') [Name] -> [Name] -> Bool
forall a. Eq a => a -> a -> Bool
/= [Name] -> [Name]
forall a. Ord a => [a] -> [a]
sort (((Name, UncheckedPat) -> Name) -> [(Name, UncheckedPat)] -> [Name]
forall a b. (a -> b) -> [a] -> [b]
map (Name, UncheckedPat) -> Name
forall a b. (a, b) -> a
fst [(Name, UncheckedPat)]
fields)) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Doc
"Duplicate fields in record pattern" Doc -> Doc -> Doc
<+> UncheckedPat -> Doc
forall a. Pretty a => a -> Doc
ppr UncheckedPat
p Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."

  Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"matching a record pattern") (ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (Map Name StructType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. Map Name (TypeBase dim as) -> ScalarTypeBase dim as
Record Map Name StructType
fields')) (StructType -> TermTypeM ()) -> StructType -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t
  PatType
t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
t
  [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p (InferredType -> TermTypeM Pat) -> InferredType -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ PatType -> InferredType
Ascribed PatType
t'
checkPat' [SizeBinder VName]
sizes (RecordPat [(Name, UncheckedPat)]
fs SrcLoc
loc) InferredType
NoneInferred =
  [(Name, Pat)] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
[(Name, PatBase f vn)] -> SrcLoc -> PatBase f vn
RecordPat ([(Name, Pat)] -> SrcLoc -> Pat)
-> (Map Name Pat -> [(Name, Pat)]) -> Map Name Pat -> SrcLoc -> Pat
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Name Pat -> [(Name, Pat)]
forall k a. Map k a -> [(k, a)]
M.toList
    (Map Name Pat -> SrcLoc -> Pat)
-> TermTypeM (Map Name Pat) -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (UncheckedPat -> TermTypeM Pat)
-> Map Name UncheckedPat -> TermTypeM (Map Name Pat)
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse (\UncheckedPat
p -> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p InferredType
NoneInferred) ([(Name, UncheckedPat)] -> Map Name UncheckedPat
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList [(Name, UncheckedPat)]
fs)
    TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
checkPat' [SizeBinder VName]
sizes (PatAscription UncheckedPat
p (TypeDecl TypeExp Name
t NoInfo StructType
NoInfo) SrcLoc
loc) InferredType
maybe_outer_t = do
  (TypeExp VName
t', StructType
st_nodims, Liftedness
_) <- TypeExp Name -> TermTypeM (TypeExp VName, StructType, Liftedness)
forall (m :: * -> *).
MonadTypeChecker m =>
TypeExp Name -> m (TypeExp VName, StructType, Liftedness)
checkTypeExp TypeExp Name
t
  (StructType
st, [VName]
_) <- SrcLoc
-> String
-> Rigidity
-> StructType
-> TermTypeM (StructType, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"impl" Rigidity
Nonrigid StructType
st_nodims

  let st' :: PatType
st' = StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
st
  case InferredType
maybe_outer_t of
    Ascribed PatType
outer_t -> do
      StructType
st_forunify <- [SizeBinder VName] -> StructType -> TermTypeM StructType
nonrigidFor [SizeBinder VName]
sizes StructType
st
      Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"explicit type ascription") StructType
st_forunify (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
outer_t)

      -- We also have to make sure that uniqueness matches.  This is
      -- done explicitly, because it is ignored by unification.
      PatType
st'' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
st'
      PatType
outer_t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
outer_t
      case (Uniqueness -> Uniqueness -> Maybe Uniqueness)
-> PatType -> PatType -> Maybe PatType
forall als dim.
(Monoid als, ArrayDim dim) =>
(Uniqueness -> Uniqueness -> Maybe Uniqueness)
-> TypeBase dim als -> TypeBase dim als -> Maybe (TypeBase dim als)
unifyTypesU Uniqueness -> Uniqueness -> Maybe Uniqueness
unifyUniqueness PatType
st'' PatType
outer_t' of
        Just PatType
outer_t'' ->
          Pat -> TypeDeclBase Info VName -> SrcLoc -> Pat
forall (f :: * -> *) vn.
PatBase f vn -> TypeDeclBase f vn -> SrcLoc -> PatBase f vn
PatAscription (Pat -> TypeDeclBase Info VName -> SrcLoc -> Pat)
-> TermTypeM Pat
-> TermTypeM (TypeDeclBase Info VName -> SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p (PatType -> InferredType
Ascribed PatType
outer_t'')
            TermTypeM (TypeDeclBase Info VName -> SrcLoc -> Pat)
-> TermTypeM (TypeDeclBase Info VName) -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> TypeDeclBase Info VName -> TermTypeM (TypeDeclBase Info VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (TypeExp VName -> Info StructType -> TypeDeclBase Info VName
forall (f :: * -> *) vn.
TypeExp vn -> f StructType -> TypeDeclBase f vn
TypeDecl TypeExp VName
t' (StructType -> Info StructType
forall a. a -> Info a
Info StructType
st))
            TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
        Maybe PatType
Nothing ->
          SrcLoc -> Notes -> Doc -> TermTypeM Pat
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM Pat) -> Doc -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$
            Doc
"Cannot match type" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
outer_t') Doc -> Doc -> Doc
<+> Doc
"with expected type"
              Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
st'') Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
    InferredType
NoneInferred ->
      Pat -> TypeDeclBase Info VName -> SrcLoc -> Pat
forall (f :: * -> *) vn.
PatBase f vn -> TypeDeclBase f vn -> SrcLoc -> PatBase f vn
PatAscription (Pat -> TypeDeclBase Info VName -> SrcLoc -> Pat)
-> TermTypeM Pat
-> TermTypeM (TypeDeclBase Info VName -> SrcLoc -> Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p (PatType -> InferredType
Ascribed PatType
st')
        TermTypeM (TypeDeclBase Info VName -> SrcLoc -> Pat)
-> TermTypeM (TypeDeclBase Info VName) -> TermTypeM (SrcLoc -> Pat)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> TypeDeclBase Info VName -> TermTypeM (TypeDeclBase Info VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (TypeExp VName -> Info StructType -> TypeDeclBase Info VName
forall (f :: * -> *) vn.
TypeExp vn -> f StructType -> TypeDeclBase f vn
TypeDecl TypeExp VName
t' (StructType -> Info StructType
forall a. a -> Info a
Info StructType
st))
        TermTypeM (SrcLoc -> Pat) -> TermTypeM SrcLoc -> TermTypeM Pat
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
  where
    unifyUniqueness :: Uniqueness -> Uniqueness -> Maybe Uniqueness
unifyUniqueness Uniqueness
u1 Uniqueness
u2 = if Uniqueness
u2 Uniqueness -> Uniqueness -> Bool
`subuniqueOf` Uniqueness
u1 then Uniqueness -> Maybe Uniqueness
forall a. a -> Maybe a
Just Uniqueness
u1 else Maybe Uniqueness
forall a. Maybe a
Nothing
checkPat' [SizeBinder VName]
_ (PatLit PatLit
l NoInfo PatType
NoInfo SrcLoc
loc) (Ascribed PatType
t) = do
  StructType
t' <- PatLit -> SrcLoc -> TermTypeM StructType
patLitMkType PatLit
l SrcLoc
loc
  Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"matching against literal") StructType
t' (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t)
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ PatLit -> Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn.
PatLit -> f PatType -> SrcLoc -> PatBase f vn
PatLit PatLit
l (PatType -> Info PatType
forall a. a -> Info a
Info (StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t')) SrcLoc
loc
checkPat' [SizeBinder VName]
_ (PatLit PatLit
l NoInfo PatType
NoInfo SrcLoc
loc) InferredType
NoneInferred = do
  StructType
t' <- PatLit -> SrcLoc -> TermTypeM StructType
patLitMkType PatLit
l SrcLoc
loc
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ PatLit -> Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn.
PatLit -> f PatType -> SrcLoc -> PatBase f vn
PatLit PatLit
l (PatType -> Info PatType
forall a. a -> Info a
Info (StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t')) SrcLoc
loc
checkPat' [SizeBinder VName]
sizes (PatConstr Name
n NoInfo PatType
NoInfo [UncheckedPat]
ps SrcLoc
loc) (Ascribed (Scalar (Sum Map Name [PatType]
cs)))
  | Just [PatType]
ts <- Name -> Map Name [PatType] -> Maybe [PatType]
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup Name
n Map Name [PatType]
cs = do
    [Pat]
ps' <- (UncheckedPat -> InferredType -> TermTypeM Pat)
-> [UncheckedPat] -> [InferredType] -> TermTypeM [Pat]
forall (m :: * -> *) a b c.
Applicative m =>
(a -> b -> m c) -> [a] -> [b] -> m [c]
zipWithM ([SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes) [UncheckedPat]
ps ([InferredType] -> TermTypeM [Pat])
-> [InferredType] -> TermTypeM [Pat]
forall a b. (a -> b) -> a -> b
$ (PatType -> InferredType) -> [PatType] -> [InferredType]
forall a b. (a -> b) -> [a] -> [b]
map PatType -> InferredType
Ascribed [PatType]
ts
    Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ Name -> Info PatType -> [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
Name -> f PatType -> [PatBase f vn] -> SrcLoc -> PatBase f vn
PatConstr Name
n (PatType -> Info PatType
forall a. a -> Info a
Info (ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (Map Name [PatType] -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. Map Name [TypeBase dim as] -> ScalarTypeBase dim as
Sum Map Name [PatType]
cs))) [Pat]
ps' SrcLoc
loc
checkPat' [SizeBinder VName]
sizes (PatConstr Name
n NoInfo PatType
NoInfo [UncheckedPat]
ps SrcLoc
loc) (Ascribed PatType
t) = do
  StructType
t' <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  [Pat]
ps' <- (UncheckedPat -> TermTypeM Pat)
-> [UncheckedPat] -> TermTypeM [Pat]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM (\UncheckedPat
p -> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p InferredType
NoneInferred) [UncheckedPat]
ps
  Usage -> Name -> StructType -> [StructType] -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> StructType -> [StructType] -> m ()
mustHaveConstr Usage
usage Name
n StructType
t' (Pat -> StructType
patternStructType (Pat -> StructType) -> [Pat] -> [StructType]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [Pat]
ps')
  Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify Usage
usage StructType
t' (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t)
  PatType
t'' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
t
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ Name -> Info PatType -> [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
Name -> f PatType -> [PatBase f vn] -> SrcLoc -> PatBase f vn
PatConstr Name
n (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t'') [Pat]
ps' SrcLoc
loc
  where
    usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"matching against constructor"
checkPat' [SizeBinder VName]
sizes (PatConstr Name
n NoInfo PatType
NoInfo [UncheckedPat]
ps SrcLoc
loc) InferredType
NoneInferred = do
  [Pat]
ps' <- (UncheckedPat -> TermTypeM Pat)
-> [UncheckedPat] -> TermTypeM [Pat]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM (\UncheckedPat
p -> [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p InferredType
NoneInferred) [UncheckedPat]
ps
  StructType
t <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  Usage -> Name -> StructType -> [StructType] -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> StructType -> [StructType] -> m ()
mustHaveConstr Usage
usage Name
n StructType
t (Pat -> StructType
patternStructType (Pat -> StructType) -> [Pat] -> [StructType]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [Pat]
ps')
  Pat -> TermTypeM Pat
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> TermTypeM Pat) -> Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ Name -> Info PatType -> [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
Name -> f PatType -> [PatBase f vn] -> SrcLoc -> PatBase f vn
PatConstr Name
n (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t) [Pat]
ps' SrcLoc
loc
  where
    usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"matching against constructor"

patternNameMap :: Pat -> NameMap
patternNameMap :: Pat -> NameMap
patternNameMap = [((Namespace, Name), QualName VName)] -> NameMap
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([((Namespace, Name), QualName VName)] -> NameMap)
-> (Pat -> [((Namespace, Name), QualName VName)]) -> Pat -> NameMap
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (VName -> ((Namespace, Name), QualName VName))
-> [VName] -> [((Namespace, Name), QualName VName)]
forall a b. (a -> b) -> [a] -> [b]
map VName -> ((Namespace, Name), QualName VName)
asTerm ([VName] -> [((Namespace, Name), QualName VName)])
-> (Pat -> [VName]) -> Pat -> [((Namespace, Name), QualName VName)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Names -> [VName]
forall a. Set a -> [a]
S.toList (Names -> [VName]) -> (Pat -> Names) -> Pat -> [VName]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames
  where
    asTerm :: VName -> ((Namespace, Name), QualName VName)
asTerm VName
v = ((Namespace
Term, VName -> Name
baseName VName
v), VName -> QualName VName
forall v. v -> QualName v
qualName VName
v)

checkPat ::
  [SizeBinder VName] ->
  UncheckedPat ->
  InferredType ->
  (Pat -> TermTypeM a) ->
  TermTypeM a
checkPat :: [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
checkPat [SizeBinder VName]
sizes UncheckedPat
p InferredType
t Pat -> TermTypeM a
m = do
  [UncheckedPat] -> TermTypeM ()
forall (m :: * -> *). MonadTypeChecker m => [UncheckedPat] -> m ()
checkForDuplicateNames [UncheckedPat
p]
  Pat
p' <- Checking -> TermTypeM Pat -> TermTypeM Pat
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (UncheckedPat -> InferredType -> Checking
CheckingPat UncheckedPat
p InferredType
t) (TermTypeM Pat -> TermTypeM Pat) -> TermTypeM Pat -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$ [SizeBinder VName] -> UncheckedPat -> InferredType -> TermTypeM Pat
checkPat' [SizeBinder VName]
sizes UncheckedPat
p InferredType
t

  let explicit :: Names
explicit = StructType -> Names
mustBeExplicitInType (StructType -> Names) -> StructType -> Names
forall a b. (a -> b) -> a -> b
$ Pat -> StructType
patternStructType Pat
p'

  case (SizeBinder VName -> Bool)
-> [SizeBinder VName] -> [SizeBinder VName]
forall a. (a -> Bool) -> [a] -> [a]
filter ((VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
explicit) (VName -> Bool)
-> (SizeBinder VName -> VName) -> SizeBinder VName -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SizeBinder VName -> VName
forall vn. SizeBinder vn -> vn
sizeName) [SizeBinder VName]
sizes of
    SizeBinder VName
size : [SizeBinder VName]
_ ->
      SizeBinder VName -> Notes -> Doc -> TermTypeM a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SizeBinder VName
size Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM a) -> Doc -> TermTypeM a
forall a b. (a -> b) -> a -> b
$
        Doc
"Cannot bind" Doc -> Doc -> Doc
<+> SizeBinder VName -> Doc
forall a. Pretty a => a -> Doc
ppr SizeBinder VName
size
          Doc -> Doc -> Doc
<+> Doc
"as it is never used as the size of a concrete (non-function) value."
    [] ->
      NameMap -> TermTypeM a -> TermTypeM a
forall (m :: * -> *) a. MonadTypeChecker m => NameMap -> m a -> m a
bindNameMap (Pat -> NameMap
patternNameMap Pat
p') (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ Pat -> TermTypeM a
m Pat
p'

binding :: [Ident] -> TermTypeM a -> TermTypeM a
binding :: [Ident] -> TermTypeM a -> TermTypeM a
binding [Ident]
stms = TermTypeM a -> TermTypeM a
forall b. TermTypeM b -> TermTypeM b
check (TermTypeM a -> TermTypeM a)
-> (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM a -> TermTypeM a
forall b. TermTypeM b -> TermTypeM b
handleVars
  where
    handleVars :: TermTypeM a -> TermTypeM a
handleVars TermTypeM a
m =
      (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope (TermScope -> [Ident] -> TermScope
`bindVars` [Ident]
stms) (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ do
        -- Those identifiers that can potentially also be sizes are
        -- added as type constraints.  This is necessary so that we
        -- can properly detect scope violations during unification.
        -- We do this for *all* identifiers, not just those that are
        -- integers, because they may become integers later due to
        -- inference...
        [Ident] -> (Ident -> TermTypeM ()) -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
t a -> (a -> m b) -> m ()
forM_ [Ident]
stms ((Ident -> TermTypeM ()) -> TermTypeM ())
-> (Ident -> TermTypeM ()) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \Ident
ident ->
          VName -> Constraint -> TermTypeM ()
constrain (Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
ident) (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Constraint
ParamSize (SrcLoc -> Constraint) -> SrcLoc -> Constraint
forall a b. (a -> b) -> a -> b
$ Ident -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Ident
ident
        TermTypeM a
m

    bindVars :: TermScope -> [Ident] -> TermScope
    bindVars :: TermScope -> [Ident] -> TermScope
bindVars = (TermScope -> Ident -> TermScope)
-> TermScope -> [Ident] -> TermScope
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl TermScope -> Ident -> TermScope
bindVar

    bindVar :: TermScope -> Ident -> TermScope
    bindVar :: TermScope -> Ident -> TermScope
bindVar TermScope
scope (Ident VName
name (Info PatType
tp) SrcLoc
_) =
      let inedges :: Names
inedges = Aliasing -> Names
boundAliases (Aliasing -> Names) -> Aliasing -> Names
forall a b. (a -> b) -> a -> b
$ PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
tp
          update :: ValBinding -> ValBinding
update (BoundV Locality
l [TypeParam]
tparams PatType
in_t)
            -- If 'name' is record or sum-typed, don't alias the
            -- components to 'name', because these no identity
            -- beyond their components.
            | Array {} <- PatType
tp = Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
l [TypeParam]
tparams (PatType
in_t PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
`addAliases` Alias -> Aliasing -> Aliasing
forall a. Ord a => a -> Set a -> Set a
S.insert (VName -> Alias
AliasBound VName
name))
            | Bool
otherwise = Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
l [TypeParam]
tparams PatType
in_t
          update ValBinding
b = ValBinding
b

          tp' :: PatType
tp' = PatType
tp PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
`addAliases` Alias -> Aliasing -> Aliasing
forall a. Ord a => a -> Set a -> Set a
S.insert (VName -> Alias
AliasBound VName
name)
       in TermScope
scope
            { scopeVtable :: Map VName ValBinding
scopeVtable =
                VName -> ValBinding -> Map VName ValBinding -> Map VName ValBinding
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
name (Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
Local [] PatType
tp') (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$
                  (ValBinding -> ValBinding)
-> Names -> Map VName ValBinding -> Map VName ValBinding
forall (t :: * -> *) k a.
(Foldable t, Ord k) =>
(a -> a) -> t k -> Map k a -> Map k a
adjustSeveral ValBinding -> ValBinding
update Names
inedges (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$
                    TermScope -> Map VName ValBinding
scopeVtable TermScope
scope
            }

    adjustSeveral :: (a -> a) -> t k -> Map k a -> Map k a
adjustSeveral a -> a
f = (Map k a -> t k -> Map k a) -> t k -> Map k a -> Map k a
forall a b c. (a -> b -> c) -> b -> a -> c
flip ((Map k a -> t k -> Map k a) -> t k -> Map k a -> Map k a)
-> (Map k a -> t k -> Map k a) -> t k -> Map k a -> Map k a
forall a b. (a -> b) -> a -> b
$ (Map k a -> k -> Map k a) -> Map k a -> t k -> Map k a
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl ((Map k a -> k -> Map k a) -> Map k a -> t k -> Map k a)
-> (Map k a -> k -> Map k a) -> Map k a -> t k -> Map k a
forall a b. (a -> b) -> a -> b
$ (k -> Map k a -> Map k a) -> Map k a -> k -> Map k a
forall a b c. (a -> b -> c) -> b -> a -> c
flip ((k -> Map k a -> Map k a) -> Map k a -> k -> Map k a)
-> (k -> Map k a -> Map k a) -> Map k a -> k -> Map k a
forall a b. (a -> b) -> a -> b
$ (a -> a) -> k -> Map k a -> Map k a
forall k a. Ord k => (a -> a) -> k -> Map k a -> Map k a
M.adjust a -> a
f

    -- Check whether the bound variables have been used correctly
    -- within their scope.
    check :: TermTypeM b -> TermTypeM b
check TermTypeM b
m = do
      (b
a, [Occurence]
usages) <- TermTypeM b -> TermTypeM (b, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectBindingsOccurences TermTypeM b
m
      [Occurence] -> TermTypeM ()
checkOccurences [Occurence]
usages

      (Ident -> TermTypeM ()) -> [Ident] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ ([Occurence] -> Ident -> TermTypeM ()
checkIfUsed [Occurence]
usages) [Ident]
stms

      b -> TermTypeM b
forall (m :: * -> *) a. Monad m => a -> m a
return b
a

    -- Collect and remove all occurences in @stms@.  This relies
    -- on the fact that no variables shadow any other.
    collectBindingsOccurences :: TermTypeM a -> TermTypeM (a, [Occurence])
collectBindingsOccurences TermTypeM a
m = do
      (a
x, [Occurence]
usage) <- TermTypeM a -> TermTypeM (a, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences TermTypeM a
m
      let ([Occurence]
relevant, [Occurence]
rest) = [Occurence] -> ([Occurence], [Occurence])
split [Occurence]
usage
      [Occurence] -> TermTypeM ()
occur [Occurence]
rest
      (a, [Occurence]) -> TermTypeM (a, [Occurence])
forall (f :: * -> *) a. Applicative f => a -> f a
pure (a
x, [Occurence]
relevant)
      where
        split :: [Occurence] -> ([Occurence], [Occurence])
split =
          [(Occurence, Occurence)] -> ([Occurence], [Occurence])
forall a b. [(a, b)] -> ([a], [b])
unzip
            ([(Occurence, Occurence)] -> ([Occurence], [Occurence]))
-> ([Occurence] -> [(Occurence, Occurence)])
-> [Occurence]
-> ([Occurence], [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Occurence -> (Occurence, Occurence))
-> [Occurence] -> [(Occurence, Occurence)]
forall a b. (a -> b) -> [a] -> [b]
map
              ( \Occurence
occ ->
                  let (Names
obs1, Names
obs2) = Names -> (Names, Names)
divide (Names -> (Names, Names)) -> Names -> (Names, Names)
forall a b. (a -> b) -> a -> b
$ Occurence -> Names
observed Occurence
occ
                      occ_cons :: Maybe (Names, Names)
occ_cons = Names -> (Names, Names)
divide (Names -> (Names, Names)) -> Maybe Names -> Maybe (Names, Names)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Occurence -> Maybe Names
consumed Occurence
occ
                      con1 :: Maybe Names
con1 = (Names, Names) -> Names
forall a b. (a, b) -> a
fst ((Names, Names) -> Names) -> Maybe (Names, Names) -> Maybe Names
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe (Names, Names)
occ_cons
                      con2 :: Maybe Names
con2 = (Names, Names) -> Names
forall a b. (a, b) -> b
snd ((Names, Names) -> Names) -> Maybe (Names, Names) -> Maybe Names
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe (Names, Names)
occ_cons
                   in ( Occurence
occ {observed :: Names
observed = Names
obs1, consumed :: Maybe Names
consumed = Maybe Names
con1},
                        Occurence
occ {observed :: Names
observed = Names
obs2, consumed :: Maybe Names
consumed = Maybe Names
con2}
                      )
              )
        names :: Names
names = [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList ([VName] -> Names) -> [VName] -> Names
forall a b. (a -> b) -> a -> b
$ (Ident -> VName) -> [Ident] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName [Ident]
stms
        divide :: Names -> (Names, Names)
divide Names
s = (Names
s Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` Names
names, Names
s Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` Names
names)

bindingTypes ::
  [Either (VName, TypeBinding) (VName, Constraint)] ->
  TermTypeM a ->
  TermTypeM a
bindingTypes :: [Either (VName, TypeBinding) (VName, Constraint)]
-> TermTypeM a -> TermTypeM a
bindingTypes [Either (VName, TypeBinding) (VName, Constraint)]
types TermTypeM a
m = do
  Int
lvl <- TermTypeM Int
forall (m :: * -> *). MonadUnify m => m Int
curLevel
  (Constraints -> Constraints) -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
(Constraints -> Constraints) -> m ()
modifyConstraints (Constraints -> Constraints -> Constraints
forall a. Semigroup a => a -> a -> a
<> (Constraint -> (Int, Constraint))
-> Map VName Constraint -> Constraints
forall a b k. (a -> b) -> Map k a -> Map k b
M.map (Int
lvl,) ([(VName, Constraint)] -> Map VName Constraint
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList [(VName, Constraint)]
constraints))
  (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope TermScope -> TermScope
extend TermTypeM a
m
  where
    ([(VName, TypeBinding)]
tbinds, [(VName, Constraint)]
constraints) = [Either (VName, TypeBinding) (VName, Constraint)]
-> ([(VName, TypeBinding)], [(VName, Constraint)])
forall a b. [Either a b] -> ([a], [b])
partitionEithers [Either (VName, TypeBinding) (VName, Constraint)]
types
    extend :: TermScope -> TermScope
extend TermScope
scope =
      TermScope
scope
        { scopeTypeTable :: Map VName TypeBinding
scopeTypeTable = [(VName, TypeBinding)] -> Map VName TypeBinding
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList [(VName, TypeBinding)]
tbinds Map VName TypeBinding
-> Map VName TypeBinding -> Map VName TypeBinding
forall a. Semigroup a => a -> a -> a
<> TermScope -> Map VName TypeBinding
scopeTypeTable TermScope
scope
        }

bindingTypeParams :: [TypeParam] -> TermTypeM a -> TermTypeM a
bindingTypeParams :: [TypeParam] -> TermTypeM a -> TermTypeM a
bindingTypeParams [TypeParam]
tparams =
  [Ident] -> TermTypeM a -> TermTypeM a
forall a. [Ident] -> TermTypeM a -> TermTypeM a
binding ((TypeParam -> Maybe Ident) -> [TypeParam] -> [Ident]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe TypeParam -> Maybe Ident
typeParamIdent [TypeParam]
tparams)
    (TermTypeM a -> TermTypeM a)
-> (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Either (VName, TypeBinding) (VName, Constraint)]
-> TermTypeM a -> TermTypeM a
forall a.
[Either (VName, TypeBinding) (VName, Constraint)]
-> TermTypeM a -> TermTypeM a
bindingTypes ((TypeParam -> [Either (VName, TypeBinding) (VName, Constraint)])
-> [TypeParam] -> [Either (VName, TypeBinding) (VName, Constraint)]
forall (t :: * -> *) a b. Foldable t => (a -> [b]) -> t a -> [b]
concatMap TypeParam -> [Either (VName, TypeBinding) (VName, Constraint)]
typeParamType [TypeParam]
tparams)
  where
    typeParamType :: TypeParam -> [Either (VName, TypeBinding) (VName, Constraint)]
typeParamType (TypeParamType Liftedness
l VName
v SrcLoc
loc) =
      [ (VName, TypeBinding)
-> Either (VName, TypeBinding) (VName, Constraint)
forall a b. a -> Either a b
Left (VName
v, Liftedness -> [TypeParam] -> StructType -> TypeBinding
TypeAbbr Liftedness
l [] (ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (()
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) ()
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar () Uniqueness
Nonunique (VName -> TypeName
typeName VName
v) []))),
        (VName, Constraint)
-> Either (VName, TypeBinding) (VName, Constraint)
forall a b. b -> Either a b
Right (VName
v, Liftedness -> SrcLoc -> Constraint
ParamType Liftedness
l SrcLoc
loc)
      ]
    typeParamType (TypeParamDim VName
v SrcLoc
loc) =
      [(VName, Constraint)
-> Either (VName, TypeBinding) (VName, Constraint)
forall a b. b -> Either a b
Right (VName
v, SrcLoc -> Constraint
ParamSize SrcLoc
loc)]

typeParamIdent :: TypeParam -> Maybe Ident
typeParamIdent :: TypeParam -> Maybe Ident
typeParamIdent (TypeParamDim VName
v SrcLoc
loc) =
  Ident -> Maybe Ident
forall a. a -> Maybe a
Just (Ident -> Maybe Ident) -> Ident -> Maybe Ident
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> Ident
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident VName
v (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int64) SrcLoc
loc
typeParamIdent TypeParam
_ = Maybe Ident
forall a. Maybe a
Nothing

bindingIdent ::
  IdentBase NoInfo Name ->
  PatType ->
  (Ident -> TermTypeM a) ->
  TermTypeM a
bindingIdent :: IdentBase NoInfo Name
-> PatType -> (Ident -> TermTypeM a) -> TermTypeM a
bindingIdent (Ident Name
v NoInfo PatType
NoInfo SrcLoc
vloc) PatType
t Ident -> TermTypeM a
m =
  [(Namespace, Name)] -> TermTypeM a -> TermTypeM a
forall (m :: * -> *) a.
MonadTypeChecker m =>
[(Namespace, Name)] -> m a -> m a
bindSpaced [(Namespace
Term, Name
v)] (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ do
    VName
v' <- Namespace -> Name -> SrcLoc -> TermTypeM VName
forall (m :: * -> *).
MonadTypeChecker m =>
Namespace -> Name -> SrcLoc -> m VName
checkName Namespace
Term Name
v SrcLoc
vloc
    let ident :: Ident
ident = VName -> Info PatType -> SrcLoc -> Ident
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident VName
v' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
vloc
    [Ident] -> TermTypeM a -> TermTypeM a
forall a. [Ident] -> TermTypeM a -> TermTypeM a
binding [Ident
ident] (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ Ident -> TermTypeM a
m Ident
ident

bindingParams ::
  [UncheckedTypeParam] ->
  [UncheckedPat] ->
  ([TypeParam] -> [Pat] -> TermTypeM a) ->
  TermTypeM a
bindingParams :: [UncheckedTypeParam]
-> [UncheckedPat]
-> ([TypeParam] -> [Pat] -> TermTypeM a)
-> TermTypeM a
bindingParams [UncheckedTypeParam]
tps [UncheckedPat]
orig_ps [TypeParam] -> [Pat] -> TermTypeM a
m = do
  [UncheckedPat] -> TermTypeM ()
forall (m :: * -> *). MonadTypeChecker m => [UncheckedPat] -> m ()
checkForDuplicateNames [UncheckedPat]
orig_ps
  [UncheckedTypeParam] -> ([TypeParam] -> TermTypeM a) -> TermTypeM a
forall (m :: * -> *) a.
MonadTypeChecker m =>
[UncheckedTypeParam] -> ([TypeParam] -> m a) -> m a
checkTypeParams [UncheckedTypeParam]
tps (([TypeParam] -> TermTypeM a) -> TermTypeM a)
-> ([TypeParam] -> TermTypeM a) -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \[TypeParam]
tps' -> [TypeParam] -> TermTypeM a -> TermTypeM a
forall a. [TypeParam] -> TermTypeM a -> TermTypeM a
bindingTypeParams [TypeParam]
tps' (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ do
    let descend :: [Pat] -> [UncheckedPat] -> TermTypeM a
descend [Pat]
ps' (UncheckedPat
p : [UncheckedPat]
ps) =
          [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
checkPat [] UncheckedPat
p InferredType
NoneInferred ((Pat -> TermTypeM a) -> TermTypeM a)
-> (Pat -> TermTypeM a) -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \Pat
p' ->
            [Ident] -> TermTypeM a -> TermTypeM a
forall a. [Ident] -> TermTypeM a -> TermTypeM a
binding (Set Ident -> [Ident]
forall a. Set a -> [a]
S.toList (Set Ident -> [Ident]) -> Set Ident -> [Ident]
forall a b. (a -> b) -> a -> b
$ Pat -> Set Ident
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set (IdentBase f vn)
patIdents Pat
p') (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ [Pat] -> [UncheckedPat] -> TermTypeM a
descend (Pat
p' Pat -> [Pat] -> [Pat]
forall a. a -> [a] -> [a]
: [Pat]
ps') [UncheckedPat]
ps
        descend [Pat]
ps' [] = do
          -- Perform an observation of every type parameter.  This
          -- prevents unused-name warnings for otherwise unused
          -- dimensions.
          (Ident -> TermTypeM ()) -> [Ident] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ Ident -> TermTypeM ()
observe ([Ident] -> TermTypeM ()) -> [Ident] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ (TypeParam -> Maybe Ident) -> [TypeParam] -> [Ident]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe TypeParam -> Maybe Ident
typeParamIdent [TypeParam]
tps'
          [TypeParam] -> [Pat] -> TermTypeM a
m [TypeParam]
tps' ([Pat] -> TermTypeM a) -> [Pat] -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ [Pat] -> [Pat]
forall a. [a] -> [a]
reverse [Pat]
ps'

    [Pat] -> [UncheckedPat] -> TermTypeM a
descend [] [UncheckedPat]
orig_ps

bindingSizes :: [SizeBinder Name] -> ([SizeBinder VName] -> TermTypeM a) -> TermTypeM a
bindingSizes :: [SizeBinder Name]
-> ([SizeBinder VName] -> TermTypeM a) -> TermTypeM a
bindingSizes [] [SizeBinder VName] -> TermTypeM a
m = [SizeBinder VName] -> TermTypeM a
m [] -- Minor optimisation.
bindingSizes [SizeBinder Name]
sizes [SizeBinder VName] -> TermTypeM a
m = do
  (Map Name SrcLoc -> SizeBinder Name -> TermTypeM (Map Name SrcLoc))
-> Map Name SrcLoc -> [SizeBinder Name] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m ()
foldM_ Map Name SrcLoc -> SizeBinder Name -> TermTypeM (Map Name SrcLoc)
forall k (m :: * -> *).
(Ord k, MonadTypeChecker m) =>
Map k SrcLoc -> SizeBinder k -> m (Map k SrcLoc)
lookForDuplicates Map Name SrcLoc
forall a. Monoid a => a
mempty [SizeBinder Name]
sizes
  [(Namespace, Name)] -> TermTypeM a -> TermTypeM a
forall (m :: * -> *) a.
MonadTypeChecker m =>
[(Namespace, Name)] -> m a -> m a
bindSpaced ((SizeBinder Name -> (Namespace, Name))
-> [SizeBinder Name] -> [(Namespace, Name)]
forall a b. (a -> b) -> [a] -> [b]
map SizeBinder Name -> (Namespace, Name)
forall b. SizeBinder b -> (Namespace, b)
sizeWithSpace [SizeBinder Name]
sizes) (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ do
    [SizeBinder VName]
sizes' <- (SizeBinder Name -> TermTypeM (SizeBinder VName))
-> [SizeBinder Name] -> TermTypeM [SizeBinder VName]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM SizeBinder Name -> TermTypeM (SizeBinder VName)
forall (f :: * -> *).
MonadTypeChecker f =>
SizeBinder Name -> f (SizeBinder VName)
check [SizeBinder Name]
sizes
    [Ident] -> TermTypeM a -> TermTypeM a
forall a. [Ident] -> TermTypeM a -> TermTypeM a
binding ((SizeBinder VName -> Ident) -> [SizeBinder VName] -> [Ident]
forall a b. (a -> b) -> [a] -> [b]
map SizeBinder VName -> Ident
forall vn. SizeBinder vn -> IdentBase Info vn
sizeWithType [SizeBinder VName]
sizes') (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ [SizeBinder VName] -> TermTypeM a
m [SizeBinder VName]
sizes'
  where
    lookForDuplicates :: Map k SrcLoc -> SizeBinder k -> m (Map k SrcLoc)
lookForDuplicates Map k SrcLoc
prev SizeBinder k
size
      | Just SrcLoc
prevloc <- k -> Map k SrcLoc -> Maybe SrcLoc
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup (SizeBinder k -> k
forall vn. SizeBinder vn -> vn
sizeName SizeBinder k
size) Map k SrcLoc
prev =
        SizeBinder k -> Notes -> Doc -> m (Map k SrcLoc)
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SizeBinder k
size Notes
forall a. Monoid a => a
mempty (Doc -> m (Map k SrcLoc)) -> Doc -> m (Map k SrcLoc)
forall a b. (a -> b) -> a -> b
$
          Doc
"Size name also bound at "
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> String -> Doc
text (SrcLoc -> SrcLoc -> String
forall a b. (Located a, Located b) => a -> b -> String
locStrRel (SizeBinder k -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf SizeBinder k
size) SrcLoc
prevloc)
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
      | Bool
otherwise =
        Map k SrcLoc -> m (Map k SrcLoc)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Map k SrcLoc -> m (Map k SrcLoc))
-> Map k SrcLoc -> m (Map k SrcLoc)
forall a b. (a -> b) -> a -> b
$ k -> SrcLoc -> Map k SrcLoc -> Map k SrcLoc
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert (SizeBinder k -> k
forall vn. SizeBinder vn -> vn
sizeName SizeBinder k
size) (SizeBinder k -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf SizeBinder k
size) Map k SrcLoc
prev

    sizeWithSpace :: SizeBinder b -> (Namespace, b)
sizeWithSpace SizeBinder b
size =
      (Namespace
Term, SizeBinder b -> b
forall vn. SizeBinder vn -> vn
sizeName SizeBinder b
size)
    sizeWithType :: SizeBinder vn -> IdentBase Info vn
sizeWithType SizeBinder vn
size =
      vn -> Info PatType -> SrcLoc -> IdentBase Info vn
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident (SizeBinder vn -> vn
forall vn. SizeBinder vn -> vn
sizeName SizeBinder vn
size) (PatType -> Info PatType
forall a. a -> Info a
Info (ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (IntType -> PrimType
Signed IntType
Int64)))) (SizeBinder vn -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf SizeBinder vn
size)

    check :: SizeBinder Name -> f (SizeBinder VName)
check (SizeBinder Name
v SrcLoc
loc) =
      VName -> SrcLoc -> SizeBinder VName
forall vn. vn -> SrcLoc -> SizeBinder vn
SizeBinder (VName -> SrcLoc -> SizeBinder VName)
-> f VName -> f (SrcLoc -> SizeBinder VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Namespace -> Name -> SrcLoc -> f VName
forall (m :: * -> *).
MonadTypeChecker m =>
Namespace -> Name -> SrcLoc -> m VName
checkName Namespace
Term Name
v SrcLoc
loc f (SrcLoc -> SizeBinder VName) -> f SrcLoc -> f (SizeBinder VName)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> f SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc

bindingPat ::
  [SizeBinder VName] ->
  PatBase NoInfo Name ->
  InferredType ->
  (Pat -> TermTypeM a) ->
  TermTypeM a
bindingPat :: [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [SizeBinder VName]
sizes UncheckedPat
p InferredType
t Pat -> TermTypeM a
m = do
  [UncheckedPat] -> TermTypeM ()
forall (m :: * -> *). MonadTypeChecker m => [UncheckedPat] -> m ()
checkForDuplicateNames [UncheckedPat
p]
  [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
checkPat [SizeBinder VName]
sizes UncheckedPat
p InferredType
t ((Pat -> TermTypeM a) -> TermTypeM a)
-> (Pat -> TermTypeM a) -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ \Pat
p' -> [Ident] -> TermTypeM a -> TermTypeM a
forall a. [Ident] -> TermTypeM a -> TermTypeM a
binding (Set Ident -> [Ident]
forall a. Set a -> [a]
S.toList (Set Ident -> [Ident]) -> Set Ident -> [Ident]
forall a b. (a -> b) -> a -> b
$ Pat -> Set Ident
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set (IdentBase f vn)
patIdents Pat
p') (TermTypeM a -> TermTypeM a) -> TermTypeM a -> TermTypeM a
forall a b. (a -> b) -> a -> b
$ do
    -- Perform an observation of every declared dimension.  This
    -- prevents unused-name warnings for otherwise unused dimensions.
    (Ident -> TermTypeM ()) -> [Ident] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ Ident -> TermTypeM ()
observe ([Ident] -> TermTypeM ()) -> [Ident] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Pat -> [Ident]
patternDims Pat
p'

    let used_sizes :: Names
used_sizes = StructType -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNames (StructType -> Names) -> StructType -> Names
forall a b. (a -> b) -> a -> b
$ Pat -> StructType
patternStructType Pat
p'
    case (SizeBinder VName -> Bool)
-> [SizeBinder VName] -> [SizeBinder VName]
forall a. (a -> Bool) -> [a] -> [a]
filter ((VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.notMember` Names
used_sizes) (VName -> Bool)
-> (SizeBinder VName -> VName) -> SizeBinder VName -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SizeBinder VName -> VName
forall vn. SizeBinder vn -> vn
sizeName) [SizeBinder VName]
sizes of
      [] -> Pat -> TermTypeM a
m Pat
p'
      SizeBinder VName
size : [SizeBinder VName]
_ -> SizeBinder VName -> TermTypeM a
forall (m :: * -> *) a.
MonadTypeChecker m =>
SizeBinder VName -> m a
unusedSize SizeBinder VName
size

patternDims :: Pat -> [Ident]
patternDims :: Pat -> [Ident]
patternDims (PatParens Pat
p SrcLoc
_) = Pat -> [Ident]
patternDims Pat
p
patternDims (TuplePat [Pat]
pats SrcLoc
_) = (Pat -> [Ident]) -> [Pat] -> [Ident]
forall (t :: * -> *) a b. Foldable t => (a -> [b]) -> t a -> [b]
concatMap Pat -> [Ident]
patternDims [Pat]
pats
patternDims (PatAscription Pat
p (TypeDecl TypeExp VName
_ (Info StructType
t)) SrcLoc
_) =
  Pat -> [Ident]
patternDims Pat
p [Ident] -> [Ident] -> [Ident]
forall a. Semigroup a => a -> a -> a
<> (DimDecl VName -> Maybe Ident) -> [DimDecl VName] -> [Ident]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe (SrcLoc -> DimDecl VName -> Maybe Ident
forall p vn a. p -> DimDecl vn -> Maybe a
dimIdent (Pat -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Pat
p)) (StructType -> [DimDecl VName]
forall as. TypeBase (DimDecl VName) as -> [DimDecl VName]
nestedDims StructType
t)
  where
    dimIdent :: p -> DimDecl vn -> Maybe a
dimIdent p
_ (AnyDim Maybe vn
_) = Maybe a
forall a. Maybe a
Nothing
    dimIdent p
_ (ConstDim Int
_) = Maybe a
forall a. Maybe a
Nothing
    dimIdent p
_ NamedDim {} = Maybe a
forall a. Maybe a
Nothing
patternDims Pat
_ = []

sliceShape ::
  Maybe (SrcLoc, Rigidity) ->
  Slice ->
  TypeBase (DimDecl VName) as ->
  TermTypeM (TypeBase (DimDecl VName) as, [VName])
sliceShape :: Maybe (SrcLoc, Rigidity)
-> Slice
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
sliceShape Maybe (SrcLoc, Rigidity)
r Slice
slice t :: TypeBase (DimDecl VName) as
t@(Array as
als Uniqueness
u ScalarTypeBase (DimDecl VName) ()
et (ShapeDecl [DimDecl VName]
orig_dims)) =
  StateT [VName] TermTypeM (TypeBase (DimDecl VName) as)
-> [VName] -> TermTypeM (TypeBase (DimDecl VName) as, [VName])
forall s (m :: * -> *) a. StateT s m a -> s -> m (a, s)
runStateT ([DimDecl VName] -> TypeBase (DimDecl VName) as
setDims ([DimDecl VName] -> TypeBase (DimDecl VName) as)
-> StateT [VName] TermTypeM [DimDecl VName]
-> StateT [VName] TermTypeM (TypeBase (DimDecl VName) as)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Slice
-> [DimDecl VName] -> StateT [VName] TermTypeM [DimDecl VName]
forall (t :: (* -> *) -> * -> *).
(MonadTrans t, MonadState [VName] (t TermTypeM)) =>
Slice -> [DimDecl VName] -> t TermTypeM [DimDecl VName]
adjustDims Slice
slice [DimDecl VName]
orig_dims) []
  where
    setDims :: [DimDecl VName] -> TypeBase (DimDecl VName) as
setDims [] = Int -> TypeBase (DimDecl VName) as -> TypeBase (DimDecl VName) as
forall dim as. Int -> TypeBase dim as -> TypeBase dim as
stripArray ([DimDecl VName] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [DimDecl VName]
orig_dims) TypeBase (DimDecl VName) as
t
    setDims [DimDecl VName]
dims' = as
-> Uniqueness
-> ScalarTypeBase (DimDecl VName) ()
-> ShapeDecl (DimDecl VName)
-> TypeBase (DimDecl VName) as
forall dim as.
as
-> Uniqueness
-> ScalarTypeBase dim ()
-> ShapeDecl dim
-> TypeBase dim as
Array as
als Uniqueness
u ScalarTypeBase (DimDecl VName) ()
et (ShapeDecl (DimDecl VName) -> TypeBase (DimDecl VName) as)
-> ShapeDecl (DimDecl VName) -> TypeBase (DimDecl VName) as
forall a b. (a -> b) -> a -> b
$ [DimDecl VName] -> ShapeDecl (DimDecl VName)
forall dim. [dim] -> ShapeDecl dim
ShapeDecl [DimDecl VName]
dims'

    -- If the result is supposed to be AnyDim or a nonrigid size
    -- variable, then don't bother trying to create
    -- non-existential sizes.  This is necessary to make programs
    -- type-check without too much ceremony; see
    -- e.g. tests/inplace5.fut.
    isRigid :: Rigidity -> Bool
isRigid Rigid {} = Bool
True
    isRigid Rigidity
_ = Bool
False
    refine_sizes :: Bool
refine_sizes = Bool
-> ((SrcLoc, Rigidity) -> Bool) -> Maybe (SrcLoc, Rigidity) -> Bool
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Bool
False (Rigidity -> Bool
isRigid (Rigidity -> Bool)
-> ((SrcLoc, Rigidity) -> Rigidity) -> (SrcLoc, Rigidity) -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (SrcLoc, Rigidity) -> Rigidity
forall a b. (a, b) -> b
snd) Maybe (SrcLoc, Rigidity)
r

    sliceSize :: DimDecl VName
-> Maybe Exp
-> Maybe Exp
-> Maybe Exp
-> t TermTypeM (DimDecl VName)
sliceSize DimDecl VName
orig_d Maybe Exp
i Maybe Exp
j Maybe Exp
stride =
      case Maybe (SrcLoc, Rigidity)
r of
        Just (SrcLoc
loc, Rigid RigidSource
_) -> do
          (DimDecl VName
d, Maybe VName
ext) <-
            TermTypeM (DimDecl VName, Maybe VName)
-> t TermTypeM (DimDecl VName, Maybe VName)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (TermTypeM (DimDecl VName, Maybe VName)
 -> t TermTypeM (DimDecl VName, Maybe VName))
-> TermTypeM (DimDecl VName, Maybe VName)
-> t TermTypeM (DimDecl VName, Maybe VName)
forall a b. (a -> b) -> a -> b
$
              SrcLoc -> SizeSource -> TermTypeM (DimDecl VName, Maybe VName)
extSize SrcLoc
loc (SizeSource -> TermTypeM (DimDecl VName, Maybe VName))
-> SizeSource -> TermTypeM (DimDecl VName, Maybe VName)
forall a b. (a -> b) -> a -> b
$
                Maybe (DimDecl VName)
-> Maybe (ExpBase NoInfo VName)
-> Maybe (ExpBase NoInfo VName)
-> Maybe (ExpBase NoInfo VName)
-> SizeSource
SourceSlice Maybe (DimDecl VName)
orig_d' (Exp -> ExpBase NoInfo VName
bareExp (Exp -> ExpBase NoInfo VName)
-> Maybe Exp -> Maybe (ExpBase NoInfo VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe Exp
i) (Exp -> ExpBase NoInfo VName
bareExp (Exp -> ExpBase NoInfo VName)
-> Maybe Exp -> Maybe (ExpBase NoInfo VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe Exp
j) (Exp -> ExpBase NoInfo VName
bareExp (Exp -> ExpBase NoInfo VName)
-> Maybe Exp -> Maybe (ExpBase NoInfo VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe Exp
stride)
          ([VName] -> [VName]) -> t TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify (Maybe VName -> [VName]
forall a. Maybe a -> [a]
maybeToList Maybe VName
ext [VName] -> [VName] -> [VName]
forall a. [a] -> [a] -> [a]
++)
          DimDecl VName -> t TermTypeM (DimDecl VName)
forall (m :: * -> *) a. Monad m => a -> m a
return DimDecl VName
d
        Just (SrcLoc
loc, Rigidity
Nonrigid) ->
          TermTypeM (DimDecl VName) -> t TermTypeM (DimDecl VName)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (TermTypeM (DimDecl VName) -> t TermTypeM (DimDecl VName))
-> TermTypeM (DimDecl VName) -> t TermTypeM (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> (VName -> QualName VName) -> VName -> DimDecl VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> QualName VName
forall v. v -> QualName v
qualName (VName -> DimDecl VName)
-> TermTypeM VName -> TermTypeM (DimDecl VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> SrcLoc -> Rigidity -> String -> TermTypeM VName
forall (m :: * -> *).
MonadUnify m =>
SrcLoc -> Rigidity -> String -> m VName
newDimVar SrcLoc
loc Rigidity
Nonrigid String
"slice_dim"
        Maybe (SrcLoc, Rigidity)
Nothing ->
          DimDecl VName -> t TermTypeM (DimDecl VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (DimDecl VName -> t TermTypeM (DimDecl VName))
-> DimDecl VName -> t TermTypeM (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim Maybe VName
forall a. Maybe a
Nothing
      where
        -- The original size does not matter if the slice is fully specified.
        orig_d' :: Maybe (DimDecl VName)
orig_d'
          | Maybe Exp -> Bool
forall a. Maybe a -> Bool
isJust Maybe Exp
i, Maybe Exp -> Bool
forall a. Maybe a -> Bool
isJust Maybe Exp
j = Maybe (DimDecl VName)
forall a. Maybe a
Nothing
          | Bool
otherwise = DimDecl VName -> Maybe (DimDecl VName)
forall a. a -> Maybe a
Just DimDecl VName
orig_d

    adjustDims :: Slice -> [DimDecl VName] -> t TermTypeM [DimDecl VName]
adjustDims (DimFix {} : Slice
idxes') (DimDecl VName
_ : [DimDecl VName]
dims) =
      Slice -> [DimDecl VName] -> t TermTypeM [DimDecl VName]
adjustDims Slice
idxes' [DimDecl VName]
dims
    -- Pat match some known slices to be non-existential.
    adjustDims (DimSlice Maybe Exp
i Maybe Exp
j Maybe Exp
stride : Slice
idxes') (DimDecl VName
_ : [DimDecl VName]
dims)
      | Bool
refine_sizes,
        Bool -> (Exp -> Bool) -> Maybe Exp -> Bool
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Bool
True ((Maybe Int64 -> Maybe Int64 -> Bool
forall a. Eq a => a -> a -> Bool
== Int64 -> Maybe Int64
forall a. a -> Maybe a
Just Int64
0) (Maybe Int64 -> Bool) -> (Exp -> Maybe Int64) -> Exp -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> Maybe Int64
isInt64) Maybe Exp
i,
        Just DimDecl VName
j' <- Exp -> Maybe (DimDecl VName)
maybeDimFromExp (Exp -> Maybe (DimDecl VName))
-> Maybe Exp -> Maybe (DimDecl VName)
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Maybe Exp
j,
        Bool -> (Exp -> Bool) -> Maybe Exp -> Bool
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Bool
True ((Maybe Int64 -> Maybe Int64 -> Bool
forall a. Eq a => a -> a -> Bool
== Int64 -> Maybe Int64
forall a. a -> Maybe a
Just Int64
1) (Maybe Int64 -> Bool) -> (Exp -> Maybe Int64) -> Exp -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> Maybe Int64
isInt64) Maybe Exp
stride =
        (DimDecl VName
j' DimDecl VName -> [DimDecl VName] -> [DimDecl VName]
forall a. a -> [a] -> [a]
:) ([DimDecl VName] -> [DimDecl VName])
-> t TermTypeM [DimDecl VName] -> t TermTypeM [DimDecl VName]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Slice -> [DimDecl VName] -> t TermTypeM [DimDecl VName]
adjustDims Slice
idxes' [DimDecl VName]
dims
    adjustDims (DimSlice Maybe Exp
Nothing Maybe Exp
Nothing Maybe Exp
stride : Slice
idxes') (DimDecl VName
d : [DimDecl VName]
dims)
      | Bool
refine_sizes,
        Bool -> (Exp -> Bool) -> Maybe Exp -> Bool
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Bool
True (Bool -> (Int64 -> Bool) -> Maybe Int64 -> Bool
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Bool
False ((Int64 -> Int64 -> Bool
forall a. Eq a => a -> a -> Bool
== Int64
1) (Int64 -> Bool) -> (Int64 -> Int64) -> Int64 -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int64 -> Int64
forall a. Num a => a -> a
abs) (Maybe Int64 -> Bool) -> (Exp -> Maybe Int64) -> Exp -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> Maybe Int64
isInt64) Maybe Exp
stride =
        (DimDecl VName
d DimDecl VName -> [DimDecl VName] -> [DimDecl VName]
forall a. a -> [a] -> [a]
:) ([DimDecl VName] -> [DimDecl VName])
-> t TermTypeM [DimDecl VName] -> t TermTypeM [DimDecl VName]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Slice -> [DimDecl VName] -> t TermTypeM [DimDecl VName]
adjustDims Slice
idxes' [DimDecl VName]
dims
    adjustDims (DimSlice Maybe Exp
i Maybe Exp
j Maybe Exp
stride : Slice
idxes') (DimDecl VName
d : [DimDecl VName]
dims) =
      (:) (DimDecl VName -> [DimDecl VName] -> [DimDecl VName])
-> t TermTypeM (DimDecl VName)
-> t TermTypeM ([DimDecl VName] -> [DimDecl VName])
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> DimDecl VName
-> Maybe Exp
-> Maybe Exp
-> Maybe Exp
-> t TermTypeM (DimDecl VName)
forall (t :: (* -> *) -> * -> *).
(MonadTrans t, MonadState [VName] (t TermTypeM)) =>
DimDecl VName
-> Maybe Exp
-> Maybe Exp
-> Maybe Exp
-> t TermTypeM (DimDecl VName)
sliceSize DimDecl VName
d Maybe Exp
i Maybe Exp
j Maybe Exp
stride t TermTypeM ([DimDecl VName] -> [DimDecl VName])
-> t TermTypeM [DimDecl VName] -> t TermTypeM [DimDecl VName]
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> Slice -> [DimDecl VName] -> t TermTypeM [DimDecl VName]
adjustDims Slice
idxes' [DimDecl VName]
dims
    adjustDims Slice
_ [DimDecl VName]
dims =
      [DimDecl VName] -> t TermTypeM [DimDecl VName]
forall (f :: * -> *) a. Applicative f => a -> f a
pure [DimDecl VName]
dims
sliceShape Maybe (SrcLoc, Rigidity)
_ Slice
_ TypeBase (DimDecl VName) as
t = (TypeBase (DimDecl VName) as, [VName])
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
forall (f :: * -> *) a. Applicative f => a -> f a
pure (TypeBase (DimDecl VName) as
t, [])

--- Main checkers

-- | @require ts e@ causes a 'TypeError' if @expType e@ is not one of
-- the types in @ts@.  Otherwise, simply returns @e@.
require :: String -> [PrimType] -> Exp -> TermTypeM Exp
require :: String -> [PrimType] -> Exp -> TermTypeM Exp
require String
why [PrimType]
ts Exp
e = do
  [PrimType] -> Usage -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
[PrimType] -> Usage -> StructType -> m ()
mustBeOneOf [PrimType]
ts (SrcLoc -> String -> Usage
mkUsage (Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
e) String
why) (StructType -> TermTypeM ())
-> (PatType -> StructType) -> PatType -> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> TermTypeM ()) -> TermTypeM PatType -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expType Exp
e
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e

unifies :: String -> StructType -> Exp -> TermTypeM Exp
unifies :: String -> StructType -> Exp -> TermTypeM Exp
unifies String
why StructType
t Exp
e = do
  Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage (Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
e) String
why) StructType
t (StructType -> TermTypeM ())
-> (PatType -> StructType) -> PatType -> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> TermTypeM ()) -> TermTypeM PatType -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expType Exp
e
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e

-- The closure of a lambda or local function are those variables that
-- it references, and which local to the current top-level function.
lexicalClosure :: [Pat] -> Occurences -> TermTypeM Aliasing
lexicalClosure :: [Pat] -> [Occurence] -> TermTypeM Aliasing
lexicalClosure [Pat]
params [Occurence]
closure = do
  Map VName ValBinding
vtable <- (TermEnv -> Map VName ValBinding)
-> TermTypeM (Map VName ValBinding)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((TermEnv -> Map VName ValBinding)
 -> TermTypeM (Map VName ValBinding))
-> (TermEnv -> Map VName ValBinding)
-> TermTypeM (Map VName ValBinding)
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable (TermScope -> Map VName ValBinding)
-> (TermEnv -> TermScope) -> TermEnv -> Map VName ValBinding
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermEnv -> TermScope
termScope
  let isLocal :: VName -> Bool
isLocal VName
v = case VName
v VName -> Map VName ValBinding -> Maybe ValBinding
forall k a. Ord k => k -> Map k a -> Maybe a
`M.lookup` Map VName ValBinding
vtable of
        Just (BoundV Locality
Local [TypeParam]
_ PatType
_) -> Bool
True
        Maybe ValBinding
_ -> Bool
False
  Aliasing -> TermTypeM Aliasing
forall (m :: * -> *) a. Monad m => a -> m a
return (Aliasing -> TermTypeM Aliasing) -> Aliasing -> TermTypeM Aliasing
forall a b. (a -> b) -> a -> b
$
    (VName -> Alias) -> Names -> Aliasing
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map VName -> Alias
AliasBound (Names -> Aliasing) -> Names -> Aliasing
forall a b. (a -> b) -> a -> b
$
      (VName -> Bool) -> Names -> Names
forall a. (a -> Bool) -> Set a -> Set a
S.filter VName -> Bool
isLocal (Names -> Names) -> Names -> Names
forall a b. (a -> b) -> a -> b
$
        [Occurence] -> Names
allOccuring [Occurence]
closure Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
S.\\ [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ((Pat -> Names) -> [Pat] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames [Pat]
params)

noAliasesIfOverloaded :: PatType -> TermTypeM PatType
noAliasesIfOverloaded :: PatType -> TermTypeM PatType
noAliasesIfOverloaded t :: PatType
t@(Scalar (TypeVar Aliasing
_ Uniqueness
u TypeName
tn [])) = do
  Maybe Constraint
subst <- ((Int, Constraint) -> Constraint)
-> Maybe (Int, Constraint) -> Maybe Constraint
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Int, Constraint) -> Constraint
forall a b. (a, b) -> b
snd (Maybe (Int, Constraint) -> Maybe Constraint)
-> (Constraints -> Maybe (Int, Constraint))
-> Constraints
-> Maybe Constraint
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> Constraints -> Maybe (Int, Constraint)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup (TypeName -> VName
typeLeaf TypeName
tn) (Constraints -> Maybe Constraint)
-> TermTypeM Constraints -> TermTypeM (Maybe Constraint)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> TermTypeM Constraints
forall (m :: * -> *). MonadUnify m => m Constraints
getConstraints
  case Maybe Constraint
subst of
    Just Overloaded {} -> PatType -> TermTypeM PatType
forall (m :: * -> *) a. Monad m => a -> m a
return (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar Aliasing
forall a. Monoid a => a
mempty Uniqueness
u TypeName
tn []
    Maybe Constraint
_ -> PatType -> TermTypeM PatType
forall (m :: * -> *) a. Monad m => a -> m a
return PatType
t
noAliasesIfOverloaded PatType
t =
  PatType -> TermTypeM PatType
forall (m :: * -> *) a. Monad m => a -> m a
return PatType
t

-- Check the common parts of ascription and coercion.
checkAscript ::
  SrcLoc ->
  UncheckedTypeDecl ->
  UncheckedExp ->
  (StructType -> TermTypeM StructType) ->
  TermTypeM (TypeDecl, Exp)
checkAscript :: SrcLoc
-> TypeDeclBase NoInfo Name
-> UncheckedExp
-> (StructType -> TermTypeM StructType)
-> TermTypeM (TypeDeclBase Info VName, Exp)
checkAscript SrcLoc
loc TypeDeclBase NoInfo Name
decl UncheckedExp
e StructType -> TermTypeM StructType
shapef = do
  TypeDeclBase Info VName
decl' <- TypeDeclBase NoInfo Name -> TermTypeM (TypeDeclBase Info VName)
checkTypeDecl TypeDeclBase NoInfo Name
decl
  Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
  PatType
t <- Exp -> TermTypeM PatType
expTypeFully Exp
e'

  (StructType
decl_t_nonrigid, [VName]
_) <-
    SrcLoc
-> String
-> Rigidity
-> StructType
-> TermTypeM (StructType, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"impl" Rigidity
Nonrigid
      (StructType -> TermTypeM (StructType, [VName]))
-> TermTypeM StructType -> TermTypeM (StructType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< StructType -> TermTypeM StructType
shapef (Info StructType -> StructType
forall a. Info a -> a
unInfo (Info StructType -> StructType) -> Info StructType -> StructType
forall a b. (a -> b) -> a -> b
$ TypeDeclBase Info VName -> Info StructType
forall (f :: * -> *) vn. TypeDeclBase f vn -> f StructType
expandedType TypeDeclBase Info VName
decl')

  Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (StructType -> StructType -> Checking
CheckingAscription (Info StructType -> StructType
forall a. Info a -> a
unInfo (Info StructType -> StructType) -> Info StructType -> StructType
forall a b. (a -> b) -> a -> b
$ TypeDeclBase Info VName -> Info StructType
forall (f :: * -> *) vn. TypeDeclBase f vn -> f StructType
expandedType TypeDeclBase Info VName
decl') (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t)) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"type ascription") StructType
decl_t_nonrigid (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t)

  -- We also have to make sure that uniqueness matches.  This is done
  -- explicitly, because uniqueness is ignored by unification.
  PatType
t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
t
  StructType
decl_t' <- StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully (StructType -> TermTypeM StructType)
-> StructType -> TermTypeM StructType
forall a b. (a -> b) -> a -> b
$ Info StructType -> StructType
forall a. Info a -> a
unInfo (Info StructType -> StructType) -> Info StructType -> StructType
forall a b. (a -> b) -> a -> b
$ TypeDeclBase Info VName -> Info StructType
forall (f :: * -> *) vn. TypeDeclBase f vn -> f StructType
expandedType TypeDeclBase Info VName
decl'
  Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (PatType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural PatType
t' TypeBase () () -> TypeBase () () -> Bool
`subtypeOf` StructType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural StructType
decl_t') (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
      Doc
"Type" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
t') Doc -> Doc -> Doc
<+> Doc
"is not a subtype of"
        Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
decl_t') Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."

  (TypeDeclBase Info VName, Exp)
-> TermTypeM (TypeDeclBase Info VName, Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return (TypeDeclBase Info VName
decl', Exp
e')

unscopeType ::
  SrcLoc ->
  M.Map VName Ident ->
  PatType ->
  TermTypeM (PatType, [VName])
unscopeType :: SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType SrcLoc
tloc Map VName Ident
unscoped PatType
t = do
  (PatType
t', Map VName VName
m) <- StateT (Map VName VName) TermTypeM PatType
-> Map VName VName -> TermTypeM (PatType, Map VName VName)
forall s (m :: * -> *) a. StateT s m a -> s -> m (a, s)
runStateT ((Names
 -> DimPos
 -> DimDecl VName
 -> StateT (Map VName VName) TermTypeM (DimDecl VName))
-> PatType -> StateT (Map VName VName) TermTypeM PatType
forall (f :: * -> *) fdim tdim als.
Applicative f =>
(Names -> DimPos -> fdim -> f tdim)
-> TypeBase fdim als -> f (TypeBase tdim als)
traverseDims Names
-> DimPos
-> DimDecl VName
-> StateT (Map VName VName) TermTypeM (DimDecl VName)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) p.
(MonadState (Map VName VName) (t m), MonadTrans t, MonadUnify m) =>
p -> DimPos -> DimDecl VName -> t m (DimDecl VName)
onDim PatType
t) Map VName VName
forall a. Monoid a => a
mempty
  (PatType, [VName]) -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return (PatType
t' PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
`addAliases` (Alias -> Alias) -> Aliasing -> Aliasing
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> Alias
unAlias, Map VName VName -> [VName]
forall k a. Map k a -> [a]
M.elems Map VName VName
m)
  where
    onDim :: p -> DimPos -> DimDecl VName -> t m (DimDecl VName)
onDim p
_ DimPos
p (NamedDim QualName VName
d)
      | Just SrcLoc
loc <- Ident -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf (Ident -> SrcLoc) -> Maybe Ident -> Maybe SrcLoc
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> VName -> Map VName Ident -> Maybe Ident
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup (QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
d) Map VName Ident
unscoped =
        if DimPos
p DimPos -> DimPos -> Bool
forall a. Eq a => a -> a -> Bool
== DimPos
PosImmediate Bool -> Bool -> Bool
|| DimPos
p DimPos -> DimPos -> Bool
forall a. Eq a => a -> a -> Bool
== DimPos
PosParam
          then SrcLoc -> VName -> t m (DimDecl VName)
forall (t :: (* -> *) -> * -> *) (m :: * -> *).
(MonadState (Map VName VName) (t m), MonadTrans t, MonadUnify m) =>
SrcLoc -> VName -> t m (DimDecl VName)
inst SrcLoc
loc (VName -> t m (DimDecl VName)) -> VName -> t m (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
d
          else DimDecl VName -> t m (DimDecl VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (DimDecl VName -> t m (DimDecl VName))
-> DimDecl VName -> t m (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim (Maybe VName -> DimDecl VName) -> Maybe VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> Maybe VName
forall a. a -> Maybe a
Just (VName -> Maybe VName) -> VName -> Maybe VName
forall a b. (a -> b) -> a -> b
$ QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
d
    onDim p
_ DimPos
_ DimDecl VName
d = DimDecl VName -> t m (DimDecl VName)
forall (m :: * -> *) a. Monad m => a -> m a
return DimDecl VName
d

    inst :: SrcLoc -> VName -> t m (DimDecl VName)
inst SrcLoc
loc VName
d = do
      Maybe VName
prev <- (Map VName VName -> Maybe VName) -> t m (Maybe VName)
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets ((Map VName VName -> Maybe VName) -> t m (Maybe VName))
-> (Map VName VName -> Maybe VName) -> t m (Maybe VName)
forall a b. (a -> b) -> a -> b
$ VName -> Map VName VName -> Maybe VName
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
d
      case Maybe VName
prev of
        Just VName
d' -> DimDecl VName -> t m (DimDecl VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (DimDecl VName -> t m (DimDecl VName))
-> DimDecl VName -> t m (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
d'
        Maybe VName
Nothing -> do
          VName
d' <- m VName -> t m VName
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m VName -> t m VName) -> m VName -> t m VName
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Rigidity -> String -> m VName
forall (m :: * -> *).
MonadUnify m =>
SrcLoc -> Rigidity -> String -> m VName
newDimVar SrcLoc
tloc (RigidSource -> Rigidity
Rigid (RigidSource -> Rigidity) -> RigidSource -> Rigidity
forall a b. (a -> b) -> a -> b
$ SrcLoc -> VName -> RigidSource
RigidOutOfScope SrcLoc
loc VName
d) String
"d"
          (Map VName VName -> Map VName VName) -> t m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Map VName VName -> Map VName VName) -> t m ())
-> (Map VName VName -> Map VName VName) -> t m ()
forall a b. (a -> b) -> a -> b
$ VName -> VName -> Map VName VName -> Map VName VName
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
d VName
d'
          DimDecl VName -> t m (DimDecl VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (DimDecl VName -> t m (DimDecl VName))
-> DimDecl VName -> t m (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
d'

    unAlias :: Alias -> Alias
unAlias (AliasBound VName
v) | VName
v VName -> Map VName Ident -> Bool
forall k a. Ord k => k -> Map k a -> Bool
`M.member` Map VName Ident
unscoped = VName -> Alias
AliasFree VName
v
    unAlias Alias
a = Alias
a

-- When a function result is not immediately bound to a name, we need
-- to invent a name for it so we can track it during aliasing
-- (uniqueness-error54.fut, uniqueness-error55.fut).
addResultAliases :: NameReason -> PatType -> TermTypeM PatType
addResultAliases :: NameReason -> PatType -> TermTypeM PatType
addResultAliases NameReason
r (Scalar (Record Map Name PatType
fs)) =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> (Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> Map Name PatType
-> PatType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. Map Name (TypeBase dim as) -> ScalarTypeBase dim as
Record (Map Name PatType -> PatType)
-> TermTypeM (Map Name PatType) -> TermTypeM PatType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (PatType -> TermTypeM PatType)
-> Map Name PatType -> TermTypeM (Map Name PatType)
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse (NameReason -> PatType -> TermTypeM PatType
addResultAliases NameReason
r) Map Name PatType
fs
addResultAliases NameReason
r (Scalar (Sum Map Name [PatType]
fs)) =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> (Map Name [PatType] -> ScalarTypeBase (DimDecl VName) Aliasing)
-> Map Name [PatType]
-> PatType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Name [PatType] -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. Map Name [TypeBase dim as] -> ScalarTypeBase dim as
Sum (Map Name [PatType] -> PatType)
-> TermTypeM (Map Name [PatType]) -> TermTypeM PatType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ([PatType] -> TermTypeM [PatType])
-> Map Name [PatType] -> TermTypeM (Map Name [PatType])
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse ((PatType -> TermTypeM PatType) -> [PatType] -> TermTypeM [PatType]
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse (NameReason -> PatType -> TermTypeM PatType
addResultAliases NameReason
r)) Map Name [PatType]
fs
addResultAliases NameReason
r (Scalar (TypeVar Aliasing
as Uniqueness
u TypeName
tn [TypeArg (DimDecl VName)]
targs)) = do
  VName
v <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID Name
"internal_app_result"
  (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateNames :: Map VName NameReason
stateNames = VName -> NameReason -> Map VName NameReason -> Map VName NameReason
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
v NameReason
r (Map VName NameReason -> Map VName NameReason)
-> Map VName NameReason -> Map VName NameReason
forall a b. (a -> b) -> a -> b
$ TermTypeState -> Map VName NameReason
stateNames TermTypeState
s}
  PatType -> TermTypeM PatType
forall (f :: * -> *) a. Applicative f => a -> f a
pure (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar (Alias -> Aliasing -> Aliasing
forall a. Ord a => a -> Set a -> Set a
S.insert (VName -> Alias
AliasFree VName
v) Aliasing
as) Uniqueness
u TypeName
tn [TypeArg (DimDecl VName)]
targs
addResultAliases NameReason
_ (Scalar t :: ScalarTypeBase (DimDecl VName) Aliasing
t@Prim {}) = PatType -> TermTypeM PatType
forall (f :: * -> *) a. Applicative f => a -> f a
pure (ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar ScalarTypeBase (DimDecl VName) Aliasing
t)
addResultAliases NameReason
_ (Scalar t :: ScalarTypeBase (DimDecl VName) Aliasing
t@Arrow {}) = PatType -> TermTypeM PatType
forall (f :: * -> *) a. Applicative f => a -> f a
pure (ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar ScalarTypeBase (DimDecl VName) Aliasing
t)
addResultAliases NameReason
r (Array Aliasing
als Uniqueness
u ScalarTypeBase (DimDecl VName) ()
t ShapeDecl (DimDecl VName)
shape) = do
  VName
v <- Name -> TermTypeM VName
forall (m :: * -> *). MonadTypeChecker m => Name -> m VName
newID Name
"internal_app_result"
  (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateNames :: Map VName NameReason
stateNames = VName -> NameReason -> Map VName NameReason -> Map VName NameReason
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
v NameReason
r (Map VName NameReason -> Map VName NameReason)
-> Map VName NameReason -> Map VName NameReason
forall a b. (a -> b) -> a -> b
$ TermTypeState -> Map VName NameReason
stateNames TermTypeState
s}
  PatType -> TermTypeM PatType
forall (f :: * -> *) a. Applicative f => a -> f a
pure (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> Uniqueness
-> ScalarTypeBase (DimDecl VName) ()
-> ShapeDecl (DimDecl VName)
-> PatType
forall dim as.
as
-> Uniqueness
-> ScalarTypeBase dim ()
-> ShapeDecl dim
-> TypeBase dim as
Array (Alias -> Aliasing -> Aliasing
forall a. Ord a => a -> Set a -> Set a
S.insert (VName -> Alias
AliasFree VName
v) Aliasing
als) Uniqueness
u ScalarTypeBase (DimDecl VName) ()
t ShapeDecl (DimDecl VName)
shape

-- 'checkApplyExp' is like 'checkExp', but tries to find the "root
-- function", for better error messages.
checkApplyExp :: UncheckedExp -> TermTypeM (Exp, ApplyOp)
checkApplyExp :: UncheckedExp -> TermTypeM (Exp, ApplyOp)
checkApplyExp (AppExp (Apply UncheckedExp
e1 UncheckedExp
e2 NoInfo (Diet, Maybe VName)
_ SrcLoc
loc) NoInfo AppRes
_) = do
  Arg
arg <- UncheckedExp -> TermTypeM Arg
checkArg UncheckedExp
e2
  (Exp
e1', (Maybe (QualName VName)
fname, Int
i)) <- UncheckedExp -> TermTypeM (Exp, ApplyOp)
checkApplyExp UncheckedExp
e1
  PatType
t <- Exp -> TermTypeM PatType
expType Exp
e1'
  (PatType
t1, PatType
rt, Maybe VName
argext, [VName]
exts) <- SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply SrcLoc
loc (Maybe (QualName VName)
fname, Int
i) PatType
t Arg
arg
  PatType
rt' <- NameReason -> PatType -> TermTypeM PatType
addResultAliases (Maybe (QualName VName) -> SrcLoc -> NameReason
NameAppRes Maybe (QualName VName)
fname SrcLoc
loc) PatType
rt
  (Exp, ApplyOp) -> TermTypeM (Exp, ApplyOp)
forall (m :: * -> *) a. Monad m => a -> m a
return
    ( AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp
        (Exp
-> Exp
-> Info (Diet, Maybe VName)
-> SrcLoc
-> AppExpBase Info VName
forall (f :: * -> *) vn.
ExpBase f vn
-> ExpBase f vn
-> f (Diet, Maybe VName)
-> SrcLoc
-> AppExpBase f vn
Apply Exp
e1' (Arg -> Exp
argExp Arg
arg) ((Diet, Maybe VName) -> Info (Diet, Maybe VName)
forall a. a -> Info a
Info (PatType -> Diet
forall shape as. TypeBase shape as -> Diet
diet PatType
t1, Maybe VName
argext)) SrcLoc
loc)
        (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
rt' [VName]
exts),
      (Maybe (QualName VName)
fname, Int
i Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
    )
checkApplyExp UncheckedExp
e = do
  Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
  (Exp, ApplyOp) -> TermTypeM (Exp, ApplyOp)
forall (m :: * -> *) a. Monad m => a -> m a
return
    ( Exp
e',
      ( case Exp
e' of
          Var QualName VName
qn Info PatType
_ SrcLoc
_ -> QualName VName -> Maybe (QualName VName)
forall a. a -> Maybe a
Just QualName VName
qn
          Exp
_ -> Maybe (QualName VName)
forall a. Maybe a
Nothing,
        Int
0
      )
    )

checkExp :: UncheckedExp -> TermTypeM Exp
checkExp :: UncheckedExp -> TermTypeM Exp
checkExp (Literal PrimValue
val SrcLoc
loc) =
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ PrimValue -> SrcLoc -> Exp
forall (f :: * -> *) vn. PrimValue -> SrcLoc -> ExpBase f vn
Literal PrimValue
val SrcLoc
loc
checkExp (StringLit [Word8]
vs SrcLoc
loc) =
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ [Word8] -> SrcLoc -> Exp
forall (f :: * -> *) vn. [Word8] -> SrcLoc -> ExpBase f vn
StringLit [Word8]
vs SrcLoc
loc
checkExp (IntLit Integer
val NoInfo PatType
NoInfo SrcLoc
loc) = do
  StructType
t <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  [PrimType] -> Usage -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
[PrimType] -> Usage -> StructType -> m ()
mustBeOneOf [PrimType]
anyNumberType (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"integer literal") StructType
t
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Integer -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
Integer -> f PatType -> SrcLoc -> ExpBase f vn
IntLit Integer
val (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t) SrcLoc
loc
checkExp (FloatLit Double
val NoInfo PatType
NoInfo SrcLoc
loc) = do
  StructType
t <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  [PrimType] -> Usage -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
[PrimType] -> Usage -> StructType -> m ()
mustBeOneOf [PrimType]
anyFloatType (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"float literal") StructType
t
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Double -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
Double -> f PatType -> SrcLoc -> ExpBase f vn
FloatLit Double
val (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t) SrcLoc
loc
checkExp (TupLit [UncheckedExp]
es SrcLoc
loc) =
  [Exp] -> SrcLoc -> Exp
forall (f :: * -> *) vn. [ExpBase f vn] -> SrcLoc -> ExpBase f vn
TupLit ([Exp] -> SrcLoc -> Exp)
-> TermTypeM [Exp] -> TermTypeM (SrcLoc -> Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (UncheckedExp -> TermTypeM Exp)
-> [UncheckedExp] -> TermTypeM [Exp]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM UncheckedExp -> TermTypeM Exp
checkExp [UncheckedExp]
es TermTypeM (SrcLoc -> Exp) -> TermTypeM SrcLoc -> TermTypeM Exp
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
checkExp (RecordLit [FieldBase NoInfo Name]
fs SrcLoc
loc) = do
  [FieldBase Info VName]
fs' <- StateT (Map Name SrcLoc) TermTypeM [FieldBase Info VName]
-> Map Name SrcLoc -> TermTypeM [FieldBase Info VName]
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m a
evalStateT ((FieldBase NoInfo Name
 -> StateT (Map Name SrcLoc) TermTypeM (FieldBase Info VName))
-> [FieldBase NoInfo Name]
-> StateT (Map Name SrcLoc) TermTypeM [FieldBase Info VName]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM FieldBase NoInfo Name
-> StateT (Map Name SrcLoc) TermTypeM (FieldBase Info VName)
forall (t :: (* -> *) -> * -> *).
(MonadState (Map Name SrcLoc) (t TermTypeM), MonadTrans t) =>
FieldBase NoInfo Name -> t TermTypeM (FieldBase Info VName)
checkField [FieldBase NoInfo Name]
fs) Map Name SrcLoc
forall a. Monoid a => a
mempty

  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ [FieldBase Info VName] -> SrcLoc -> Exp
forall (f :: * -> *) vn. [FieldBase f vn] -> SrcLoc -> ExpBase f vn
RecordLit [FieldBase Info VName]
fs' SrcLoc
loc
  where
    checkField :: FieldBase NoInfo Name -> t TermTypeM (FieldBase Info VName)
checkField (RecordFieldExplicit Name
f UncheckedExp
e SrcLoc
rloc) = do
      Name -> SrcLoc -> t TermTypeM ()
forall a b (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadState (Map a b) (t m), Ord a, MonadTrans t,
 MonadTypeChecker m, Pretty a, Located a, Located b) =>
a -> a -> t m ()
errIfAlreadySet Name
f SrcLoc
rloc
      (Map Name SrcLoc -> Map Name SrcLoc) -> t TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Map Name SrcLoc -> Map Name SrcLoc) -> t TermTypeM ())
-> (Map Name SrcLoc -> Map Name SrcLoc) -> t TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Name -> SrcLoc -> Map Name SrcLoc -> Map Name SrcLoc
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert Name
f SrcLoc
rloc
      Name -> Exp -> SrcLoc -> FieldBase Info VName
forall (f :: * -> *) vn.
Name -> ExpBase f vn -> SrcLoc -> FieldBase f vn
RecordFieldExplicit Name
f (Exp -> SrcLoc -> FieldBase Info VName)
-> t TermTypeM Exp -> t TermTypeM (SrcLoc -> FieldBase Info VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> TermTypeM Exp -> t TermTypeM Exp
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e) t TermTypeM (SrcLoc -> FieldBase Info VName)
-> t TermTypeM SrcLoc -> t TermTypeM (FieldBase Info VName)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> t TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
rloc
    checkField (RecordFieldImplicit Name
name NoInfo PatType
NoInfo SrcLoc
rloc) = do
      Name -> SrcLoc -> t TermTypeM ()
forall a b (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadState (Map a b) (t m), Ord a, MonadTrans t,
 MonadTypeChecker m, Pretty a, Located a, Located b) =>
a -> a -> t m ()
errIfAlreadySet Name
name SrcLoc
rloc
      (QualName [VName]
_ VName
name', PatType
t) <- TermTypeM (QualName VName, PatType)
-> t TermTypeM (QualName VName, PatType)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (TermTypeM (QualName VName, PatType)
 -> t TermTypeM (QualName VName, PatType))
-> TermTypeM (QualName VName, PatType)
-> t TermTypeM (QualName VName, PatType)
forall a b. (a -> b) -> a -> b
$ SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
rloc (QualName Name -> TermTypeM (QualName VName, PatType))
-> QualName Name -> TermTypeM (QualName VName, PatType)
forall a b. (a -> b) -> a -> b
$ Name -> QualName Name
forall v. v -> QualName v
qualName Name
name
      (Map Name SrcLoc -> Map Name SrcLoc) -> t TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Map Name SrcLoc -> Map Name SrcLoc) -> t TermTypeM ())
-> (Map Name SrcLoc -> Map Name SrcLoc) -> t TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Name -> SrcLoc -> Map Name SrcLoc -> Map Name SrcLoc
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert Name
name SrcLoc
rloc
      FieldBase Info VName -> t TermTypeM (FieldBase Info VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (FieldBase Info VName -> t TermTypeM (FieldBase Info VName))
-> FieldBase Info VName -> t TermTypeM (FieldBase Info VName)
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> FieldBase Info VName
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> FieldBase f vn
RecordFieldImplicit VName
name' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
rloc

    errIfAlreadySet :: a -> a -> t m ()
errIfAlreadySet a
f a
rloc = do
      Maybe b
maybe_sloc <- (Map a b -> Maybe b) -> t m (Maybe b)
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets ((Map a b -> Maybe b) -> t m (Maybe b))
-> (Map a b -> Maybe b) -> t m (Maybe b)
forall a b. (a -> b) -> a -> b
$ a -> Map a b -> Maybe b
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup a
f
      case Maybe b
maybe_sloc of
        Just b
sloc ->
          m () -> t m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m () -> t m ()) -> (Doc -> m ()) -> Doc -> t m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError a
rloc Notes
forall a. Monoid a => a
mempty (Doc -> t m ()) -> Doc -> t m ()
forall a b. (a -> b) -> a -> b
$
            Doc
"Field" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (a -> Doc
forall a. Pretty a => a -> Doc
ppr a
f)
              Doc -> Doc -> Doc
<+> Doc
"previously defined at"
              Doc -> Doc -> Doc
<+> String -> Doc
text (a -> b -> String
forall a b. (Located a, Located b) => a -> b -> String
locStrRel a
rloc b
sloc) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
        Maybe b
Nothing -> () -> t m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
checkExp (ArrayLit [UncheckedExp]
all_es NoInfo PatType
_ SrcLoc
loc) =
  -- Construct the result type and unify all elements with it.  We
  -- only create a type variable for empty arrays; otherwise we use
  -- the type of the first element.  This significantly cuts down on
  -- the number of type variables generated for pathologically large
  -- multidimensional array literals.
  case [UncheckedExp]
all_es of
    [] -> do
      PatType
et <- SrcLoc -> String -> TermTypeM PatType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
      PatType
t <- SrcLoc
-> PatType
-> ShapeDecl (DimDecl VName)
-> Uniqueness
-> TermTypeM PatType
forall dim as.
(Pretty (ShapeDecl dim), Monoid as) =>
SrcLoc
-> TypeBase dim as
-> ShapeDecl dim
-> Uniqueness
-> TermTypeM (TypeBase dim as)
arrayOfM SrcLoc
loc PatType
et ([DimDecl VName] -> ShapeDecl (DimDecl VName)
forall dim. [dim] -> ShapeDecl dim
ShapeDecl [Int -> DimDecl VName
forall vn. Int -> DimDecl vn
ConstDim Int
0]) Uniqueness
Unique
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ [Exp] -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
[ExpBase f vn] -> f PatType -> SrcLoc -> ExpBase f vn
ArrayLit [] (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc
    UncheckedExp
e : [UncheckedExp]
es -> do
      Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
      PatType
et <- Exp -> TermTypeM PatType
expType Exp
e'
      [Exp]
es' <- (UncheckedExp -> TermTypeM Exp)
-> [UncheckedExp] -> TermTypeM [Exp]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM (String -> StructType -> Exp -> TermTypeM Exp
unifies String
"type of first array element" (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
et) (Exp -> TermTypeM Exp)
-> (UncheckedExp -> TermTypeM Exp) -> UncheckedExp -> TermTypeM Exp
forall (m :: * -> *) b c a.
Monad m =>
(b -> m c) -> (a -> m b) -> a -> m c
<=< UncheckedExp -> TermTypeM Exp
checkExp) [UncheckedExp]
es
      PatType
et' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
et
      PatType
t <- SrcLoc
-> PatType
-> ShapeDecl (DimDecl VName)
-> Uniqueness
-> TermTypeM PatType
forall dim as.
(Pretty (ShapeDecl dim), Monoid as) =>
SrcLoc
-> TypeBase dim as
-> ShapeDecl dim
-> Uniqueness
-> TermTypeM (TypeBase dim as)
arrayOfM SrcLoc
loc PatType
et' ([DimDecl VName] -> ShapeDecl (DimDecl VName)
forall dim. [dim] -> ShapeDecl dim
ShapeDecl [Int -> DimDecl VName
forall vn. Int -> DimDecl vn
ConstDim (Int -> DimDecl VName) -> Int -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ [UncheckedExp] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [UncheckedExp]
all_es]) Uniqueness
Unique
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ [Exp] -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
[ExpBase f vn] -> f PatType -> SrcLoc -> ExpBase f vn
ArrayLit (Exp
e' Exp -> [Exp] -> [Exp]
forall a. a -> [a] -> [a]
: [Exp]
es') (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc
checkExp (AppExp (Range UncheckedExp
start Maybe UncheckedExp
maybe_step Inclusiveness UncheckedExp
end SrcLoc
loc) NoInfo AppRes
_) = do
  Exp
start' <- String -> [PrimType] -> Exp -> TermTypeM Exp
require String
"use in range expression" [PrimType]
anySignedType (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
start
  StructType
start_t <- PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType)
-> TermTypeM PatType -> TermTypeM StructType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Exp -> TermTypeM PatType
expTypeFully Exp
start'
  Maybe Exp
maybe_step' <- case Maybe UncheckedExp
maybe_step of
    Maybe UncheckedExp
Nothing -> Maybe Exp -> TermTypeM (Maybe Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe Exp
forall a. Maybe a
Nothing
    Just UncheckedExp
step -> do
      let warning :: TermTypeM ()
warning = SrcLoc -> Doc -> TermTypeM ()
forall (m :: * -> *) loc.
(MonadTypeChecker m, Located loc) =>
loc -> Doc -> m ()
warn SrcLoc
loc Doc
"First and second element of range are identical, this will produce an empty array."
      case (UncheckedExp
start, UncheckedExp
step) of
        (Literal PrimValue
x SrcLoc
_, Literal PrimValue
y SrcLoc
_) -> Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (PrimValue
x PrimValue -> PrimValue -> Bool
forall a. Eq a => a -> a -> Bool
== PrimValue
y) TermTypeM ()
warning
        (Var QualName Name
x_name NoInfo PatType
_ SrcLoc
_, Var QualName Name
y_name NoInfo PatType
_ SrcLoc
_) -> Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (QualName Name
x_name QualName Name -> QualName Name -> Bool
forall a. Eq a => a -> a -> Bool
== QualName Name
y_name) TermTypeM ()
warning
        (UncheckedExp, UncheckedExp)
_ -> () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
      Exp -> Maybe Exp
forall a. a -> Maybe a
Just (Exp -> Maybe Exp) -> TermTypeM Exp -> TermTypeM (Maybe Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (String -> StructType -> Exp -> TermTypeM Exp
unifies String
"use in range expression" StructType
start_t (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
step)

  let unifyRange :: UncheckedExp -> TermTypeM Exp
unifyRange UncheckedExp
e = String -> StructType -> Exp -> TermTypeM Exp
unifies String
"use in range expression" StructType
start_t (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
  Inclusiveness Exp
end' <- (UncheckedExp -> TermTypeM Exp)
-> Inclusiveness UncheckedExp -> TermTypeM (Inclusiveness Exp)
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse UncheckedExp -> TermTypeM Exp
unifyRange Inclusiveness UncheckedExp
end

  PatType
end_t <- case Inclusiveness Exp
end' of
    DownToExclusive Exp
e -> Exp -> TermTypeM PatType
expType Exp
e
    ToInclusive Exp
e -> Exp -> TermTypeM PatType
expType Exp
e
    UpToExclusive Exp
e -> Exp -> TermTypeM PatType
expType Exp
e

  -- Special case some ranges to give them a known size.
  let dimFromBound :: Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromBound = (Exp -> SizeSource)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromExp (ExpBase NoInfo VName -> SizeSource
SourceBound (ExpBase NoInfo VName -> SizeSource)
-> (Exp -> ExpBase NoInfo VName) -> Exp -> SizeSource
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> ExpBase NoInfo VName
bareExp)
  (DimDecl VName
dim, Maybe VName
retext) <-
    case (Exp -> Maybe Int64
isInt64 Exp
start', Exp -> Maybe Int64
isInt64 (Exp -> Maybe Int64) -> Maybe Exp -> Maybe (Maybe Int64)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe Exp
maybe_step', Inclusiveness Exp
end') of
      (Just Int64
0, Just (Just Int64
1), UpToExclusive Exp
end'')
        | Scalar (Prim (Signed IntType
Int64)) <- PatType
end_t ->
          Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromBound Exp
end''
      (Just Int64
0, Maybe (Maybe Int64)
Nothing, UpToExclusive Exp
end'')
        | Scalar (Prim (Signed IntType
Int64)) <- PatType
end_t ->
          Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromBound Exp
end''
      (Just Int64
1, Just (Just Int64
2), ToInclusive Exp
end'')
        | Scalar (Prim (Signed IntType
Int64)) <- PatType
end_t ->
          Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromBound Exp
end''
      (Maybe Int64, Maybe (Maybe Int64), Inclusiveness Exp)
_ -> do
        VName
d <- SrcLoc -> Rigidity -> String -> TermTypeM VName
forall (m :: * -> *).
MonadUnify m =>
SrcLoc -> Rigidity -> String -> m VName
newDimVar SrcLoc
loc (RigidSource -> Rigidity
Rigid RigidSource
RigidRange) String
"range_dim"
        (DimDecl VName, Maybe VName)
-> TermTypeM (DimDecl VName, Maybe VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
d, VName -> Maybe VName
forall a. a -> Maybe a
Just VName
d)

  StructType
t <- SrcLoc
-> StructType
-> ShapeDecl (DimDecl VName)
-> Uniqueness
-> TermTypeM StructType
forall dim as.
(Pretty (ShapeDecl dim), Monoid as) =>
SrcLoc
-> TypeBase dim as
-> ShapeDecl dim
-> Uniqueness
-> TermTypeM (TypeBase dim as)
arrayOfM SrcLoc
loc StructType
start_t ([DimDecl VName] -> ShapeDecl (DimDecl VName)
forall dim. [dim] -> ShapeDecl dim
ShapeDecl [DimDecl VName
dim]) Uniqueness
Unique
  let res :: AppRes
res = PatType -> [VName] -> AppRes
AppRes (StructType
t StructType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` Aliasing
forall a. Monoid a => a
mempty) (Maybe VName -> [VName]
forall a. Maybe a -> [a]
maybeToList Maybe VName
retext)

  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp (Exp
-> Maybe Exp
-> Inclusiveness Exp
-> SrcLoc
-> AppExpBase Info VName
forall (f :: * -> *) vn.
ExpBase f vn
-> Maybe (ExpBase f vn)
-> Inclusiveness (ExpBase f vn)
-> SrcLoc
-> AppExpBase f vn
Range Exp
start' Maybe Exp
maybe_step' Inclusiveness Exp
end' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info AppRes
res)
checkExp (Ascript UncheckedExp
e TypeDeclBase NoInfo Name
decl SrcLoc
loc) = do
  (TypeDeclBase Info VName
decl', Exp
e') <- SrcLoc
-> TypeDeclBase NoInfo Name
-> UncheckedExp
-> (StructType -> TermTypeM StructType)
-> TermTypeM (TypeDeclBase Info VName, Exp)
checkAscript SrcLoc
loc TypeDeclBase NoInfo Name
decl UncheckedExp
e StructType -> TermTypeM StructType
forall (f :: * -> *) a. Applicative f => a -> f a
pure
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Exp -> TypeDeclBase Info VName -> SrcLoc -> Exp
forall (f :: * -> *) vn.
ExpBase f vn -> TypeDeclBase f vn -> SrcLoc -> ExpBase f vn
Ascript Exp
e' TypeDeclBase Info VName
decl' SrcLoc
loc
checkExp (AppExp (Coerce UncheckedExp
e TypeDeclBase NoInfo Name
decl SrcLoc
loc) NoInfo AppRes
_) = do
  -- We instantiate the declared types with all dimensions as nonrigid
  -- fresh type variables, which we then use to unify with the type of
  -- 'e'.  This lets 'e' have whatever sizes it wants, but the overall
  -- type must still match.  Eventually we will throw away those sizes
  -- (they will end up being unified with various sizes in 'e', which
  -- is fine).
  (TypeDeclBase Info VName
decl', Exp
e') <- SrcLoc
-> TypeDeclBase NoInfo Name
-> UncheckedExp
-> (StructType -> TermTypeM StructType)
-> TermTypeM (TypeDeclBase Info VName, Exp)
checkAscript SrcLoc
loc TypeDeclBase NoInfo Name
decl UncheckedExp
e ((StructType -> TermTypeM StructType)
 -> TermTypeM (TypeDeclBase Info VName, Exp))
-> (StructType -> TermTypeM StructType)
-> TermTypeM (TypeDeclBase Info VName, Exp)
forall a b. (a -> b) -> a -> b
$ StructType -> TermTypeM StructType
forall (f :: * -> *) a. Applicative f => a -> f a
pure (StructType -> TermTypeM StructType)
-> (StructType -> StructType) -> StructType -> TermTypeM StructType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. StructType -> StructType
forall vn as. TypeBase (DimDecl vn) as -> TypeBase (DimDecl vn) as
anySizes

  -- Now we instantiate the declared type again, but this time we keep
  -- around the sizes as existentials.  This is the result of the
  -- ascription as a whole.  We use matchDims to obtain the aliasing
  -- of 'e'.
  (StructType
decl_t_rigid, [VName]
ext) <-
    SrcLoc
-> Maybe (QualName VName)
-> StructType
-> TermTypeM (StructType, [VName])
forall als.
SrcLoc
-> Maybe (QualName VName)
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInReturnType SrcLoc
loc Maybe (QualName VName)
forall a. Maybe a
Nothing (StructType -> TermTypeM (StructType, [VName]))
-> StructType -> TermTypeM (StructType, [VName])
forall a b. (a -> b) -> a -> b
$ Info StructType -> StructType
forall a. Info a -> a
unInfo (Info StructType -> StructType) -> Info StructType -> StructType
forall a b. (a -> b) -> a -> b
$ TypeDeclBase Info VName -> Info StructType
forall (f :: * -> *) vn. TypeDeclBase f vn -> f StructType
expandedType TypeDeclBase Info VName
decl'

  PatType
t <- Exp -> TermTypeM PatType
expTypeFully Exp
e'

  PatType
t' <- (DimDecl VName -> DimDecl VName -> TermTypeM (DimDecl VName))
-> PatType -> PatType -> TermTypeM PatType
forall as (m :: * -> *) d1 d2.
(Monoid as, Monad m) =>
(d1 -> d2 -> m d1)
-> TypeBase d1 as -> TypeBase d2 as -> m (TypeBase d1 as)
matchDims ((DimDecl VName -> TermTypeM (DimDecl VName))
-> DimDecl VName -> DimDecl VName -> TermTypeM (DimDecl VName)
forall a b. a -> b -> a
const DimDecl VName -> TermTypeM (DimDecl VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure) PatType
t (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
decl_t_rigid

  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp (Exp -> TypeDeclBase Info VName -> SrcLoc -> AppExpBase Info VName
forall (f :: * -> *) vn.
ExpBase f vn -> TypeDeclBase f vn -> SrcLoc -> AppExpBase f vn
Coerce Exp
e' TypeDeclBase Info VName
decl' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
t' [VName]
ext)
checkExp (AppExp (BinOp (QualName Name
op, SrcLoc
oploc) NoInfo PatType
NoInfo (UncheckedExp
e1, NoInfo (StructType, Maybe VName)
_) (UncheckedExp
e2, NoInfo (StructType, Maybe VName)
_) SrcLoc
loc) NoInfo AppRes
NoInfo) = do
  (QualName VName
op', PatType
ftype) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
oploc QualName Name
op
  Arg
e1_arg <- UncheckedExp -> TermTypeM Arg
checkArg UncheckedExp
e1
  Arg
e2_arg <- UncheckedExp -> TermTypeM Arg
checkArg UncheckedExp
e2

  -- Note that the application to the first operand cannot fix any
  -- existential sizes, because it must by necessity be a function.
  (PatType
p1_t, PatType
rt, Maybe VName
p1_ext, [VName]
_) <- SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply SrcLoc
loc (QualName VName -> Maybe (QualName VName)
forall a. a -> Maybe a
Just QualName VName
op', Int
0) PatType
ftype Arg
e1_arg
  (PatType
p2_t, PatType
rt', Maybe VName
p2_ext, [VName]
retext) <- SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply SrcLoc
loc (QualName VName -> Maybe (QualName VName)
forall a. a -> Maybe a
Just QualName VName
op', Int
1) PatType
rt Arg
e2_arg

  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
    AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp
      ( (QualName VName, SrcLoc)
-> Info PatType
-> (Exp, Info (StructType, Maybe VName))
-> (Exp, Info (StructType, Maybe VName))
-> SrcLoc
-> AppExpBase Info VName
forall (f :: * -> *) vn.
(QualName vn, SrcLoc)
-> f PatType
-> (ExpBase f vn, f (StructType, Maybe VName))
-> (ExpBase f vn, f (StructType, Maybe VName))
-> SrcLoc
-> AppExpBase f vn
BinOp
          (QualName VName
op', SrcLoc
oploc)
          (PatType -> Info PatType
forall a. a -> Info a
Info PatType
ftype)
          (Arg -> Exp
argExp Arg
e1_arg, (StructType, Maybe VName) -> Info (StructType, Maybe VName)
forall a. a -> Info a
Info (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
p1_t, Maybe VName
p1_ext))
          (Arg -> Exp
argExp Arg
e2_arg, (StructType, Maybe VName) -> Info (StructType, Maybe VName)
forall a. a -> Info a
Info (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
p2_t, Maybe VName
p2_ext))
          SrcLoc
loc
      )
      (AppRes -> Info AppRes
forall a. a -> Info a
Info (PatType -> [VName] -> AppRes
AppRes PatType
rt' [VName]
retext))
checkExp (Project Name
k UncheckedExp
e NoInfo PatType
NoInfo SrcLoc
loc) = do
  Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
  PatType
t <- Exp -> TermTypeM PatType
expType Exp
e'
  PatType
kt <- Usage -> Name -> PatType -> TermTypeM PatType
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> PatType -> m PatType
mustHaveField (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc (String -> Usage) -> String -> Usage
forall a b. (a -> b) -> a -> b
$ String
"projection of field " String -> ShowS
forall a. [a] -> [a] -> [a]
++ ShowS
quote (Name -> String
forall a. Pretty a => a -> String
pretty Name
k)) Name
k PatType
t
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Name -> Exp -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
Name -> ExpBase f vn -> f PatType -> SrcLoc -> ExpBase f vn
Project Name
k Exp
e' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
kt) SrcLoc
loc
checkExp (AppExp (If UncheckedExp
e1 UncheckedExp
e2 UncheckedExp
e3 SrcLoc
loc) NoInfo AppRes
_) =
  TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially TermTypeM Exp
checkCond ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
e1' [Occurence]
_ -> do
    ((Exp
e2', Exp
e3'), [Occurence]
dflow) <- TermTypeM (Exp, Exp) -> TermTypeM ((Exp, Exp), [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences (TermTypeM (Exp, Exp) -> TermTypeM ((Exp, Exp), [Occurence]))
-> TermTypeM (Exp, Exp) -> TermTypeM ((Exp, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e2 TermTypeM Exp -> TermTypeM Exp -> TermTypeM (Exp, Exp)
forall a b. TermTypeM a -> TermTypeM b -> TermTypeM (a, b)
`alternative` UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e3

    (PatType
brancht, [VName]
retext) <- SrcLoc -> Exp -> Exp -> TermTypeM (PatType, [VName])
unifyBranches SrcLoc
loc Exp
e2' Exp
e3'
    let t' :: PatType
t' = PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
addAliases PatType
brancht ((Aliasing -> Aliasing) -> PatType)
-> (Aliasing -> Aliasing) -> PatType
forall a b. (a -> b) -> a -> b
$ (Alias -> Bool) -> Aliasing -> Aliasing
forall a. (a -> Bool) -> Set a -> Set a
S.filter ((Alias -> Bool) -> Aliasing -> Aliasing)
-> (Alias -> Bool) -> Aliasing -> Aliasing
forall a b. (a -> b) -> a -> b
$ (VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.notMember` [Occurence] -> Names
allConsumed [Occurence]
dflow) (VName -> Bool) -> (Alias -> VName) -> Alias -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Alias -> VName
aliasVar

    Usage -> String -> PatType -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> String -> TypeBase dim as -> m ()
zeroOrderType
      (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"returning value of this type from 'if' expression")
      String
"type returned from branch"
      PatType
t'

    Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp (Exp -> Exp -> Exp -> SrcLoc -> AppExpBase Info VName
forall (f :: * -> *) vn.
ExpBase f vn
-> ExpBase f vn -> ExpBase f vn -> SrcLoc -> AppExpBase f vn
If Exp
e1' Exp
e2' Exp
e3' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
t' [VName]
retext)
  where
    checkCond :: TermTypeM Exp
checkCond = do
      Exp
e1' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e1
      let bool :: TypeBase dim as
bool = ScalarTypeBase dim as -> TypeBase dim as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim as -> TypeBase dim as)
-> ScalarTypeBase dim as -> TypeBase dim as
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase dim as
forall dim as. PrimType -> ScalarTypeBase dim as
Prim PrimType
Bool
      StructType
e1_t <- PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType)
-> TermTypeM PatType -> TermTypeM StructType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Exp -> TermTypeM PatType
expType Exp
e1'
      Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure ([StructType] -> StructType -> Checking
CheckingRequired [StructType
forall dim as. TypeBase dim as
bool] StructType
e1_t) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage (Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
e1') String
"use as 'if' condition") StructType
forall dim as. TypeBase dim as
bool StructType
e1_t
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e1'
checkExp (Parens UncheckedExp
e SrcLoc
loc) =
  Exp -> SrcLoc -> Exp
forall (f :: * -> *) vn. ExpBase f vn -> SrcLoc -> ExpBase f vn
Parens (Exp -> SrcLoc -> Exp)
-> TermTypeM Exp -> TermTypeM (SrcLoc -> Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e TermTypeM (SrcLoc -> Exp) -> TermTypeM SrcLoc -> TermTypeM Exp
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc
checkExp (QualParens (QualName Name
modname, SrcLoc
modnameloc) UncheckedExp
e SrcLoc
loc) = do
  (QualName VName
modname', Mod
mod) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, Mod)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, Mod)
lookupMod SrcLoc
loc QualName Name
modname
  case Mod
mod of
    ModEnv Env
env -> (TermEnv -> TermEnv) -> TermTypeM Exp -> TermTypeM Exp
forall r (m :: * -> *) a. MonadReader r m => (r -> r) -> m a -> m a
local (TermEnv -> Env -> TermEnv
`withEnv` QualName VName -> Env -> Env
qualifyEnv QualName VName
modname' Env
env) (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ do
      Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ (QualName VName, SrcLoc) -> Exp -> SrcLoc -> Exp
forall (f :: * -> *) vn.
(QualName vn, SrcLoc) -> ExpBase f vn -> SrcLoc -> ExpBase f vn
QualParens (QualName VName
modname', SrcLoc
modnameloc) Exp
e' SrcLoc
loc
    ModFun {} ->
      SrcLoc -> Notes -> Doc -> TermTypeM Exp
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM Exp) -> Doc -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Doc
"Module" Doc -> Doc -> Doc
<+> QualName Name -> Doc
forall a. Pretty a => a -> Doc
ppr QualName Name
modname Doc -> Doc -> Doc
<+> Doc
" is a parametric module."
  where
    qualifyEnv :: QualName VName -> Env -> Env
qualifyEnv QualName VName
modname' Env
env =
      Env
env {envNameMap :: NameMap
envNameMap = (QualName VName -> QualName VName) -> NameMap -> NameMap
forall a b k. (a -> b) -> Map k a -> Map k b
M.map (QualName VName -> QualName VName -> QualName VName
forall vn. QualName vn -> QualName vn -> QualName vn
qualify' QualName VName
modname') (NameMap -> NameMap) -> NameMap -> NameMap
forall a b. (a -> b) -> a -> b
$ Env -> NameMap
envNameMap Env
env}
    qualify' :: QualName vn -> QualName vn -> QualName vn
qualify' QualName vn
modname' (QualName [vn]
qs vn
name) =
      [vn] -> vn -> QualName vn
forall vn. [vn] -> vn -> QualName vn
QualName (QualName vn -> [vn]
forall vn. QualName vn -> [vn]
qualQuals QualName vn
modname' [vn] -> [vn] -> [vn]
forall a. [a] -> [a] -> [a]
++ [QualName vn -> vn
forall vn. QualName vn -> vn
qualLeaf QualName vn
modname'] [vn] -> [vn] -> [vn]
forall a. [a] -> [a] -> [a]
++ [vn]
qs) vn
name
checkExp (Var QualName Name
qn NoInfo PatType
NoInfo SrcLoc
loc) = do
  -- The qualifiers of a variable is divided into two parts: first a
  -- possibly-empty sequence of module qualifiers, followed by a
  -- possible-empty sequence of record field accesses.  We use scope
  -- information to perform the split, by taking qualifiers off the
  -- end until we find a module.

  (QualName VName
qn', PatType
t, [Name]
fields) <- [Name] -> Name -> TermTypeM (QualName VName, PatType, [Name])
forall b (m :: * -> *).
(MonadError b m, MonadTypeChecker m) =>
[Name] -> Name -> m (QualName VName, PatType, [Name])
findRootVar (QualName Name -> [Name]
forall vn. QualName vn -> [vn]
qualQuals QualName Name
qn) (QualName Name -> Name
forall vn. QualName vn -> vn
qualLeaf QualName Name
qn)

  (Exp -> Name -> TermTypeM Exp) -> Exp -> [Name] -> TermTypeM Exp
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m b
foldM Exp -> Name -> TermTypeM Exp
checkField (QualName VName -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
QualName vn -> f PatType -> SrcLoc -> ExpBase f vn
Var QualName VName
qn' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t) SrcLoc
loc) [Name]
fields
  where
    findRootVar :: [Name] -> Name -> m (QualName VName, PatType, [Name])
findRootVar [Name]
qs Name
name =
      ((QualName VName, PatType) -> (QualName VName, PatType, [Name])
forall a b a. (a, b) -> (a, b, [a])
whenFound ((QualName VName, PatType) -> (QualName VName, PatType, [Name]))
-> m (QualName VName, PatType)
-> m (QualName VName, PatType, [Name])
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> SrcLoc -> QualName Name -> m (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
loc ([Name] -> Name -> QualName Name
forall vn. [vn] -> vn -> QualName vn
QualName [Name]
qs Name
name)) m (QualName VName, PatType, [Name])
-> (b -> m (QualName VName, PatType, [Name]))
-> m (QualName VName, PatType, [Name])
forall e (m :: * -> *) a.
MonadError e m =>
m a -> (e -> m a) -> m a
`catchError` [Name] -> Name -> b -> m (QualName VName, PatType, [Name])
notFound [Name]
qs Name
name

    whenFound :: (a, b) -> (a, b, [a])
whenFound (a
qn', b
t) = (a
qn', b
t, [])

    notFound :: [Name] -> Name -> b -> m (QualName VName, PatType, [Name])
notFound [Name]
qs Name
name b
err
      | [Name] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [Name]
qs = b -> m (QualName VName, PatType, [Name])
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError b
err
      | Bool
otherwise = do
        (QualName VName
qn', PatType
t, [Name]
fields) <-
          [Name] -> Name -> m (QualName VName, PatType, [Name])
findRootVar ([Name] -> [Name]
forall a. [a] -> [a]
init [Name]
qs) ([Name] -> Name
forall a. [a] -> a
last [Name]
qs)
            m (QualName VName, PatType, [Name])
-> (b -> m (QualName VName, PatType, [Name]))
-> m (QualName VName, PatType, [Name])
forall e (m :: * -> *) a.
MonadError e m =>
m a -> (e -> m a) -> m a
`catchError` m (QualName VName, PatType, [Name])
-> b -> m (QualName VName, PatType, [Name])
forall a b. a -> b -> a
const (b -> m (QualName VName, PatType, [Name])
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError b
err)
        (QualName VName, PatType, [Name])
-> m (QualName VName, PatType, [Name])
forall (m :: * -> *) a. Monad m => a -> m a
return (QualName VName
qn', PatType
t, [Name]
fields [Name] -> [Name] -> [Name]
forall a. [a] -> [a] -> [a]
++ [Name
name])

    checkField :: Exp -> Name -> TermTypeM Exp
checkField Exp
e Name
k = do
      PatType
t <- Exp -> TermTypeM PatType
expType Exp
e
      let usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage SrcLoc
loc (String -> Usage) -> String -> Usage
forall a b. (a -> b) -> a -> b
$ String
"projection of field " String -> ShowS
forall a. [a] -> [a] -> [a]
++ ShowS
quote (Name -> String
forall a. Pretty a => a -> String
pretty Name
k)
      PatType
kt <- Usage -> Name -> PatType -> TermTypeM PatType
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> PatType -> m PatType
mustHaveField Usage
usage Name
k PatType
t
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Name -> Exp -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
Name -> ExpBase f vn -> f PatType -> SrcLoc -> ExpBase f vn
Project Name
k Exp
e (PatType -> Info PatType
forall a. a -> Info a
Info PatType
kt) SrcLoc
loc
checkExp (Negate UncheckedExp
arg SrcLoc
loc) = do
  Exp
arg' <- String -> [PrimType] -> Exp -> TermTypeM Exp
require String
"numeric negation" [PrimType]
anyNumberType (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
arg
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Exp -> SrcLoc -> Exp
forall (f :: * -> *) vn. ExpBase f vn -> SrcLoc -> ExpBase f vn
Negate Exp
arg' SrcLoc
loc
checkExp (Not UncheckedExp
arg SrcLoc
loc) = do
  Exp
arg' <- String -> [PrimType] -> Exp -> TermTypeM Exp
require String
"logical negation" (PrimType
Bool PrimType -> [PrimType] -> [PrimType]
forall a. a -> [a] -> [a]
: [PrimType]
anyIntType) (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
arg
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Exp -> SrcLoc -> Exp
forall (f :: * -> *) vn. ExpBase f vn -> SrcLoc -> ExpBase f vn
Not Exp
arg' SrcLoc
loc
checkExp e :: UncheckedExp
e@(AppExp Apply {} NoInfo AppRes
_) = (Exp, ApplyOp) -> Exp
forall a b. (a, b) -> a
fst ((Exp, ApplyOp) -> Exp)
-> TermTypeM (Exp, ApplyOp) -> TermTypeM Exp
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> UncheckedExp -> TermTypeM (Exp, ApplyOp)
checkApplyExp UncheckedExp
e
checkExp (AppExp (LetPat [SizeBinder Name]
sizes UncheckedPat
pat UncheckedExp
e UncheckedExp
body SrcLoc
loc) NoInfo AppRes
_) =
  TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e) ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
e' [Occurence]
e_occs -> do
    -- Not technically an ascription, but we want the pattern to have
    -- exactly the type of 'e'.
    PatType
t <- Exp -> TermTypeM PatType
expType Exp
e'
    case [Occurence] -> Maybe Occurence
anyConsumption [Occurence]
e_occs of
      Just Occurence
c ->
        let msg :: String
msg = String
"type computed with consumption at " String -> ShowS
forall a. [a] -> [a] -> [a]
++ SrcLoc -> String
forall a. Located a => a -> String
locStr (Occurence -> SrcLoc
location Occurence
c)
         in Usage -> String -> PatType -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> String -> TypeBase dim as -> m ()
zeroOrderType (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"consumption in right-hand side of 'let'-binding") String
msg PatType
t
      Maybe Occurence
_ -> () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

    TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
incLevel (TermTypeM Exp -> TermTypeM Exp)
-> (([SizeBinder VName] -> TermTypeM Exp) -> TermTypeM Exp)
-> ([SizeBinder VName] -> TermTypeM Exp)
-> TermTypeM Exp
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [SizeBinder Name]
-> ([SizeBinder VName] -> TermTypeM Exp) -> TermTypeM Exp
forall a.
[SizeBinder Name]
-> ([SizeBinder VName] -> TermTypeM a) -> TermTypeM a
bindingSizes [SizeBinder Name]
sizes (([SizeBinder VName] -> TermTypeM Exp) -> TermTypeM Exp)
-> ([SizeBinder VName] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \[SizeBinder VName]
sizes' ->
      [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM Exp)
-> TermTypeM Exp
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [SizeBinder VName]
sizes' UncheckedPat
pat (PatType -> InferredType
Ascribed PatType
t) ((Pat -> TermTypeM Exp) -> TermTypeM Exp)
-> (Pat -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Pat
pat' -> do
        Exp
body' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
body
        (PatType
body_t, [VName]
retext) <-
          SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType SrcLoc
loc ([SizeBinder VName] -> Map VName Ident
sizesMap [SizeBinder VName]
sizes' Map VName Ident -> Map VName Ident -> Map VName Ident
forall a. Semigroup a => a -> a -> a
<> Pat -> Map VName Ident
forall (f :: * -> *).
Functor f =>
PatBase f VName -> Map VName (IdentBase f VName)
patternMap Pat
pat') (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
body'

        Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp ([SizeBinder VName]
-> Pat -> Exp -> Exp -> SrcLoc -> AppExpBase Info VName
forall (f :: * -> *) vn.
[SizeBinder vn]
-> PatBase f vn
-> ExpBase f vn
-> ExpBase f vn
-> SrcLoc
-> AppExpBase f vn
LetPat [SizeBinder VName]
sizes' Pat
pat' Exp
e' Exp
body' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
body_t [VName]
retext)
  where
    sizesMap :: [SizeBinder VName] -> Map VName Ident
sizesMap = (SizeBinder VName -> Map VName Ident)
-> [SizeBinder VName] -> Map VName Ident
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap SizeBinder VName -> Map VName Ident
forall vn. SizeBinder vn -> Map vn (IdentBase Info vn)
onSize
    onSize :: SizeBinder vn -> Map vn (IdentBase Info vn)
onSize SizeBinder vn
size =
      vn -> IdentBase Info vn -> Map vn (IdentBase Info vn)
forall k a. k -> a -> Map k a
M.singleton (SizeBinder vn -> vn
forall vn. SizeBinder vn -> vn
sizeName SizeBinder vn
size) (IdentBase Info vn -> Map vn (IdentBase Info vn))
-> IdentBase Info vn -> Map vn (IdentBase Info vn)
forall a b. (a -> b) -> a -> b
$
        vn -> Info PatType -> SrcLoc -> IdentBase Info vn
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident (SizeBinder vn -> vn
forall vn. SizeBinder vn -> vn
sizeName SizeBinder vn
size) (PatType -> Info PatType
forall a. a -> Info a
Info (ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int64)) (SizeBinder vn -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf SizeBinder vn
size)
checkExp (AppExp (LetFun Name
name ([UncheckedTypeParam]
tparams, [UncheckedPat]
params, Maybe (TypeExp Name)
maybe_retdecl, NoInfo StructType
NoInfo, UncheckedExp
e) UncheckedExp
body SrcLoc
loc) NoInfo AppRes
_) =
  TermTypeM
  ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
   Exp)
-> (([TypeParam], [Pat], Maybe (TypeExp VName), StructType,
     [VName], Exp)
    -> [Occurence] -> TermTypeM Exp)
-> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially ((Name, Maybe (TypeExp Name), [UncheckedTypeParam], [UncheckedPat],
 UncheckedExp, SrcLoc)
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
checkBinding (Name
name, Maybe (TypeExp Name)
maybe_retdecl, [UncheckedTypeParam]
tparams, [UncheckedPat]
params, UncheckedExp
e, SrcLoc
loc)) ((([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
   Exp)
  -> [Occurence] -> TermTypeM Exp)
 -> TermTypeM Exp)
-> (([TypeParam], [Pat], Maybe (TypeExp VName), StructType,
     [VName], Exp)
    -> [Occurence] -> TermTypeM Exp)
-> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
    \([TypeParam]
tparams', [Pat]
params', Maybe (TypeExp VName)
maybe_retdecl', StructType
rettype, [VName]
_, Exp
e') [Occurence]
closure -> do
      Aliasing
closure' <- [Pat] -> [Occurence] -> TermTypeM Aliasing
lexicalClosure [Pat]
params' [Occurence]
closure

      [(Namespace, Name)] -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a.
MonadTypeChecker m =>
[(Namespace, Name)] -> m a -> m a
bindSpaced [(Namespace
Term, Name
name)] (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ do
        VName
name' <- Namespace -> Name -> SrcLoc -> TermTypeM VName
forall (m :: * -> *).
MonadTypeChecker m =>
Namespace -> Name -> SrcLoc -> m VName
checkName Namespace
Term Name
name SrcLoc
loc

        let arrow :: (PName, TypeBase dim ()) -> TypeBase dim () -> TypeBase dim ()
arrow (PName
xp, TypeBase dim ()
xt) TypeBase dim ()
yt = ScalarTypeBase dim () -> TypeBase dim ()
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase dim () -> TypeBase dim ())
-> ScalarTypeBase dim () -> TypeBase dim ()
forall a b. (a -> b) -> a -> b
$ ()
-> PName
-> TypeBase dim ()
-> TypeBase dim ()
-> ScalarTypeBase dim ()
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow () PName
xp TypeBase dim ()
xt TypeBase dim ()
yt
            ftype :: StructType
ftype = (Pat -> StructType -> StructType)
-> StructType -> [Pat] -> StructType
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr ((PName, StructType) -> StructType -> StructType
forall dim.
(PName, TypeBase dim ()) -> TypeBase dim () -> TypeBase dim ()
arrow ((PName, StructType) -> StructType -> StructType)
-> (Pat -> (PName, StructType)) -> Pat -> StructType -> StructType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Pat -> (PName, StructType)
patternParam) StructType
rettype [Pat]
params'
            entry :: ValBinding
entry = Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
Local [TypeParam]
tparams' (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$ StructType
ftype StructType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` Aliasing
closure'
            bindF :: TermScope -> TermScope
bindF TermScope
scope =
              TermScope
scope
                { scopeVtable :: Map VName ValBinding
scopeVtable =
                    VName -> ValBinding -> Map VName ValBinding -> Map VName ValBinding
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
name' ValBinding
entry (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable TermScope
scope,
                  scopeNameMap :: NameMap
scopeNameMap =
                    (Namespace, Name) -> QualName VName -> NameMap -> NameMap
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert (Namespace
Term, Name
name) (VName -> QualName VName
forall v. v -> QualName v
qualName VName
name') (NameMap -> NameMap) -> NameMap -> NameMap
forall a b. (a -> b) -> a -> b
$
                      TermScope -> NameMap
scopeNameMap TermScope
scope
                }
        Exp
body' <- (TermScope -> TermScope) -> TermTypeM Exp -> TermTypeM Exp
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope TermScope -> TermScope
bindF (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
body

        -- We fake an ident here, but it's OK as it can't be a size
        -- anyway.
        let fake_ident :: Ident
fake_ident = VName -> Info PatType -> SrcLoc -> Ident
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident VName
name' (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
ftype) SrcLoc
forall a. Monoid a => a
mempty
        (PatType
body_t, [VName]
ext) <-
          SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType SrcLoc
loc (VName -> Ident -> Map VName Ident
forall k a. k -> a -> Map k a
M.singleton VName
name' Ident
fake_ident)
            (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
body'

        Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
          AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp
            ( VName
-> ([TypeParam], [Pat], Maybe (TypeExp VName), Info StructType,
    Exp)
-> Exp
-> SrcLoc
-> AppExpBase Info VName
forall (f :: * -> *) vn.
vn
-> ([TypeParamBase vn], [PatBase f vn], Maybe (TypeExp vn),
    f StructType, ExpBase f vn)
-> ExpBase f vn
-> SrcLoc
-> AppExpBase f vn
LetFun
                VName
name'
                ([TypeParam]
tparams', [Pat]
params', Maybe (TypeExp VName)
maybe_retdecl', StructType -> Info StructType
forall a. a -> Info a
Info StructType
rettype, Exp
e')
                Exp
body'
                SrcLoc
loc
            )
            (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
body_t [VName]
ext)
checkExp (AppExp (LetWith IdentBase NoInfo Name
dest IdentBase NoInfo Name
src SliceBase NoInfo Name
slice UncheckedExp
ve UncheckedExp
body SrcLoc
loc) NoInfo AppRes
_) =
  TermTypeM Ident
-> (Ident -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (IdentBase NoInfo Name -> TermTypeM Ident
checkIdent IdentBase NoInfo Name
src) ((Ident -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Ident -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Ident
src' [Occurence]
_ -> do
    Slice
slice' <- SliceBase NoInfo Name -> TermTypeM Slice
checkSlice SliceBase NoInfo Name
slice
    (StructType
t, StructType
_) <- SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType (IdentBase NoInfo Name -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf IdentBase NoInfo Name
src) String
"src" (Int -> TermTypeM (StructType, StructType))
-> Int -> TermTypeM (StructType, StructType)
forall a b. (a -> b) -> a -> b
$ Slice -> Int
sliceDims Slice
slice'
    Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"type of target array") StructType
t (StructType -> TermTypeM ()) -> StructType -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType) -> PatType -> StructType
forall a b. (a -> b) -> a -> b
$ Info PatType -> PatType
forall a. Info a -> a
unInfo (Info PatType -> PatType) -> Info PatType -> PatType
forall a b. (a -> b) -> a -> b
$ Ident -> Info PatType
forall (f :: * -> *) vn. IdentBase f vn -> f PatType
identType Ident
src'

    -- Need the fully normalised type here to get the proper aliasing information.
    PatType
src_t <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ Info PatType -> PatType
forall a. Info a -> a
unInfo (Info PatType -> PatType) -> Info PatType -> PatType
forall a b. (a -> b) -> a -> b
$ Ident -> Info PatType
forall (f :: * -> *) vn. IdentBase f vn -> f PatType
identType Ident
src'

    (StructType
elemt, [VName]
_) <- Maybe (SrcLoc, Rigidity)
-> Slice -> StructType -> TermTypeM (StructType, [VName])
forall as.
Maybe (SrcLoc, Rigidity)
-> Slice
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
sliceShape ((SrcLoc, Rigidity) -> Maybe (SrcLoc, Rigidity)
forall a. a -> Maybe a
Just (SrcLoc
loc, Rigidity
Nonrigid)) Slice
slice' (StructType -> TermTypeM (StructType, [VName]))
-> TermTypeM StructType -> TermTypeM (StructType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully StructType
t

    Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
src_t) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Doc -> TermTypeM ()
forall (m :: * -> *) b. MonadTypeChecker m => SrcLoc -> Doc -> m b
notConsumable SrcLoc
loc (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Doc -> Doc
pquote (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ Name -> Doc
forall v. IsName v => v -> Doc
pprName (Name -> Doc) -> Name -> Doc
forall a b. (a -> b) -> a -> b
$ IdentBase NoInfo Name -> Name
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName IdentBase NoInfo Name
src

    TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (String -> StructType -> Exp -> TermTypeM Exp
unifies String
"type of target array" (StructType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct StructType
elemt) (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
ve) ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
ve' [Occurence]
_ -> do
      PatType
ve_t <- Exp -> TermTypeM PatType
expTypeFully Exp
ve'
      Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (VName -> Alias
AliasBound (Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
src') Alias -> Aliasing -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
ve_t) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        IdentBase NoInfo Name -> UncheckedExp -> SrcLoc -> TermTypeM ()
forall arr src a.
(Pretty arr, Pretty src) =>
arr -> src -> SrcLoc -> TermTypeM a
badLetWithValue IdentBase NoInfo Name
src UncheckedExp
ve SrcLoc
loc

      IdentBase NoInfo Name
-> PatType -> (Ident -> TermTypeM Exp) -> TermTypeM Exp
forall a.
IdentBase NoInfo Name
-> PatType -> (Ident -> TermTypeM a) -> TermTypeM a
bindingIdent IdentBase NoInfo Name
dest (PatType
src_t PatType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` Aliasing
forall a. Set a
S.empty) ((Ident -> TermTypeM Exp) -> TermTypeM Exp)
-> (Ident -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Ident
dest' -> do
        Exp
body' <- Ident -> TermTypeM Exp -> TermTypeM Exp
forall a. Ident -> TermTypeM a -> TermTypeM a
consuming Ident
src' (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
body
        (PatType
body_t, [VName]
ext) <-
          SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType SrcLoc
loc (VName -> Ident -> Map VName Ident
forall k a. k -> a -> Map k a
M.singleton (Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
dest') Ident
dest')
            (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
body'
        Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp (Ident
-> Ident -> Slice -> Exp -> Exp -> SrcLoc -> AppExpBase Info VName
forall (f :: * -> *) vn.
IdentBase f vn
-> IdentBase f vn
-> SliceBase f vn
-> ExpBase f vn
-> ExpBase f vn
-> SrcLoc
-> AppExpBase f vn
LetWith Ident
dest' Ident
src' Slice
slice' Exp
ve' Exp
body' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
body_t [VName]
ext)
checkExp (Update UncheckedExp
src SliceBase NoInfo Name
slice UncheckedExp
ve SrcLoc
loc) = do
  Slice
slice' <- SliceBase NoInfo Name -> TermTypeM Slice
checkSlice SliceBase NoInfo Name
slice
  (StructType
t, StructType
_) <- SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
src) String
"src" (Int -> TermTypeM (StructType, StructType))
-> Int -> TermTypeM (StructType, StructType)
forall a b. (a -> b) -> a -> b
$ Slice -> Int
sliceDims Slice
slice'
  (StructType
elemt, [VName]
_) <- Maybe (SrcLoc, Rigidity)
-> Slice -> StructType -> TermTypeM (StructType, [VName])
forall as.
Maybe (SrcLoc, Rigidity)
-> Slice
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
sliceShape ((SrcLoc, Rigidity) -> Maybe (SrcLoc, Rigidity)
forall a. a -> Maybe a
Just (SrcLoc
loc, Rigidity
Nonrigid)) Slice
slice' (StructType -> TermTypeM (StructType, [VName]))
-> TermTypeM StructType -> TermTypeM (StructType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully StructType
t

  TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
ve TermTypeM Exp -> (Exp -> TermTypeM Exp) -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= String -> StructType -> Exp -> TermTypeM Exp
unifies String
"type of target array" StructType
elemt) ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
ve' [Occurence]
_ ->
    TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
src TermTypeM Exp -> (Exp -> TermTypeM Exp) -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= String -> StructType -> Exp -> TermTypeM Exp
unifies String
"type of target array" StructType
t) ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
src' [Occurence]
_ -> do
      PatType
src_t <- Exp -> TermTypeM PatType
expTypeFully Exp
src'

      Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
src_t) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ SrcLoc -> Doc -> TermTypeM ()
forall (m :: * -> *) b. MonadTypeChecker m => SrcLoc -> Doc -> m b
notConsumable SrcLoc
loc (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Doc -> Doc
pquote (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> Doc
forall a. Pretty a => a -> Doc
ppr UncheckedExp
src

      let src_als :: Aliasing
src_als = PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
src_t
      PatType
ve_t <- Exp -> TermTypeM PatType
expTypeFully Exp
ve'
      Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (Aliasing -> Bool
forall a. Set a -> Bool
S.null (Aliasing -> Bool) -> Aliasing -> Bool
forall a b. (a -> b) -> a -> b
$ Aliasing
src_als Aliasing -> Aliasing -> Aliasing
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
ve_t) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> UncheckedExp -> SrcLoc -> TermTypeM ()
forall arr src a.
(Pretty arr, Pretty src) =>
arr -> src -> SrcLoc -> TermTypeM a
badLetWithValue UncheckedExp
src UncheckedExp
ve SrcLoc
loc

      SrcLoc -> Aliasing -> TermTypeM ()
consume SrcLoc
loc Aliasing
src_als
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Exp -> Slice -> Exp -> SrcLoc -> Exp
forall (f :: * -> *) vn.
ExpBase f vn
-> SliceBase f vn -> ExpBase f vn -> SrcLoc -> ExpBase f vn
Update Exp
src' Slice
slice' Exp
ve' SrcLoc
loc

-- Record updates are a bit hacky, because we do not have row typing
-- (yet?).  For now, we only permit record updates where we know the
-- full type up to the field we are updating.
checkExp (RecordUpdate UncheckedExp
src [Name]
fields UncheckedExp
ve NoInfo PatType
NoInfo SrcLoc
loc) = do
  Exp
src' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
src
  Exp
ve' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
ve
  PatType
a <- Exp -> TermTypeM PatType
expTypeFully Exp
src'
  (PatType -> Name -> TermTypeM PatType)
-> PatType -> [Name] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m ()
foldM_ ((Name -> PatType -> TermTypeM PatType)
-> PatType -> Name -> TermTypeM PatType
forall a b c. (a -> b -> c) -> b -> a -> c
flip ((Name -> PatType -> TermTypeM PatType)
 -> PatType -> Name -> TermTypeM PatType)
-> (Name -> PatType -> TermTypeM PatType)
-> PatType
-> Name
-> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ Usage -> Name -> PatType -> TermTypeM PatType
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> PatType -> m PatType
mustHaveField Usage
usage) PatType
a [Name]
fields
  PatType
ve_t <- Exp -> TermTypeM PatType
expType Exp
ve'
  PatType
updated_t <- [Name] -> PatType -> PatType -> TermTypeM PatType
forall as.
[Name]
-> TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as)
updateField [Name]
fields PatType
ve_t (PatType -> TermTypeM PatType)
-> TermTypeM PatType -> TermTypeM PatType
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
src'
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Exp -> [Name] -> Exp -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
ExpBase f vn
-> [Name] -> ExpBase f vn -> f PatType -> SrcLoc -> ExpBase f vn
RecordUpdate Exp
src' [Name]
fields Exp
ve' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
updated_t) SrcLoc
loc
  where
    usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"record update"
    updateField :: [Name]
-> TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as)
updateField [] TypeBase (DimDecl VName) as
ve_t TypeBase (DimDecl VName) as
src_t = do
      (TypeBase (DimDecl VName) as
src_t', [VName]
_) <- SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"any" Rigidity
Nonrigid (TypeBase (DimDecl VName) as
 -> TermTypeM (TypeBase (DimDecl VName) as, [VName]))
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
forall a b. (a -> b) -> a -> b
$ TypeBase (DimDecl VName) as -> TypeBase (DimDecl VName) as
forall vn as. TypeBase (DimDecl vn) as -> TypeBase (DimDecl vn) as
anySizes TypeBase (DimDecl VName) as
src_t
      Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure ([Name] -> StructType -> StructType -> Checking
CheckingRecordUpdate [Name]
fields (TypeBase (DimDecl VName) as -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase (DimDecl VName) as
src_t') (TypeBase (DimDecl VName) as -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase (DimDecl VName) as
ve_t)) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify Usage
usage (TypeBase (DimDecl VName) as -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase (DimDecl VName) as
src_t') (TypeBase (DimDecl VName) as -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase (DimDecl VName) as
ve_t)
      -- Important that we return ve_t so that we get the right aliases.
      TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as)
forall (f :: * -> *) a. Applicative f => a -> f a
pure TypeBase (DimDecl VName) as
ve_t
    updateField (Name
f : [Name]
fs) TypeBase (DimDecl VName) as
ve_t (Scalar (Record Map Name (TypeBase (DimDecl VName) as)
m))
      | Just TypeBase (DimDecl VName) as
f_t <- Name
-> Map Name (TypeBase (DimDecl VName) as)
-> Maybe (TypeBase (DimDecl VName) as)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup Name
f Map Name (TypeBase (DimDecl VName) as)
m = do
        TypeBase (DimDecl VName) as
f_t' <- [Name]
-> TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as)
updateField [Name]
fs TypeBase (DimDecl VName) as
ve_t TypeBase (DimDecl VName) as
f_t
        TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (TypeBase (DimDecl VName) as
 -> TermTypeM (TypeBase (DimDecl VName) as))
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as)
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) as -> TypeBase (DimDecl VName) as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) as -> TypeBase (DimDecl VName) as)
-> ScalarTypeBase (DimDecl VName) as -> TypeBase (DimDecl VName) as
forall a b. (a -> b) -> a -> b
$ Map Name (TypeBase (DimDecl VName) as)
-> ScalarTypeBase (DimDecl VName) as
forall dim as. Map Name (TypeBase dim as) -> ScalarTypeBase dim as
Record (Map Name (TypeBase (DimDecl VName) as)
 -> ScalarTypeBase (DimDecl VName) as)
-> Map Name (TypeBase (DimDecl VName) as)
-> ScalarTypeBase (DimDecl VName) as
forall a b. (a -> b) -> a -> b
$ Name
-> TypeBase (DimDecl VName) as
-> Map Name (TypeBase (DimDecl VName) as)
-> Map Name (TypeBase (DimDecl VName) as)
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert Name
f TypeBase (DimDecl VName) as
f_t' Map Name (TypeBase (DimDecl VName) as)
m
    updateField [Name]
_ TypeBase (DimDecl VName) as
_ TypeBase (DimDecl VName) as
_ =
      SrcLoc -> Notes -> Doc -> TermTypeM (TypeBase (DimDecl VName) as)
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM (TypeBase (DimDecl VName) as))
-> Doc -> TermTypeM (TypeBase (DimDecl VName) as)
forall a b. (a -> b) -> a -> b
$
        Doc
"Full type of"
          Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (UncheckedExp -> Doc
forall a. Pretty a => a -> Doc
ppr UncheckedExp
src)
          Doc -> Doc -> Doc
</> String -> Doc
textwrap String
" is not known at this point.  Add a size annotation to the original record to disambiguate."

--
checkExp (AppExp (Index UncheckedExp
e SliceBase NoInfo Name
slice SrcLoc
loc) NoInfo AppRes
_) = do
  Slice
slice' <- SliceBase NoInfo Name -> TermTypeM Slice
checkSlice SliceBase NoInfo Name
slice
  (StructType
t, StructType
_) <- SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType SrcLoc
loc String
"e" (Int -> TermTypeM (StructType, StructType))
-> Int -> TermTypeM (StructType, StructType)
forall a b. (a -> b) -> a -> b
$ Slice -> Int
sliceDims Slice
slice'
  Exp
e' <- String -> StructType -> Exp -> TermTypeM Exp
unifies String
"being indexed at" StructType
t (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
  -- XXX, the RigidSlice here will be overridden in sliceShape with a proper value.
  (PatType
t', [VName]
retext) <-
    Maybe (SrcLoc, Rigidity)
-> Slice -> PatType -> TermTypeM (PatType, [VName])
forall as.
Maybe (SrcLoc, Rigidity)
-> Slice
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
sliceShape ((SrcLoc, Rigidity) -> Maybe (SrcLoc, Rigidity)
forall a. a -> Maybe a
Just (SrcLoc
loc, RigidSource -> Rigidity
Rigid (Maybe (DimDecl VName) -> String -> RigidSource
RigidSlice Maybe (DimDecl VName)
forall a. Maybe a
Nothing String
""))) Slice
slice'
      (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
e'

  -- Remove aliases if the result is an overloaded type, because that
  -- will certainly not be aliased.
  PatType
t'' <- PatType -> TermTypeM PatType
noAliasesIfOverloaded PatType
t'

  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp (Exp -> Slice -> SrcLoc -> AppExpBase Info VName
forall (f :: * -> *) vn.
ExpBase f vn -> SliceBase f vn -> SrcLoc -> AppExpBase f vn
Index Exp
e' Slice
slice' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
t'' [VName]
retext)
checkExp (Assert UncheckedExp
e1 UncheckedExp
e2 NoInfo String
NoInfo SrcLoc
loc) = do
  Exp
e1' <- String -> [PrimType] -> Exp -> TermTypeM Exp
require String
"being asserted" [PrimType
Bool] (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e1
  Exp
e2' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e2
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Exp -> Exp -> Info String -> SrcLoc -> Exp
forall (f :: * -> *) vn.
ExpBase f vn -> ExpBase f vn -> f String -> SrcLoc -> ExpBase f vn
Assert Exp
e1' Exp
e2' (String -> Info String
forall a. a -> Info a
Info (UncheckedExp -> String
forall a. Pretty a => a -> String
pretty UncheckedExp
e1)) SrcLoc
loc
checkExp (Lambda [UncheckedPat]
params UncheckedExp
body Maybe (TypeExp Name)
rettype_te NoInfo (Aliasing, StructType)
NoInfo SrcLoc
loc) =
  TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
removeSeminullOccurences (TermTypeM Exp -> TermTypeM Exp)
-> (([TypeParam] -> [Pat] -> TermTypeM Exp) -> TermTypeM Exp)
-> ([TypeParam] -> [Pat] -> TermTypeM Exp)
-> TermTypeM Exp
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
noUnique (TermTypeM Exp -> TermTypeM Exp)
-> (([TypeParam] -> [Pat] -> TermTypeM Exp) -> TermTypeM Exp)
-> ([TypeParam] -> [Pat] -> TermTypeM Exp)
-> TermTypeM Exp
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
incLevel (TermTypeM Exp -> TermTypeM Exp)
-> (([TypeParam] -> [Pat] -> TermTypeM Exp) -> TermTypeM Exp)
-> ([TypeParam] -> [Pat] -> TermTypeM Exp)
-> TermTypeM Exp
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [UncheckedTypeParam]
-> [UncheckedPat]
-> ([TypeParam] -> [Pat] -> TermTypeM Exp)
-> TermTypeM Exp
forall a.
[UncheckedTypeParam]
-> [UncheckedPat]
-> ([TypeParam] -> [Pat] -> TermTypeM a)
-> TermTypeM a
bindingParams [] [UncheckedPat]
params (([TypeParam] -> [Pat] -> TermTypeM Exp) -> TermTypeM Exp)
-> ([TypeParam] -> [Pat] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \[TypeParam]
_ [Pat]
params' -> do
    Maybe (TypeExp VName, StructType, Liftedness)
rettype_checked <- (TypeExp Name -> TermTypeM (TypeExp VName, StructType, Liftedness))
-> Maybe (TypeExp Name)
-> TermTypeM (Maybe (TypeExp VName, StructType, Liftedness))
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse TypeExp Name -> TermTypeM (TypeExp VName, StructType, Liftedness)
forall (m :: * -> *).
MonadTypeChecker m =>
TypeExp Name -> m (TypeExp VName, StructType, Liftedness)
checkTypeExp Maybe (TypeExp Name)
rettype_te
    let declared_rettype :: Maybe StructType
declared_rettype =
          case Maybe (TypeExp VName, StructType, Liftedness)
rettype_checked of
            Just (TypeExp VName
_, StructType
st, Liftedness
_) -> StructType -> Maybe StructType
forall a. a -> Maybe a
Just StructType
st
            Maybe (TypeExp VName, StructType, Liftedness)
Nothing -> Maybe StructType
forall a. Maybe a
Nothing
    (Exp
body', [Occurence]
closure) <-
      TermTypeM Exp -> TermTypeM (Exp, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences (TermTypeM Exp -> TermTypeM (Exp, [Occurence]))
-> TermTypeM Exp -> TermTypeM (Exp, [Occurence])
forall a b. (a -> b) -> a -> b
$ [Pat]
-> UncheckedExp -> Maybe StructType -> SrcLoc -> TermTypeM Exp
checkFunBody [Pat]
params' UncheckedExp
body Maybe StructType
declared_rettype SrcLoc
loc
    PatType
body_t <- Exp -> TermTypeM PatType
expTypeFully Exp
body'

    [Pat]
params'' <- (Pat -> TermTypeM Pat) -> [Pat] -> TermTypeM [Pat]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM Pat -> TermTypeM Pat
forall e. ASTMappable e => e -> TermTypeM e
updateTypes [Pat]
params'

    (Maybe (TypeExp VName)
rettype', StructType
rettype_st) <-
      case Maybe (TypeExp VName, StructType, Liftedness)
rettype_checked of
        Just (TypeExp VName
te, StructType
st, Liftedness
_) ->
          (Maybe (TypeExp VName), StructType)
-> TermTypeM (Maybe (TypeExp VName), StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return (TypeExp VName -> Maybe (TypeExp VName)
forall a. a -> Maybe a
Just TypeExp VName
te, StructType
st)
        Maybe (TypeExp VName, StructType, Liftedness)
Nothing -> do
          StructType
ret <-
            [Pat] -> StructType -> TermTypeM StructType
forall (m :: * -> *).
MonadUnify m =>
[Pat] -> StructType -> m StructType
inferReturnSizes [Pat]
params'' (StructType -> TermTypeM StructType)
-> StructType -> TermTypeM StructType
forall a b. (a -> b) -> a -> b
$
              PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType) -> PatType -> StructType
forall a b. (a -> b) -> a -> b
$
                [Pat] -> PatType -> PatType
inferReturnUniqueness [Pat]
params'' PatType
body_t
          (Maybe (TypeExp VName), StructType)
-> TermTypeM (Maybe (TypeExp VName), StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (TypeExp VName)
forall a. Maybe a
Nothing, StructType
ret)

    [Pat] -> PatType -> SrcLoc -> TermTypeM ()
checkGlobalAliases [Pat]
params' PatType
body_t SrcLoc
loc
    Maybe Name -> [Pat] -> TermTypeM ()
verifyFunctionParams Maybe Name
forall a. Maybe a
Nothing [Pat]
params'

    Aliasing
closure' <- [Pat] -> [Occurence] -> TermTypeM Aliasing
lexicalClosure [Pat]
params'' [Occurence]
closure

    Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ [Pat]
-> Exp
-> Maybe (TypeExp VName)
-> Info (Aliasing, StructType)
-> SrcLoc
-> Exp
forall (f :: * -> *) vn.
[PatBase f vn]
-> ExpBase f vn
-> Maybe (TypeExp vn)
-> f (Aliasing, StructType)
-> SrcLoc
-> ExpBase f vn
Lambda [Pat]
params'' Exp
body' Maybe (TypeExp VName)
rettype' ((Aliasing, StructType) -> Info (Aliasing, StructType)
forall a. a -> Info a
Info (Aliasing
closure', StructType
rettype_st)) SrcLoc
loc
  where
    -- Inferring the sizes of the return type of a lambda is a lot
    -- like let-generalisation.  We wish to remove any rigid sizes
    -- that were created when checking the body, except for those that
    -- are visible in types that existed before we entered the body,
    -- are parameters, or are used in parameters.
    inferReturnSizes :: [Pat] -> StructType -> m StructType
inferReturnSizes [Pat]
params' StructType
ret = do
      Int
cur_lvl <- m Int
forall (m :: * -> *). MonadUnify m => m Int
curLevel
      let named :: (PName, b) -> Maybe VName
named (Named VName
x, b
_) = VName -> Maybe VName
forall a. a -> Maybe a
Just VName
x
          named (PName
Unnamed, b
_) = Maybe VName
forall a. Maybe a
Nothing
          param_names :: [VName]
param_names = (Pat -> Maybe VName) -> [Pat] -> [VName]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe ((PName, StructType) -> Maybe VName
forall b. (PName, b) -> Maybe VName
named ((PName, StructType) -> Maybe VName)
-> (Pat -> (PName, StructType)) -> Pat -> Maybe VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Pat -> (PName, StructType)
patternParam) [Pat]
params'
          pos_sizes :: Names
pos_sizes =
            StructType -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNamesPos ([StructType] -> StructType -> StructType
forall as dim.
Monoid as =>
[TypeBase dim as] -> TypeBase dim as -> TypeBase dim as
foldFunType ((Pat -> StructType) -> [Pat] -> [StructType]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> StructType
patternStructType [Pat]
params') StructType
ret)
          hide :: VName -> (Int, b) -> Bool
hide VName
k (Int
lvl, b
_) =
            Int
lvl Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
cur_lvl Bool -> Bool -> Bool
&& VName
k VName -> [VName] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`notElem` [VName]
param_names Bool -> Bool -> Bool
&& VName
k VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.notMember` Names
pos_sizes

      Names
hidden_sizes <-
        [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList ([VName] -> Names)
-> (Constraints -> [VName]) -> Constraints -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Constraints -> [VName]
forall k a. Map k a -> [k]
M.keys (Constraints -> [VName])
-> (Constraints -> Constraints) -> Constraints -> [VName]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (VName -> (Int, Constraint) -> Bool) -> Constraints -> Constraints
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey VName -> (Int, Constraint) -> Bool
forall b. VName -> (Int, b) -> Bool
hide (Constraints -> Names) -> m Constraints -> m Names
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> m Constraints
forall (m :: * -> *). MonadUnify m => m Constraints
getConstraints

      let onDim :: DimDecl VName -> DimDecl VName
onDim (NamedDim QualName VName
name)
            | Bool -> Bool
not (QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
name VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
hidden_sizes) = QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim QualName VName
name
            | Bool
otherwise = Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim (Maybe VName -> DimDecl VName) -> Maybe VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> Maybe VName
forall a. a -> Maybe a
Just (VName -> Maybe VName) -> VName -> Maybe VName
forall a b. (a -> b) -> a -> b
$ QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
name
          onDim DimDecl VName
d = DimDecl VName
d

      StructType -> m StructType
forall (m :: * -> *) a. Monad m => a -> m a
return (StructType -> m StructType) -> StructType -> m StructType
forall a b. (a -> b) -> a -> b
$ (DimDecl VName -> DimDecl VName) -> StructType -> StructType
forall (p :: * -> * -> *) a b c.
Bifunctor p =>
(a -> b) -> p a c -> p b c
first DimDecl VName -> DimDecl VName
onDim StructType
ret
checkExp (OpSection QualName Name
op NoInfo PatType
_ SrcLoc
loc) = do
  (QualName VName
op', PatType
ftype) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
loc QualName Name
op
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ QualName VName -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
QualName vn -> f PatType -> SrcLoc -> ExpBase f vn
OpSection QualName VName
op' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
ftype) SrcLoc
loc
checkExp (OpSectionLeft QualName Name
op NoInfo PatType
_ UncheckedExp
e (NoInfo (PName, StructType, Maybe VName),
 NoInfo (PName, StructType))
_ (NoInfo PatType, NoInfo [VName])
_ SrcLoc
loc) = do
  (QualName VName
op', PatType
ftype) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
loc QualName Name
op
  Arg
e_arg <- UncheckedExp -> TermTypeM Arg
checkArg UncheckedExp
e
  (PatType
t1, PatType
rt, Maybe VName
argext, [VName]
retext) <- SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply SrcLoc
loc (QualName VName -> Maybe (QualName VName)
forall a. a -> Maybe a
Just QualName VName
op', Int
0) PatType
ftype Arg
e_arg
  case (PatType
ftype, PatType
rt) of
    (Scalar (Arrow Aliasing
_ PName
m1 PatType
_ PatType
_), Scalar (Arrow Aliasing
_ PName
m2 PatType
t2 PatType
rettype)) ->
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
        QualName VName
-> Info PatType
-> Exp
-> (Info (PName, StructType, Maybe VName),
    Info (PName, StructType))
-> (Info PatType, Info [VName])
-> SrcLoc
-> Exp
forall (f :: * -> *) vn.
QualName vn
-> f PatType
-> ExpBase f vn
-> (f (PName, StructType, Maybe VName), f (PName, StructType))
-> (f PatType, f [VName])
-> SrcLoc
-> ExpBase f vn
OpSectionLeft
          QualName VName
op'
          (PatType -> Info PatType
forall a. a -> Info a
Info PatType
ftype)
          (Arg -> Exp
argExp Arg
e_arg)
          ((PName, StructType, Maybe VName)
-> Info (PName, StructType, Maybe VName)
forall a. a -> Info a
Info (PName
m1, PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t1, Maybe VName
argext), (PName, StructType) -> Info (PName, StructType)
forall a. a -> Info a
Info (PName
m2, PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t2))
          (PatType -> Info PatType
forall a. a -> Info a
Info PatType
rettype, [VName] -> Info [VName]
forall a. a -> Info a
Info [VName]
retext)
          SrcLoc
loc
    (PatType, PatType)
_ ->
      SrcLoc -> Notes -> Doc -> TermTypeM Exp
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM Exp) -> Doc -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
        Doc
"Operator section with invalid operator of type" Doc -> Doc -> Doc
<+> PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
ftype
checkExp (OpSectionRight QualName Name
op NoInfo PatType
_ UncheckedExp
e (NoInfo (PName, StructType),
 NoInfo (PName, StructType, Maybe VName))
_ NoInfo PatType
NoInfo SrcLoc
loc) = do
  (QualName VName
op', PatType
ftype) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
loc QualName Name
op
  Arg
e_arg <- UncheckedExp -> TermTypeM Arg
checkArg UncheckedExp
e
  case PatType
ftype of
    Scalar (Arrow Aliasing
as1 PName
m1 PatType
t1 (Scalar (Arrow Aliasing
as2 PName
m2 PatType
t2 PatType
ret))) -> do
      (PatType
t2', PatType
ret', Maybe VName
argext, [VName]
_) <-
        SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply
          SrcLoc
loc
          (QualName VName -> Maybe (QualName VName)
forall a. a -> Maybe a
Just QualName VName
op', Int
1)
          (ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> PName
-> PatType
-> PatType
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow Aliasing
as2 PName
m2 PatType
t2 (PatType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> PName
-> PatType
-> PatType
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow Aliasing
as1 PName
m1 PatType
t1 PatType
ret)
          Arg
e_arg
      Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
        QualName VName
-> Info PatType
-> Exp
-> (Info (PName, StructType),
    Info (PName, StructType, Maybe VName))
-> Info PatType
-> SrcLoc
-> Exp
forall (f :: * -> *) vn.
QualName vn
-> f PatType
-> ExpBase f vn
-> (f (PName, StructType), f (PName, StructType, Maybe VName))
-> f PatType
-> SrcLoc
-> ExpBase f vn
OpSectionRight
          QualName VName
op'
          (PatType -> Info PatType
forall a. a -> Info a
Info PatType
ftype)
          (Arg -> Exp
argExp Arg
e_arg)
          ((PName, StructType) -> Info (PName, StructType)
forall a. a -> Info a
Info (PName
m1, PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t1), (PName, StructType, Maybe VName)
-> Info (PName, StructType, Maybe VName)
forall a. a -> Info a
Info (PName
m2, PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
t2', Maybe VName
argext))
          (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
addAliases PatType
ret (Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
ret'))
          SrcLoc
loc
    PatType
_ ->
      SrcLoc -> Notes -> Doc -> TermTypeM Exp
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM Exp) -> Doc -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
        Doc
"Operator section with invalid operator of type" Doc -> Doc -> Doc
<+> PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
ftype
checkExp (ProjectSection [Name]
fields NoInfo PatType
NoInfo SrcLoc
loc) = do
  PatType
a <- SrcLoc -> String -> TermTypeM PatType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"a"
  let usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"projection at"
  PatType
b <- (PatType -> Name -> TermTypeM PatType)
-> PatType -> [Name] -> TermTypeM PatType
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m b
foldM ((Name -> PatType -> TermTypeM PatType)
-> PatType -> Name -> TermTypeM PatType
forall a b c. (a -> b -> c) -> b -> a -> c
flip ((Name -> PatType -> TermTypeM PatType)
 -> PatType -> Name -> TermTypeM PatType)
-> (Name -> PatType -> TermTypeM PatType)
-> PatType
-> Name
-> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ Usage -> Name -> PatType -> TermTypeM PatType
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> PatType -> m PatType
mustHaveField Usage
usage) PatType
a [Name]
fields
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ [Name] -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
[Name] -> f PatType -> SrcLoc -> ExpBase f vn
ProjectSection [Name]
fields (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> PName
-> PatType
-> PatType
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow Aliasing
forall a. Monoid a => a
mempty PName
Unnamed PatType
a PatType
b) SrcLoc
loc
checkExp (IndexSection SliceBase NoInfo Name
slice NoInfo PatType
NoInfo SrcLoc
loc) = do
  Slice
slice' <- SliceBase NoInfo Name -> TermTypeM Slice
checkSlice SliceBase NoInfo Name
slice
  (StructType
t, StructType
_) <- SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType SrcLoc
loc String
"e" (Int -> TermTypeM (StructType, StructType))
-> Int -> TermTypeM (StructType, StructType)
forall a b. (a -> b) -> a -> b
$ Slice -> Int
sliceDims Slice
slice'
  (StructType
t', [VName]
_) <- Maybe (SrcLoc, Rigidity)
-> Slice -> StructType -> TermTypeM (StructType, [VName])
forall as.
Maybe (SrcLoc, Rigidity)
-> Slice
-> TypeBase (DimDecl VName) as
-> TermTypeM (TypeBase (DimDecl VName) as, [VName])
sliceShape Maybe (SrcLoc, Rigidity)
forall a. Maybe a
Nothing Slice
slice' StructType
t
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Slice -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
SliceBase f vn -> f PatType -> SrcLoc -> ExpBase f vn
IndexSection Slice
slice' (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct (StructType -> PatType) -> StructType -> PatType
forall a b. (a -> b) -> a -> b
$ ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ ()
-> PName
-> StructType
-> StructType
-> ScalarTypeBase (DimDecl VName) ()
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow ()
forall a. Monoid a => a
mempty PName
Unnamed StructType
t StructType
t') SrcLoc
loc
checkExp (AppExp (DoLoop [VName]
_ UncheckedPat
mergepat UncheckedExp
mergeexp LoopFormBase NoInfo Name
form UncheckedExp
loopbody SrcLoc
loc) NoInfo AppRes
_) =
  TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
mergeexp) ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
mergeexp' [Occurence]
_ -> do
    Usage -> String -> PatType -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> String -> TypeBase dim as -> m ()
zeroOrderType
      (SrcLoc -> String -> Usage
mkUsage (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
mergeexp) String
"use as loop variable")
      String
"type used as loop variable"
      (PatType -> TermTypeM ()) -> TermTypeM PatType -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
mergeexp'

    -- The handling of dimension sizes is a bit intricate, but very
    -- similar to checking a function, followed by checking a call to
    -- it.  The overall procedure is as follows:
    --
    -- (1) All empty dimensions in the merge pattern are instantiated
    -- with nonrigid size variables.  All explicitly specified
    -- dimensions are preserved.
    --
    -- (2) The body of the loop is type-checked.  The result type is
    -- combined with the merge pattern type to determine which sizes are
    -- variant, and these are turned into size parameters for the merge
    -- pattern.
    --
    -- (3) We now conceptually have a function parameter type and return
    -- type.  We check that it can be called with the initial merge
    -- values as argument.  The result of this is the type of the loop
    -- as a whole.
    --
    -- (There is also a convergence loop for inferring uniqueness, but
    -- that's orthogonal to the size handling.)

    (PatType
merge_t, [VName]
new_dims) <-
      SrcLoc
-> String -> Rigidity -> PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"loop" Rigidity
Nonrigid
        (PatType -> TermTypeM (PatType, [VName]))
-> (PatType -> PatType) -> PatType -> TermTypeM (PatType, [VName])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PatType -> PatType
forall vn as. TypeBase (DimDecl vn) as -> TypeBase (DimDecl vn) as
anySizes -- dim handling (1)
        (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
mergeexp'

    -- dim handling (2)
    let checkLoopReturnSize :: Pat -> Exp -> TermTypeM ([VName], Pat)
checkLoopReturnSize Pat
mergepat' Exp
loopbody' = do
          PatType
loopbody_t <- Exp -> TermTypeM PatType
expTypeFully Exp
loopbody'
          PatType
pat_t <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully (PatType -> TermTypeM PatType) -> PatType -> TermTypeM PatType
forall a b. (a -> b) -> a -> b
$ Pat -> PatType
patternType Pat
mergepat'
          -- We are ignoring the dimensions here, because any mismatches
          -- should be turned into fresh size variables.

          Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (StructType -> StructType -> Checking
CheckingLoopBody (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> PatType
forall vn as. TypeBase (DimDecl vn) as -> TypeBase (DimDecl vn) as
anySizes PatType
pat_t)) (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
loopbody_t)) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
            Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
expect
              (SrcLoc -> String -> Usage
mkUsage (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
loopbody) String
"matching loop body to loop pattern")
              (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct ([VName] -> PatType -> PatType
forall (p :: * -> * -> *) (t :: * -> *) vn c.
(Bifunctor p, Foldable t, Eq vn) =>
t vn -> p (DimDecl vn) c -> p (DimDecl vn) c
anyTheseSizes [VName]
new_dims PatType
pat_t))
              (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
loopbody_t)
          PatType
pat_t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
pat_t
          PatType
loopbody_t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
loopbody_t

          -- For each new_dims, figure out what they are instantiated
          -- with in the initial value.  This is used to determine
          -- whether a size is invariant because it always matches the
          -- initial instantiation of that size.
          let initSubst :: (DimDecl vn, b) -> Maybe (QualName vn, b)
initSubst (NamedDim QualName vn
v, b
d) = (QualName vn, b) -> Maybe (QualName vn, b)
forall a. a -> Maybe a
Just (QualName vn
v, b
d)
              initSubst (DimDecl vn, b)
_ = Maybe (QualName vn, b)
forall a. Maybe a
Nothing
          Map (QualName VName) (DimDecl VName)
init_substs <-
            [(QualName VName, DimDecl VName)]
-> Map (QualName VName) (DimDecl VName)
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([(QualName VName, DimDecl VName)]
 -> Map (QualName VName) (DimDecl VName))
-> (PatType -> [(QualName VName, DimDecl VName)])
-> PatType
-> Map (QualName VName) (DimDecl VName)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((DimDecl VName, DimDecl VName)
 -> Maybe (QualName VName, DimDecl VName))
-> [(DimDecl VName, DimDecl VName)]
-> [(QualName VName, DimDecl VName)]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe (DimDecl VName, DimDecl VName)
-> Maybe (QualName VName, DimDecl VName)
forall vn b. (DimDecl vn, b) -> Maybe (QualName vn, b)
initSubst ([(DimDecl VName, DimDecl VName)]
 -> [(QualName VName, DimDecl VName)])
-> (PatType -> [(DimDecl VName, DimDecl VName)])
-> PatType
-> [(QualName VName, DimDecl VName)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (PatType, [(DimDecl VName, DimDecl VName)])
-> [(DimDecl VName, DimDecl VName)]
forall a b. (a, b) -> b
snd
              ((PatType, [(DimDecl VName, DimDecl VName)])
 -> [(DimDecl VName, DimDecl VName)])
-> (PatType -> (PatType, [(DimDecl VName, DimDecl VName)]))
-> PatType
-> [(DimDecl VName, DimDecl VName)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PatType -> PatType -> (PatType, [(DimDecl VName, DimDecl VName)])
forall as.
Monoid as =>
TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
-> (TypeBase (DimDecl VName) as, [(DimDecl VName, DimDecl VName)])
anyDimOnMismatch PatType
pat_t'
              (PatType -> Map (QualName VName) (DimDecl VName))
-> TermTypeM PatType
-> TermTypeM (Map (QualName VName) (DimDecl VName))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Exp -> TermTypeM PatType
expTypeFully Exp
mergeexp'

          -- Figure out which of the 'new_dims' dimensions are variant.
          -- This works because we know that each dimension from
          -- new_dims in the pattern is unique and distinct.
          --
          -- Our logic here is a bit reversed: the *mismatches* (from
          -- new_dims) are what we want to extract and turn into size
          -- parameters.
          let mismatchSubst :: (DimDecl VName, DimDecl VName) -> m (Maybe (VName, Subst t))
mismatchSubst (NamedDim QualName VName
v, DimDecl VName
d)
                | QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
v VName -> [VName] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [VName]
new_dims =
                  case QualName VName
-> Map (QualName VName) (DimDecl VName) -> Maybe (DimDecl VName)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup QualName VName
v Map (QualName VName) (DimDecl VName)
init_substs of
                    Just DimDecl VName
d'
                      | DimDecl VName
d' DimDecl VName -> DimDecl VName -> Bool
forall a. Eq a => a -> a -> Bool
== DimDecl VName
d ->
                        Maybe (VName, Subst t) -> m (Maybe (VName, Subst t))
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (VName, Subst t) -> m (Maybe (VName, Subst t)))
-> Maybe (VName, Subst t) -> m (Maybe (VName, Subst t))
forall a b. (a -> b) -> a -> b
$ (VName, Subst t) -> Maybe (VName, Subst t)
forall a. a -> Maybe a
Just (QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
v, DimDecl VName -> Subst t
forall t. DimDecl VName -> Subst t
SizeSubst DimDecl VName
d)
                    Maybe (DimDecl VName)
_ -> do
                      ([VName] -> [VName]) -> m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify (QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
v VName -> [VName] -> [VName]
forall a. a -> [a] -> [a]
:)
                      Maybe (VName, Subst t) -> m (Maybe (VName, Subst t))
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe (VName, Subst t)
forall a. Maybe a
Nothing
              mismatchSubst (DimDecl VName, DimDecl VName)
_ = Maybe (VName, Subst t) -> m (Maybe (VName, Subst t))
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe (VName, Subst t)
forall a. Maybe a
Nothing

              (Map VName (Subst t)
init_substs', [VName]
sparams) =
                (State [VName] (Map VName (Subst t))
-> [VName] -> (Map VName (Subst t), [VName])
forall s a. State s a -> s -> (a, s)
`runState` [VName]
forall a. Monoid a => a
mempty) (State [VName] (Map VName (Subst t))
 -> (Map VName (Subst t), [VName]))
-> State [VName] (Map VName (Subst t))
-> (Map VName (Subst t), [VName])
forall a b. (a -> b) -> a -> b
$
                  [(VName, Subst t)] -> Map VName (Subst t)
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([(VName, Subst t)] -> Map VName (Subst t))
-> ([Maybe (VName, Subst t)] -> [(VName, Subst t)])
-> [Maybe (VName, Subst t)]
-> Map VName (Subst t)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Maybe (VName, Subst t)] -> [(VName, Subst t)]
forall a. [Maybe a] -> [a]
catMaybes
                    ([Maybe (VName, Subst t)] -> Map VName (Subst t))
-> StateT [VName] Identity [Maybe (VName, Subst t)]
-> State [VName] (Map VName (Subst t))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ((DimDecl VName, DimDecl VName)
 -> StateT [VName] Identity (Maybe (VName, Subst t)))
-> [(DimDecl VName, DimDecl VName)]
-> StateT [VName] Identity [Maybe (VName, Subst t)]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM
                      (DimDecl VName, DimDecl VName)
-> StateT [VName] Identity (Maybe (VName, Subst t))
forall (m :: * -> *) t.
MonadState [VName] m =>
(DimDecl VName, DimDecl VName) -> m (Maybe (VName, Subst t))
mismatchSubst
                      ((PatType, [(DimDecl VName, DimDecl VName)])
-> [(DimDecl VName, DimDecl VName)]
forall a b. (a, b) -> b
snd ((PatType, [(DimDecl VName, DimDecl VName)])
 -> [(DimDecl VName, DimDecl VName)])
-> (PatType, [(DimDecl VName, DimDecl VName)])
-> [(DimDecl VName, DimDecl VName)]
forall a b. (a -> b) -> a -> b
$ PatType -> PatType -> (PatType, [(DimDecl VName, DimDecl VName)])
forall as.
Monoid as =>
TypeBase (DimDecl VName) as
-> TypeBase (DimDecl VName) as
-> (TypeBase (DimDecl VName) as, [(DimDecl VName, DimDecl VName)])
anyDimOnMismatch PatType
pat_t' PatType
loopbody_t')

          -- Make sure that any of new_dims that are invariant will be
          -- replaced with the invariant size in the loop body.  Failure
          -- to do this can cause type annotations to still refer to
          -- new_dims.
          let dimToInit :: (VName, Subst t) -> TermTypeM ()
dimToInit (VName
v, SizeSubst DimDecl VName
d) =
                VName -> Constraint -> TermTypeM ()
constrain VName
v (Constraint -> TermTypeM ()) -> Constraint -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Maybe (DimDecl VName) -> Usage -> Constraint
Size (DimDecl VName -> Maybe (DimDecl VName)
forall a. a -> Maybe a
Just DimDecl VName
d) (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"size of loop parameter")
              dimToInit (VName, Subst t)
_ =
                () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
          ((VName, Subst Any) -> TermTypeM ())
-> [(VName, Subst Any)] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ (VName, Subst Any) -> TermTypeM ()
forall t. (VName, Subst t) -> TermTypeM ()
dimToInit ([(VName, Subst Any)] -> TermTypeM ())
-> [(VName, Subst Any)] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Map VName (Subst Any) -> [(VName, Subst Any)]
forall k a. Map k a -> [(k, a)]
M.toList Map VName (Subst Any)
forall t. Map VName (Subst t)
init_substs'

          Pat
mergepat'' <- TypeSubs -> Pat -> Pat
forall a. Substitutable a => TypeSubs -> a -> a
applySubst (VName -> Map VName (Subst StructType) -> Maybe (Subst StructType)
forall k a. Ord k => k -> Map k a -> Maybe a
`M.lookup` Map VName (Subst StructType)
forall t. Map VName (Subst t)
init_substs') (Pat -> Pat) -> TermTypeM Pat -> TermTypeM Pat
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Pat -> TermTypeM Pat
forall e. ASTMappable e => e -> TermTypeM e
updateTypes Pat
mergepat'
          ([VName], Pat) -> TermTypeM ([VName], Pat)
forall (m :: * -> *) a. Monad m => a -> m a
return ([VName] -> [VName]
forall a. Ord a => [a] -> [a]
nubOrd [VName]
sparams, Pat
mergepat'')

    -- First we do a basic check of the loop body to figure out which of
    -- the merge parameters are being consumed.  For this, we first need
    -- to check the merge pattern, which requires the (initial) merge
    -- expression.
    --
    -- Play a little with occurences to ensure it does not look like
    -- none of the merge variables are being used.
    (([VName]
sparams, Pat
mergepat', LoopFormBase Info VName
form', Exp
loopbody'), [Occurence]
bodyflow) <-
      case LoopFormBase NoInfo Name
form of
        For IdentBase NoInfo Name
i UncheckedExp
uboundexp -> do
          Exp
uboundexp' <- String -> [PrimType] -> Exp -> TermTypeM Exp
require String
"being the bound in a 'for' loop" [PrimType]
anySignedType (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
uboundexp
          PatType
bound_t <- Exp -> TermTypeM PatType
expTypeFully Exp
uboundexp'
          IdentBase NoInfo Name
-> PatType
-> (Ident
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a.
IdentBase NoInfo Name
-> PatType -> (Ident -> TermTypeM a) -> TermTypeM a
bindingIdent IdentBase NoInfo Name
i PatType
bound_t ((Ident
  -> TermTypeM
       (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Ident
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$ \Ident
i' ->
            TermTypeM
  (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b. TermTypeM b -> TermTypeM b
noUnique (TermTypeM
   (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> ((Pat
     -> TermTypeM
          (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [] UncheckedPat
mergepat (PatType -> InferredType
Ascribed PatType
merge_t) ((Pat
  -> TermTypeM
       (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$
              \Pat
mergepat' -> TermTypeM
  (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b. TermTypeM b -> TermTypeM b
onlySelfAliasing (TermTypeM
   (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$
                TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences (TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$ do
                  Exp
loopbody' <- TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
noSizeEscape (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
loopbody
                  ([VName]
sparams, Pat
mergepat'') <- Pat -> Exp -> TermTypeM ([VName], Pat)
checkLoopReturnSize Pat
mergepat' Exp
loopbody'
                  ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return
                    ( [VName]
sparams,
                      Pat
mergepat'',
                      Ident -> Exp -> LoopFormBase Info VName
forall (f :: * -> *) vn.
IdentBase f vn -> ExpBase f vn -> LoopFormBase f vn
For Ident
i' Exp
uboundexp',
                      Exp
loopbody'
                    )
        ForIn UncheckedPat
xpat UncheckedExp
e -> do
          (StructType
arr_t, StructType
_) <- SrcLoc -> String -> Int -> TermTypeM (StructType, StructType)
newArrayType (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
e) String
"e" Int
1
          Exp
e' <- String -> StructType -> Exp -> TermTypeM Exp
unifies String
"being iterated in a 'for-in' loop" StructType
arr_t (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
          PatType
t <- Exp -> TermTypeM PatType
expTypeFully Exp
e'
          case PatType
t of
            PatType
_
              | Just PatType
t' <- Int -> PatType -> Maybe PatType
forall dim as. Int -> TypeBase dim as -> Maybe (TypeBase dim as)
peelArray Int
1 PatType
t ->
                [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [] UncheckedPat
xpat (PatType -> InferredType
Ascribed PatType
t') ((Pat
  -> TermTypeM
       (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$ \Pat
xpat' ->
                  TermTypeM
  (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b. TermTypeM b -> TermTypeM b
noUnique (TermTypeM
   (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> ((Pat
     -> TermTypeM
          (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [] UncheckedPat
mergepat (PatType -> InferredType
Ascribed PatType
merge_t) ((Pat
  -> TermTypeM
       (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$
                    \Pat
mergepat' -> TermTypeM
  (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b. TermTypeM b -> TermTypeM b
onlySelfAliasing (TermTypeM
   (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences (TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$ do
                      Exp
loopbody' <- TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
noSizeEscape (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
loopbody
                      ([VName]
sparams, Pat
mergepat'') <- Pat -> Exp -> TermTypeM ([VName], Pat)
checkLoopReturnSize Pat
mergepat' Exp
loopbody'
                      ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return
                        ( [VName]
sparams,
                          Pat
mergepat'',
                          Pat -> Exp -> LoopFormBase Info VName
forall (f :: * -> *) vn.
PatBase f vn -> ExpBase f vn -> LoopFormBase f vn
ForIn Pat
xpat' Exp
e',
                          Exp
loopbody'
                        )
              | Bool
otherwise ->
                SrcLoc
-> Notes
-> Doc
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
e) Notes
forall a. Monoid a => a
mempty (Doc
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> Doc
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$
                  Doc
"Iteratee of a for-in loop must be an array, but expression has type"
                    Doc -> Doc -> Doc
<+> PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
t
        While UncheckedExp
cond ->
          TermTypeM
  (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b. TermTypeM b -> TermTypeM b
noUnique (TermTypeM
   (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> ((Pat
     -> TermTypeM
          (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [] UncheckedPat
mergepat (PatType -> InferredType
Ascribed PatType
merge_t) ((Pat
  -> TermTypeM
       (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (Pat
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$ \Pat
mergepat' ->
            TermTypeM
  (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b. TermTypeM b -> TermTypeM b
onlySelfAliasing (TermTypeM
   (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> (TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
    -> TermTypeM
         (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences (TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
 -> TermTypeM
      (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence]))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM
     (([VName], Pat, LoopFormBase Info VName, Exp), [Occurence])
forall a b. (a -> b) -> a -> b
$
              TermTypeM Exp
-> (Exp
    -> [Occurence]
    -> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially
                ( UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
cond
                    TermTypeM Exp -> (Exp -> TermTypeM Exp) -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= String -> StructType -> Exp -> TermTypeM Exp
unifies String
"being the condition of a 'while' loop" (ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim PrimType
Bool)
                )
                ((Exp
  -> [Occurence]
  -> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp))
 -> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp))
-> (Exp
    -> [Occurence]
    -> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp))
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
forall a b. (a -> b) -> a -> b
$ \Exp
cond' [Occurence]
_ -> do
                  Exp
loopbody' <- TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
noSizeEscape (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
loopbody
                  ([VName]
sparams, Pat
mergepat'') <- Pat -> Exp -> TermTypeM ([VName], Pat)
checkLoopReturnSize Pat
mergepat' Exp
loopbody'
                  ([VName], Pat, LoopFormBase Info VName, Exp)
-> TermTypeM ([VName], Pat, LoopFormBase Info VName, Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return
                    ( [VName]
sparams,
                      Pat
mergepat'',
                      Exp -> LoopFormBase Info VName
forall (f :: * -> *) vn. ExpBase f vn -> LoopFormBase f vn
While Exp
cond',
                      Exp
loopbody'
                    )

    Pat
mergepat'' <- do
      PatType
loopbody_t <- Exp -> TermTypeM PatType
expTypeFully Exp
loopbody'
      Pat -> Names -> PatType -> Usage -> TermTypeM Pat
forall (m :: * -> *) t.
(MonadUnify m, MonadTypeChecker m, Located t,
 MonadReader TermEnv m) =>
Pat -> Names -> PatType -> t -> m Pat
convergePat Pat
mergepat' ([Occurence] -> Names
allConsumed [Occurence]
bodyflow) PatType
loopbody_t (Usage -> TermTypeM Pat) -> Usage -> TermTypeM Pat
forall a b. (a -> b) -> a -> b
$
        SrcLoc -> String -> Usage
mkUsage (Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
loopbody') String
"being (part of) the result of the loop body"

    let consumeMerge :: PatBase Info vn -> TypeBase dim Aliasing -> TermTypeM ()
consumeMerge (Id vn
_ (Info PatType
pt) SrcLoc
ploc) TypeBase dim Aliasing
mt
          | PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
pt = SrcLoc -> Aliasing -> TermTypeM ()
consume SrcLoc
ploc (Aliasing -> TermTypeM ()) -> Aliasing -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ TypeBase dim Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase dim Aliasing
mt
        consumeMerge (TuplePat [PatBase Info vn]
pats SrcLoc
_) TypeBase dim Aliasing
t
          | Just [TypeBase dim Aliasing]
ts <- TypeBase dim Aliasing -> Maybe [TypeBase dim Aliasing]
forall dim as. TypeBase dim as -> Maybe [TypeBase dim as]
isTupleRecord TypeBase dim Aliasing
t =
            (PatBase Info vn -> TypeBase dim Aliasing -> TermTypeM ())
-> [PatBase Info vn] -> [TypeBase dim Aliasing] -> TermTypeM ()
forall (m :: * -> *) a b c.
Applicative m =>
(a -> b -> m c) -> [a] -> [b] -> m ()
zipWithM_ PatBase Info vn -> TypeBase dim Aliasing -> TermTypeM ()
consumeMerge [PatBase Info vn]
pats [TypeBase dim Aliasing]
ts
        consumeMerge (PatParens PatBase Info vn
pat SrcLoc
_) TypeBase dim Aliasing
t =
          PatBase Info vn -> TypeBase dim Aliasing -> TermTypeM ()
consumeMerge PatBase Info vn
pat TypeBase dim Aliasing
t
        consumeMerge (PatAscription PatBase Info vn
pat TypeDeclBase Info vn
_ SrcLoc
_) TypeBase dim Aliasing
t =
          PatBase Info vn -> TypeBase dim Aliasing -> TermTypeM ()
consumeMerge PatBase Info vn
pat TypeBase dim Aliasing
t
        consumeMerge PatBase Info vn
_ TypeBase dim Aliasing
_ =
          () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
    Pat -> PatType -> TermTypeM ()
forall vn dim.
PatBase Info vn -> TypeBase dim Aliasing -> TermTypeM ()
consumeMerge Pat
mergepat'' (PatType -> TermTypeM ()) -> TermTypeM PatType -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
mergeexp'

    -- dim handling (3)
    let sparams_anydim :: Map VName (Subst t)
sparams_anydim = [(VName, Subst t)] -> Map VName (Subst t)
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([(VName, Subst t)] -> Map VName (Subst t))
-> [(VName, Subst t)] -> Map VName (Subst t)
forall a b. (a -> b) -> a -> b
$ [VName] -> [Subst t] -> [(VName, Subst t)]
forall a b. [a] -> [b] -> [(a, b)]
zip [VName]
sparams ([Subst t] -> [(VName, Subst t)])
-> [Subst t] -> [(VName, Subst t)]
forall a b. (a -> b) -> a -> b
$ Subst t -> [Subst t]
forall a. a -> [a]
repeat (Subst t -> [Subst t]) -> Subst t -> [Subst t]
forall a b. (a -> b) -> a -> b
$ DimDecl VName -> Subst t
forall t. DimDecl VName -> Subst t
SizeSubst (DimDecl VName -> Subst t) -> DimDecl VName -> Subst t
forall a b. (a -> b) -> a -> b
$ Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim Maybe VName
forall a. Maybe a
Nothing
        loopt_anydims :: PatType
loopt_anydims =
          TypeSubs -> PatType -> PatType
forall a. Substitutable a => TypeSubs -> a -> a
applySubst (VName -> Map VName (Subst StructType) -> Maybe (Subst StructType)
forall k a. Ord k => k -> Map k a -> Maybe a
`M.lookup` Map VName (Subst StructType)
forall t. Map VName (Subst t)
sparams_anydim) (PatType -> PatType) -> PatType -> PatType
forall a b. (a -> b) -> a -> b
$
            Pat -> PatType
patternType Pat
mergepat''
    (StructType
merge_t', [VName]
_) <-
      SrcLoc
-> String
-> Rigidity
-> StructType
-> TermTypeM (StructType, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"loopres" Rigidity
Nonrigid (StructType -> TermTypeM (StructType, [VName]))
-> StructType -> TermTypeM (StructType, [VName])
forall a b. (a -> b) -> a -> b
$ PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
loopt_anydims
    StructType
mergeexp_t <- PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType)
-> TermTypeM PatType -> TermTypeM StructType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Exp -> TermTypeM PatType
expTypeFully Exp
mergeexp'
    Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (StructType -> StructType -> Checking
CheckingLoopInitial (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
loopt_anydims) StructType
mergeexp_t) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
      Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify
        (SrcLoc -> String -> Usage
mkUsage (Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
mergeexp') String
"matching initial loop values to pattern")
        StructType
merge_t'
        StructType
mergeexp_t

    (PatType
loopt, [VName]
retext) <- SrcLoc -> RigidSource -> PatType -> TermTypeM (PatType, [VName])
forall als.
SrcLoc
-> RigidSource
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInType SrcLoc
loc RigidSource
RigidLoop PatType
loopt_anydims
    -- We set all of the uniqueness to be unique.  This is intentional,
    -- and matches what happens for function calls.  Those arrays that
    -- really *cannot* be consumed will alias something unconsumable,
    -- and will be caught that way.
    let bound_here :: Names
bound_here = Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames Pat
mergepat'' Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList [VName]
sparams Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> Names
form_bound
        form_bound :: Names
form_bound =
          case LoopFormBase Info VName
form' of
            For Ident
v Exp
_ -> VName -> Names
forall a. a -> Set a
S.singleton (VName -> Names) -> VName -> Names
forall a b. (a -> b) -> a -> b
$ Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
v
            ForIn Pat
forpat Exp
_ -> Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames Pat
forpat
            While {} -> Names
forall a. Monoid a => a
mempty
        loopt' :: PatType
loopt' =
          (Aliasing -> Aliasing) -> PatType -> PatType
forall (p :: * -> * -> *) b c a.
Bifunctor p =>
(b -> c) -> p a b -> p a c
second (Aliasing -> Aliasing -> Aliasing
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` (VName -> Alias) -> Names -> Aliasing
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map VName -> Alias
AliasBound Names
bound_here) (PatType -> PatType) -> PatType -> PatType
forall a b. (a -> b) -> a -> b
$
            PatType
loopt PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Unique

    -- Eliminate those new_dims that turned into sparams so it won't
    -- look like we have ambiguous sizes lying around.
    (Constraints -> Constraints) -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
(Constraints -> Constraints) -> m ()
modifyConstraints ((Constraints -> Constraints) -> TermTypeM ())
-> (Constraints -> Constraints) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ (VName -> (Int, Constraint) -> Bool) -> Constraints -> Constraints
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey ((VName -> (Int, Constraint) -> Bool)
 -> Constraints -> Constraints)
-> (VName -> (Int, Constraint) -> Bool)
-> Constraints
-> Constraints
forall a b. (a -> b) -> a -> b
$ \VName
k (Int, Constraint)
_ -> VName
k VName -> [VName] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`notElem` [VName]
sparams

    Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$
      AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp
        ([VName]
-> Pat
-> Exp
-> LoopFormBase Info VName
-> Exp
-> SrcLoc
-> AppExpBase Info VName
forall (f :: * -> *) vn.
[VName]
-> PatBase f vn
-> ExpBase f vn
-> LoopFormBase f vn
-> ExpBase f vn
-> SrcLoc
-> AppExpBase f vn
DoLoop [VName]
sparams Pat
mergepat'' Exp
mergeexp' LoopFormBase Info VName
form' Exp
loopbody' SrcLoc
loc)
        (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
loopt' [VName]
retext)
  where
    anyTheseSizes :: t vn -> p (DimDecl vn) c -> p (DimDecl vn) c
anyTheseSizes t vn
to_hide = (DimDecl vn -> DimDecl vn) -> p (DimDecl vn) c -> p (DimDecl vn) c
forall (p :: * -> * -> *) a b c.
Bifunctor p =>
(a -> b) -> p a c -> p b c
first DimDecl vn -> DimDecl vn
onDim
      where
        onDim :: DimDecl vn -> DimDecl vn
onDim (NamedDim (QualName [vn]
_ vn
v))
          | vn
v vn -> t vn -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` t vn
to_hide = Maybe vn -> DimDecl vn
forall vn. Maybe vn -> DimDecl vn
AnyDim Maybe vn
forall a. Maybe a
Nothing
        onDim DimDecl vn
d = DimDecl vn
d

    convergePat :: Pat -> Names -> PatType -> t -> m Pat
convergePat Pat
pat Names
body_cons PatType
body_t t
body_loc = do
      let consumed_merge :: Names
consumed_merge = Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames Pat
pat Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` Names
body_cons

          uniquePat :: Pat -> Pat
uniquePat (Wildcard (Info PatType
t) SrcLoc
wloc) =
            Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. f PatType -> SrcLoc -> PatBase f vn
Wildcard (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ PatType
t PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique) SrcLoc
wloc
          uniquePat (PatParens Pat
p SrcLoc
ploc) =
            Pat -> SrcLoc -> Pat
forall (f :: * -> *) vn. PatBase f vn -> SrcLoc -> PatBase f vn
PatParens (Pat -> Pat
uniquePat Pat
p) SrcLoc
ploc
          uniquePat (Id VName
name (Info PatType
t) SrcLoc
iloc)
            | VName
name VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
consumed_merge =
              let t' :: PatType
t' = PatType
t PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Unique PatType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` Aliasing
forall a. Monoid a => a
mempty
               in VName -> Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. vn -> f PatType -> SrcLoc -> PatBase f vn
Id VName
name (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t') SrcLoc
iloc
            | Bool
otherwise =
              let t' :: PatType
t' = PatType
t PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique
               in VName -> Info PatType -> SrcLoc -> Pat
forall (f :: * -> *) vn. vn -> f PatType -> SrcLoc -> PatBase f vn
Id VName
name (PatType -> Info PatType
forall a. a -> Info a
Info PatType
t') SrcLoc
iloc
          uniquePat (TuplePat [Pat]
pats SrcLoc
ploc) =
            [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn. [PatBase f vn] -> SrcLoc -> PatBase f vn
TuplePat ((Pat -> Pat) -> [Pat] -> [Pat]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> Pat
uniquePat [Pat]
pats) SrcLoc
ploc
          uniquePat (RecordPat [(Name, Pat)]
fs SrcLoc
ploc) =
            [(Name, Pat)] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
[(Name, PatBase f vn)] -> SrcLoc -> PatBase f vn
RecordPat (((Name, Pat) -> (Name, Pat)) -> [(Name, Pat)] -> [(Name, Pat)]
forall a b. (a -> b) -> [a] -> [b]
map ((Pat -> Pat) -> (Name, Pat) -> (Name, Pat)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Pat -> Pat
uniquePat) [(Name, Pat)]
fs) SrcLoc
ploc
          uniquePat (PatAscription Pat
p TypeDeclBase Info VName
t SrcLoc
ploc) =
            Pat -> TypeDeclBase Info VName -> SrcLoc -> Pat
forall (f :: * -> *) vn.
PatBase f vn -> TypeDeclBase f vn -> SrcLoc -> PatBase f vn
PatAscription Pat
p TypeDeclBase Info VName
t SrcLoc
ploc
          uniquePat p :: Pat
p@PatLit {} = Pat
p
          uniquePat (PatConstr Name
n Info PatType
t [Pat]
ps SrcLoc
ploc) =
            Name -> Info PatType -> [Pat] -> SrcLoc -> Pat
forall (f :: * -> *) vn.
Name -> f PatType -> [PatBase f vn] -> SrcLoc -> PatBase f vn
PatConstr Name
n Info PatType
t ((Pat -> Pat) -> [Pat] -> [Pat]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> Pat
uniquePat [Pat]
ps) SrcLoc
ploc

          -- Make the pattern unique where needed.
          pat' :: Pat
pat' = Pat -> Pat
uniquePat Pat
pat

      PatType
pat_t <- PatType -> m PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully (PatType -> m PatType) -> PatType -> m PatType
forall a b. (a -> b) -> a -> b
$ Pat -> PatType
patternType Pat
pat'
      Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (PatType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural PatType
body_t TypeBase () () -> TypeBase () () -> Bool
`subtypeOf` PatType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural PatType
pat_t) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$
        SrcLoc -> StructType -> [StructType] -> m ()
forall (m :: * -> *) a.
MonadTypeChecker m =>
SrcLoc -> StructType -> [StructType] -> m a
unexpectedType (t -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf t
body_loc) (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
body_t) [PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
pat_t]

      -- Check that the new values of consumed merge parameters do not
      -- alias something bound outside the loop, AND that anything
      -- returned for a unique merge parameter does not alias anything
      -- else returned.  We also update the aliases for the pattern.
      Names
bound_outside <- (TermEnv -> Names) -> m Names
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((TermEnv -> Names) -> m Names) -> (TermEnv -> Names) -> m Names
forall a b. (a -> b) -> a -> b
$ [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList ([VName] -> Names) -> (TermEnv -> [VName]) -> TermEnv -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map VName ValBinding -> [VName]
forall k a. Map k a -> [k]
M.keys (Map VName ValBinding -> [VName])
-> (TermEnv -> Map VName ValBinding) -> TermEnv -> [VName]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermScope -> Map VName ValBinding
scopeVtable (TermScope -> Map VName ValBinding)
-> (TermEnv -> TermScope) -> TermEnv -> Map VName ValBinding
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermEnv -> TermScope
termScope
      let combAliases :: TypeBase dim ast -> TypeBase shape ast -> TypeBase dim ast
combAliases TypeBase dim ast
t1 TypeBase shape ast
t2 =
            case TypeBase dim ast
t1 of
              Scalar Record {} -> TypeBase dim ast
t1
              TypeBase dim ast
_ -> TypeBase dim ast
t1 TypeBase dim ast -> (ast -> ast) -> TypeBase dim ast
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
`addAliases` (ast -> ast -> ast
forall a. Semigroup a => a -> a -> a
<> TypeBase shape ast -> ast
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase shape ast
t2)

          checkMergeReturn :: PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn)
checkMergeReturn (Id vn
pat_v (Info PatType
pat_v_t) SrcLoc
patloc) TypeBase dim Aliasing
t
            | PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
pat_v_t,
              VName
v : [VName]
_ <-
                Names -> [VName]
forall a. Set a -> [a]
S.toList (Names -> [VName]) -> Names -> [VName]
forall a b. (a -> b) -> a -> b
$
                  (Alias -> VName) -> Aliasing -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> VName
aliasVar (TypeBase dim Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase dim Aliasing
t) Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` Names
bound_outside =
              m (PatBase Info vn) -> t m (PatBase Info vn)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m (PatBase Info vn) -> t m (PatBase Info vn))
-> m (PatBase Info vn) -> t m (PatBase Info vn)
forall a b. (a -> b) -> a -> b
$
                SrcLoc -> Notes -> Doc -> m (PatBase Info vn)
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m (PatBase Info vn)) -> Doc -> m (PatBase Info vn)
forall a b. (a -> b) -> a -> b
$
                  Doc
"Return value for loop parameter"
                    Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (vn -> Doc
forall v. IsName v => v -> Doc
pprName vn
pat_v)
                    Doc -> Doc -> Doc
<+> Doc
"aliases"
                    Doc -> Doc -> Doc
<+> VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
v Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
            | Bool
otherwise = do
              (Aliasing
cons, Aliasing
obs) <- t m (Aliasing, Aliasing)
forall s (m :: * -> *). MonadState s m => m s
get
              Bool -> t m () -> t m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (Aliasing -> Bool
forall a. Set a -> Bool
S.null (Aliasing -> Bool) -> Aliasing -> Bool
forall a b. (a -> b) -> a -> b
$ TypeBase dim Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase dim Aliasing
t Aliasing -> Aliasing -> Aliasing
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` Aliasing
cons) (t m () -> t m ()) -> t m () -> t m ()
forall a b. (a -> b) -> a -> b
$
                m () -> t m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m () -> t m ()) -> m () -> t m ()
forall a b. (a -> b) -> a -> b
$
                  SrcLoc -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
                    Doc
"Return value for loop parameter"
                      Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (vn -> Doc
forall v. IsName v => v -> Doc
pprName vn
pat_v)
                      Doc -> Doc -> Doc
<+> Doc
"aliases other consumed loop parameter."
              Bool -> t m () -> t m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when
                ( PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
pat_v_t
                    Bool -> Bool -> Bool
&& Bool -> Bool
not (Aliasing -> Bool
forall a. Set a -> Bool
S.null (TypeBase dim Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase dim Aliasing
t Aliasing -> Aliasing -> Aliasing
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` (Aliasing
cons Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> Aliasing
obs)))
                )
                (t m () -> t m ()) -> t m () -> t m ()
forall a b. (a -> b) -> a -> b
$ m () -> t m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m () -> t m ()) -> m () -> t m ()
forall a b. (a -> b) -> a -> b
$
                  SrcLoc -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
                    Doc
"Return value for consuming loop parameter"
                      Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (vn -> Doc
forall v. IsName v => v -> Doc
pprName vn
pat_v)
                      Doc -> Doc -> Doc
<+> Doc
"aliases previously returned value."
              if PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
pat_v_t
                then (Aliasing, Aliasing) -> t m ()
forall s (m :: * -> *). MonadState s m => s -> m ()
put (Aliasing
cons Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> TypeBase dim Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase dim Aliasing
t, Aliasing
obs)
                else (Aliasing, Aliasing) -> t m ()
forall s (m :: * -> *). MonadState s m => s -> m ()
put (Aliasing
cons, Aliasing
obs Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> TypeBase dim Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase dim Aliasing
t)

              PatBase Info vn -> t m (PatBase Info vn)
forall (m :: * -> *) a. Monad m => a -> m a
return (PatBase Info vn -> t m (PatBase Info vn))
-> PatBase Info vn -> t m (PatBase Info vn)
forall a b. (a -> b) -> a -> b
$ vn -> Info PatType -> SrcLoc -> PatBase Info vn
forall (f :: * -> *) vn. vn -> f PatType -> SrcLoc -> PatBase f vn
Id vn
pat_v (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> TypeBase dim Aliasing -> PatType
forall ast dim shape.
Monoid ast =>
TypeBase dim ast -> TypeBase shape ast -> TypeBase dim ast
combAliases PatType
pat_v_t TypeBase dim Aliasing
t)) SrcLoc
patloc
          checkMergeReturn (Wildcard (Info PatType
pat_v_t) SrcLoc
patloc) TypeBase dim Aliasing
t =
            PatBase Info vn -> t m (PatBase Info vn)
forall (m :: * -> *) a. Monad m => a -> m a
return (PatBase Info vn -> t m (PatBase Info vn))
-> PatBase Info vn -> t m (PatBase Info vn)
forall a b. (a -> b) -> a -> b
$ Info PatType -> SrcLoc -> PatBase Info vn
forall (f :: * -> *) vn. f PatType -> SrcLoc -> PatBase f vn
Wildcard (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> TypeBase dim Aliasing -> PatType
forall ast dim shape.
Monoid ast =>
TypeBase dim ast -> TypeBase shape ast -> TypeBase dim ast
combAliases PatType
pat_v_t TypeBase dim Aliasing
t)) SrcLoc
patloc
          checkMergeReturn (PatParens PatBase Info vn
p SrcLoc
_) TypeBase dim Aliasing
t =
            PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn)
checkMergeReturn PatBase Info vn
p TypeBase dim Aliasing
t
          checkMergeReturn (PatAscription PatBase Info vn
p TypeDeclBase Info vn
_ SrcLoc
_) TypeBase dim Aliasing
t =
            PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn)
checkMergeReturn PatBase Info vn
p TypeBase dim Aliasing
t
          checkMergeReturn (RecordPat [(Name, PatBase Info vn)]
pfs SrcLoc
patloc) (Scalar (Record Map Name (TypeBase dim Aliasing)
tfs)) =
            [(Name, PatBase Info vn)] -> SrcLoc -> PatBase Info vn
forall (f :: * -> *) vn.
[(Name, PatBase f vn)] -> SrcLoc -> PatBase f vn
RecordPat ([(Name, PatBase Info vn)] -> SrcLoc -> PatBase Info vn)
-> (Map Name (PatBase Info vn) -> [(Name, PatBase Info vn)])
-> Map Name (PatBase Info vn)
-> SrcLoc
-> PatBase Info vn
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Name (PatBase Info vn) -> [(Name, PatBase Info vn)]
forall k a. Map k a -> [(k, a)]
M.toList (Map Name (PatBase Info vn) -> SrcLoc -> PatBase Info vn)
-> t m (Map Name (PatBase Info vn))
-> t m (SrcLoc -> PatBase Info vn)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Map Name (t m (PatBase Info vn))
-> t m (Map Name (PatBase Info vn))
forall (t :: * -> *) (m :: * -> *) a.
(Traversable t, Monad m) =>
t (m a) -> m (t a)
sequence Map Name (t m (PatBase Info vn))
pfs' t m (SrcLoc -> PatBase Info vn)
-> t m SrcLoc -> t m (PatBase Info vn)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> t m SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
patloc
            where
              pfs' :: Map Name (t m (PatBase Info vn))
pfs' =
                (PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn))
-> Map Name (PatBase Info vn)
-> Map Name (TypeBase dim Aliasing)
-> Map Name (t m (PatBase Info vn))
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
M.intersectionWith
                  PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn)
checkMergeReturn
                  ([(Name, PatBase Info vn)] -> Map Name (PatBase Info vn)
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList [(Name, PatBase Info vn)]
pfs)
                  Map Name (TypeBase dim Aliasing)
tfs
          checkMergeReturn (TuplePat [PatBase Info vn]
pats SrcLoc
patloc) TypeBase dim Aliasing
t
            | Just [TypeBase dim Aliasing]
ts <- TypeBase dim Aliasing -> Maybe [TypeBase dim Aliasing]
forall dim as. TypeBase dim as -> Maybe [TypeBase dim as]
isTupleRecord TypeBase dim Aliasing
t =
              [PatBase Info vn] -> SrcLoc -> PatBase Info vn
forall (f :: * -> *) vn. [PatBase f vn] -> SrcLoc -> PatBase f vn
TuplePat
                ([PatBase Info vn] -> SrcLoc -> PatBase Info vn)
-> t m [PatBase Info vn] -> t m (SrcLoc -> PatBase Info vn)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn))
-> [PatBase Info vn]
-> [TypeBase dim Aliasing]
-> t m [PatBase Info vn]
forall (m :: * -> *) a b c.
Applicative m =>
(a -> b -> m c) -> [a] -> [b] -> m [c]
zipWithM PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn)
checkMergeReturn [PatBase Info vn]
pats [TypeBase dim Aliasing]
ts
                t m (SrcLoc -> PatBase Info vn)
-> t m SrcLoc -> t m (PatBase Info vn)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> t m SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
patloc
          checkMergeReturn PatBase Info vn
p TypeBase dim Aliasing
_ =
            PatBase Info vn -> t m (PatBase Info vn)
forall (m :: * -> *) a. Monad m => a -> m a
return PatBase Info vn
p

      (Pat
pat'', (Aliasing
pat_cons, Aliasing
_)) <-
        StateT (Aliasing, Aliasing) m Pat
-> (Aliasing, Aliasing) -> m (Pat, (Aliasing, Aliasing))
forall s (m :: * -> *) a. StateT s m a -> s -> m (a, s)
runStateT (Pat -> PatType -> StateT (Aliasing, Aliasing) m Pat
forall (t :: (* -> *) -> * -> *) (m :: * -> *) vn dim.
(MonadTrans t, MonadTypeChecker m, IsName vn,
 MonadState (Aliasing, Aliasing) (t m)) =>
PatBase Info vn -> TypeBase dim Aliasing -> t m (PatBase Info vn)
checkMergeReturn Pat
pat' PatType
body_t) (Aliasing
forall a. Monoid a => a
mempty, Aliasing
forall a. Monoid a => a
mempty)

      let body_cons' :: Names
body_cons' = Names
body_cons Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> (Alias -> VName) -> Aliasing -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> VName
aliasVar Aliasing
pat_cons
      if Names
body_cons' Names -> Names -> Bool
forall a. Eq a => a -> a -> Bool
== Names
body_cons Bool -> Bool -> Bool
&& Pat -> PatType
patternType Pat
pat'' PatType -> PatType -> Bool
forall a. Eq a => a -> a -> Bool
== Pat -> PatType
patternType Pat
pat
        then Pat -> m Pat
forall (m :: * -> *) a. Monad m => a -> m a
return Pat
pat'
        else Pat -> Names -> PatType -> t -> m Pat
convergePat Pat
pat'' Names
body_cons' PatType
body_t t
body_loc
checkExp (Constr Name
name [UncheckedExp]
es NoInfo PatType
NoInfo SrcLoc
loc) = do
  StructType
t <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"t"
  [Exp]
es' <- (UncheckedExp -> TermTypeM Exp)
-> [UncheckedExp] -> TermTypeM [Exp]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM UncheckedExp -> TermTypeM Exp
checkExp [UncheckedExp]
es
  [PatType]
ets <- (Exp -> TermTypeM PatType) -> [Exp] -> TermTypeM [PatType]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM Exp -> TermTypeM PatType
expTypeFully [Exp]
es'
  Usage -> Name -> StructType -> [StructType] -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> Name -> StructType -> [StructType] -> m ()
mustHaveConstr (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"use of constructor") Name
name StructType
t (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType) -> [PatType] -> [StructType]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [PatType]
ets)
  -- A sum value aliases *anything* that went into its construction.
  let als :: Aliasing
als = (PatType -> Aliasing) -> [PatType] -> Aliasing
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases [PatType]
ets
  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ Name -> [Exp] -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
Name -> [ExpBase f vn] -> f PatType -> SrcLoc -> ExpBase f vn
Constr Name
name [Exp]
es' (PatType -> Info PatType
forall a. a -> Info a
Info (PatType -> Info PatType) -> PatType -> Info PatType
forall a b. (a -> b) -> a -> b
$ StructType -> PatType
forall dim as. TypeBase dim as -> TypeBase dim Aliasing
fromStruct StructType
t PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
`addAliases` (Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> Aliasing
als)) SrcLoc
loc
checkExp (AppExp (Match UncheckedExp
e NonEmpty (CaseBase NoInfo Name)
cs SrcLoc
loc) NoInfo AppRes
_) =
  TermTypeM Exp
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b.
TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially (UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e) ((Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp)
-> (Exp -> [Occurence] -> TermTypeM Exp) -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ \Exp
e' [Occurence]
_ -> do
    PatType
mt <- Exp -> TermTypeM PatType
expTypeFully Exp
e'
    (NonEmpty (CaseBase Info VName)
cs', PatType
t, [VName]
retext) <- PatType
-> NonEmpty (CaseBase NoInfo Name)
-> TermTypeM (NonEmpty (CaseBase Info VName), PatType, [VName])
checkCases PatType
mt NonEmpty (CaseBase NoInfo Name)
cs
    Usage -> String -> PatType -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> String -> TypeBase dim as -> m ()
zeroOrderType
      (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"being returned 'match'")
      String
"type returned from pattern match"
      PatType
t
    Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp -> TermTypeM Exp) -> Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ AppExpBase Info VName -> Info AppRes -> Exp
forall (f :: * -> *) vn.
AppExpBase f vn -> f AppRes -> ExpBase f vn
AppExp (Exp
-> NonEmpty (CaseBase Info VName)
-> SrcLoc
-> AppExpBase Info VName
forall (f :: * -> *) vn.
ExpBase f vn
-> NonEmpty (CaseBase f vn) -> SrcLoc -> AppExpBase f vn
Match Exp
e' NonEmpty (CaseBase Info VName)
cs' SrcLoc
loc) (AppRes -> Info AppRes
forall a. a -> Info a
Info (AppRes -> Info AppRes) -> AppRes -> Info AppRes
forall a b. (a -> b) -> a -> b
$ PatType -> [VName] -> AppRes
AppRes PatType
t [VName]
retext)
checkExp (Attr AttrInfo
info UncheckedExp
e SrcLoc
loc) =
  AttrInfo -> Exp -> SrcLoc -> Exp
forall (f :: * -> *) vn.
AttrInfo -> ExpBase f vn -> SrcLoc -> ExpBase f vn
Attr AttrInfo
info (Exp -> SrcLoc -> Exp)
-> TermTypeM Exp -> TermTypeM (SrcLoc -> Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e TermTypeM (SrcLoc -> Exp) -> TermTypeM SrcLoc -> TermTypeM Exp
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> SrcLoc -> TermTypeM SrcLoc
forall (f :: * -> *) a. Applicative f => a -> f a
pure SrcLoc
loc

checkCases ::
  PatType ->
  NE.NonEmpty (CaseBase NoInfo Name) ->
  TermTypeM (NE.NonEmpty (CaseBase Info VName), PatType, [VName])
checkCases :: PatType
-> NonEmpty (CaseBase NoInfo Name)
-> TermTypeM (NonEmpty (CaseBase Info VName), PatType, [VName])
checkCases PatType
mt NonEmpty (CaseBase NoInfo Name)
rest_cs =
  case NonEmpty (CaseBase NoInfo Name)
-> (CaseBase NoInfo Name, Maybe (NonEmpty (CaseBase NoInfo Name)))
forall a. NonEmpty a -> (a, Maybe (NonEmpty a))
NE.uncons NonEmpty (CaseBase NoInfo Name)
rest_cs of
    (CaseBase NoInfo Name
c, Maybe (NonEmpty (CaseBase NoInfo Name))
Nothing) -> do
      (CaseBase Info VName
c', PatType
t, [VName]
retext) <- PatType
-> CaseBase NoInfo Name
-> TermTypeM (CaseBase Info VName, PatType, [VName])
checkCase PatType
mt CaseBase NoInfo Name
c
      (NonEmpty (CaseBase Info VName), PatType, [VName])
-> TermTypeM (NonEmpty (CaseBase Info VName), PatType, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return (CaseBase Info VName
c' CaseBase Info VName
-> [CaseBase Info VName] -> NonEmpty (CaseBase Info VName)
forall a. a -> [a] -> NonEmpty a
NE.:| [], PatType
t, [VName]
retext)
    (CaseBase NoInfo Name
c, Just NonEmpty (CaseBase NoInfo Name)
cs) -> do
      (((CaseBase Info VName
c', PatType
c_t, [VName]
_), (NonEmpty (CaseBase Info VName)
cs', PatType
cs_t, [VName]
_)), [Occurence]
dflow) <-
        TermTypeM
  ((CaseBase Info VName, PatType, [VName]),
   (NonEmpty (CaseBase Info VName), PatType, [VName]))
-> TermTypeM
     (((CaseBase Info VName, PatType, [VName]),
       (NonEmpty (CaseBase Info VName), PatType, [VName])),
      [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences (TermTypeM
   ((CaseBase Info VName, PatType, [VName]),
    (NonEmpty (CaseBase Info VName), PatType, [VName]))
 -> TermTypeM
      (((CaseBase Info VName, PatType, [VName]),
        (NonEmpty (CaseBase Info VName), PatType, [VName])),
       [Occurence]))
-> TermTypeM
     ((CaseBase Info VName, PatType, [VName]),
      (NonEmpty (CaseBase Info VName), PatType, [VName]))
-> TermTypeM
     (((CaseBase Info VName, PatType, [VName]),
       (NonEmpty (CaseBase Info VName), PatType, [VName])),
      [Occurence])
forall a b. (a -> b) -> a -> b
$ PatType
-> CaseBase NoInfo Name
-> TermTypeM (CaseBase Info VName, PatType, [VName])
checkCase PatType
mt CaseBase NoInfo Name
c TermTypeM (CaseBase Info VName, PatType, [VName])
-> TermTypeM (NonEmpty (CaseBase Info VName), PatType, [VName])
-> TermTypeM
     ((CaseBase Info VName, PatType, [VName]),
      (NonEmpty (CaseBase Info VName), PatType, [VName]))
forall a b. TermTypeM a -> TermTypeM b -> TermTypeM (a, b)
`alternative` PatType
-> NonEmpty (CaseBase NoInfo Name)
-> TermTypeM (NonEmpty (CaseBase Info VName), PatType, [VName])
checkCases PatType
mt NonEmpty (CaseBase NoInfo Name)
cs
      (PatType
brancht, [VName]
retext) <- SrcLoc -> PatType -> PatType -> TermTypeM (PatType, [VName])
unifyBranchTypes (CaseBase NoInfo Name -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf CaseBase NoInfo Name
c) PatType
c_t PatType
cs_t
      let t :: PatType
t =
            PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
addAliases
              PatType
brancht
              (Aliasing -> Aliasing -> Aliasing
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` (VName -> Alias) -> Names -> Aliasing
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map VName -> Alias
AliasBound ([Occurence] -> Names
allConsumed [Occurence]
dflow))
      (NonEmpty (CaseBase Info VName), PatType, [VName])
-> TermTypeM (NonEmpty (CaseBase Info VName), PatType, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return (CaseBase Info VName
-> NonEmpty (CaseBase Info VName) -> NonEmpty (CaseBase Info VName)
forall a. a -> NonEmpty a -> NonEmpty a
NE.cons CaseBase Info VName
c' NonEmpty (CaseBase Info VName)
cs', PatType
t, [VName]
retext)

checkCase ::
  PatType ->
  CaseBase NoInfo Name ->
  TermTypeM (CaseBase Info VName, PatType, [VName])
checkCase :: PatType
-> CaseBase NoInfo Name
-> TermTypeM (CaseBase Info VName, PatType, [VName])
checkCase PatType
mt (CasePat UncheckedPat
p UncheckedExp
e SrcLoc
loc) =
  [SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM (CaseBase Info VName, PatType, [VName]))
-> TermTypeM (CaseBase Info VName, PatType, [VName])
forall a.
[SizeBinder VName]
-> UncheckedPat
-> InferredType
-> (Pat -> TermTypeM a)
-> TermTypeM a
bindingPat [] UncheckedPat
p (PatType -> InferredType
Ascribed PatType
mt) ((Pat -> TermTypeM (CaseBase Info VName, PatType, [VName]))
 -> TermTypeM (CaseBase Info VName, PatType, [VName]))
-> (Pat -> TermTypeM (CaseBase Info VName, PatType, [VName]))
-> TermTypeM (CaseBase Info VName, PatType, [VName])
forall a b. (a -> b) -> a -> b
$ \Pat
p' -> do
    Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
    (PatType
t, [VName]
retext) <- SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType SrcLoc
loc (Pat -> Map VName Ident
forall (f :: * -> *).
Functor f =>
PatBase f VName -> Map VName (IdentBase f VName)
patternMap Pat
p') (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Exp -> TermTypeM PatType
expTypeFully Exp
e'
    (CaseBase Info VName, PatType, [VName])
-> TermTypeM (CaseBase Info VName, PatType, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return (Pat -> Exp -> SrcLoc -> CaseBase Info VName
forall (f :: * -> *) vn.
PatBase f vn -> ExpBase f vn -> SrcLoc -> CaseBase f vn
CasePat Pat
p' Exp
e' SrcLoc
loc, PatType
t, [VName]
retext)

-- | An unmatched pattern. Used in in the generation of
-- unmatched pattern warnings by the type checker.
data Unmatched p
  = UnmatchedNum p [PatLit]
  | UnmatchedBool p
  | UnmatchedConstr p
  | Unmatched p
  deriving (a -> Unmatched b -> Unmatched a
(a -> b) -> Unmatched a -> Unmatched b
(forall a b. (a -> b) -> Unmatched a -> Unmatched b)
-> (forall a b. a -> Unmatched b -> Unmatched a)
-> Functor Unmatched
forall a b. a -> Unmatched b -> Unmatched a
forall a b. (a -> b) -> Unmatched a -> Unmatched b
forall (f :: * -> *).
(forall a b. (a -> b) -> f a -> f b)
-> (forall a b. a -> f b -> f a) -> Functor f
<$ :: a -> Unmatched b -> Unmatched a
$c<$ :: forall a b. a -> Unmatched b -> Unmatched a
fmap :: (a -> b) -> Unmatched a -> Unmatched b
$cfmap :: forall a b. (a -> b) -> Unmatched a -> Unmatched b
Functor, Int -> Unmatched p -> ShowS
[Unmatched p] -> ShowS
Unmatched p -> String
(Int -> Unmatched p -> ShowS)
-> (Unmatched p -> String)
-> ([Unmatched p] -> ShowS)
-> Show (Unmatched p)
forall p. Show p => Int -> Unmatched p -> ShowS
forall p. Show p => [Unmatched p] -> ShowS
forall p. Show p => Unmatched p -> String
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Unmatched p] -> ShowS
$cshowList :: forall p. Show p => [Unmatched p] -> ShowS
show :: Unmatched p -> String
$cshow :: forall p. Show p => Unmatched p -> String
showsPrec :: Int -> Unmatched p -> ShowS
$cshowsPrec :: forall p. Show p => Int -> Unmatched p -> ShowS
Show)

instance Pretty (Unmatched (PatBase Info VName)) where
  ppr :: Unmatched Pat -> Doc
ppr Unmatched Pat
um = case Unmatched Pat
um of
    (UnmatchedNum Pat
p [PatLit]
nums) -> Pat -> Doc
forall v (f :: * -> *).
(Eq v, IsName v, Annot f) =>
PatBase f v -> Doc
ppr' Pat
p Doc -> Doc -> Doc
<+> Doc
"where p is not one of" Doc -> Doc -> Doc
<+> [PatLit] -> Doc
forall a. Pretty a => a -> Doc
ppr [PatLit]
nums
    (UnmatchedBool Pat
p) -> Pat -> Doc
forall v (f :: * -> *).
(Eq v, IsName v, Annot f) =>
PatBase f v -> Doc
ppr' Pat
p
    (UnmatchedConstr Pat
p) -> Pat -> Doc
forall v (f :: * -> *).
(Eq v, IsName v, Annot f) =>
PatBase f v -> Doc
ppr' Pat
p
    (Unmatched Pat
p) -> Pat -> Doc
forall v (f :: * -> *).
(Eq v, IsName v, Annot f) =>
PatBase f v -> Doc
ppr' Pat
p
    where
      ppr' :: PatBase f v -> Doc
ppr' (PatAscription PatBase f v
p TypeDeclBase f v
t SrcLoc
_) = PatBase f v -> Doc
forall a. Pretty a => a -> Doc
ppr PatBase f v
p Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
":" Doc -> Doc -> Doc
<+> TypeDeclBase f v -> Doc
forall a. Pretty a => a -> Doc
ppr TypeDeclBase f v
t
      ppr' (PatParens PatBase f v
p SrcLoc
_) = Doc -> Doc
parens (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ PatBase f v -> Doc
ppr' PatBase f v
p
      ppr' (Id v
v f PatType
_ SrcLoc
_) = v -> Doc
forall v. IsName v => v -> Doc
pprName v
v
      ppr' (TuplePat [PatBase f v]
pats SrcLoc
_) = Doc -> Doc
parens (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ [Doc] -> Doc
commasep ([Doc] -> Doc) -> [Doc] -> Doc
forall a b. (a -> b) -> a -> b
$ (PatBase f v -> Doc) -> [PatBase f v] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map PatBase f v -> Doc
ppr' [PatBase f v]
pats
      ppr' (RecordPat [(Name, PatBase f v)]
fs SrcLoc
_) = Doc -> Doc
braces (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ [Doc] -> Doc
commasep ([Doc] -> Doc) -> [Doc] -> Doc
forall a b. (a -> b) -> a -> b
$ ((Name, PatBase f v) -> Doc) -> [(Name, PatBase f v)] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map (Name, PatBase f v) -> Doc
ppField [(Name, PatBase f v)]
fs
        where
          ppField :: (Name, PatBase f v) -> Doc
ppField (Name
name, PatBase f v
t) = String -> Doc
text (Name -> String
nameToString Name
name) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
equals Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> PatBase f v -> Doc
ppr' PatBase f v
t
      ppr' Wildcard {} = Doc
"_"
      ppr' (PatLit PatLit
e f PatType
_ SrcLoc
_) = PatLit -> Doc
forall a. Pretty a => a -> Doc
ppr PatLit
e
      ppr' (PatConstr Name
n f PatType
_ [PatBase f v]
ps SrcLoc
_) = Doc
"#" Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Name -> Doc
forall a. Pretty a => a -> Doc
ppr Name
n Doc -> Doc -> Doc
<+> [Doc] -> Doc
sep ((PatBase f v -> Doc) -> [PatBase f v] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map PatBase f v -> Doc
ppr' [PatBase f v]
ps)

checkUnmatched :: Exp -> TermTypeM ()
checkUnmatched :: Exp -> TermTypeM ()
checkUnmatched Exp
e = TermTypeM Exp -> TermTypeM ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (TermTypeM Exp -> TermTypeM ()) -> TermTypeM Exp -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Exp -> TermTypeM ()
forall (m :: * -> *). MonadTypeChecker m => Exp -> m ()
checkUnmatched' Exp
e TermTypeM () -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> ASTMapper TermTypeM -> Exp -> TermTypeM Exp
forall x (m :: * -> *).
(ASTMappable x, Monad m) =>
ASTMapper m -> x -> m x
astMap ASTMapper TermTypeM
tv Exp
e
  where
    checkUnmatched' :: Exp -> m ()
checkUnmatched' (AppExp (Match Exp
_ NonEmpty (CaseBase Info VName)
cs SrcLoc
loc) Info AppRes
_) =
      let ps :: NonEmpty Pat
ps = (CaseBase Info VName -> Pat)
-> NonEmpty (CaseBase Info VName) -> NonEmpty Pat
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (\(CasePat Pat
p Exp
_ SrcLoc
_) -> Pat
p) NonEmpty (CaseBase Info VName)
cs
       in case [Pat] -> [Match]
unmatched ([Pat] -> [Match]) -> [Pat] -> [Match]
forall a b. (a -> b) -> a -> b
$ NonEmpty Pat -> [Pat]
forall a. NonEmpty a -> [a]
NE.toList NonEmpty Pat
ps of
            [] -> () -> m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
            [Match]
ps' ->
              SrcLoc -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
                Doc
"Unmatched cases in match expression:"
                  Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 ([Doc] -> Doc
stack ((Match -> Doc) -> [Match] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map Match -> Doc
forall a. Pretty a => a -> Doc
ppr [Match]
ps'))
    checkUnmatched' Exp
_ = () -> m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
    tv :: ASTMapper TermTypeM
tv =
      ASTMapper :: forall (m :: * -> *).
(Exp -> m Exp)
-> (VName -> m VName)
-> (QualName VName -> m (QualName VName))
-> (StructType -> m StructType)
-> (PatType -> m PatType)
-> ASTMapper m
ASTMapper
        { mapOnExp :: Exp -> TermTypeM Exp
mapOnExp =
            \Exp
e' -> Exp -> TermTypeM ()
forall (m :: * -> *). MonadTypeChecker m => Exp -> m ()
checkUnmatched' Exp
e' TermTypeM () -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e',
          mapOnName :: VName -> TermTypeM VName
mapOnName = VName -> TermTypeM VName
forall (f :: * -> *) a. Applicative f => a -> f a
pure,
          mapOnQualName :: QualName VName -> TermTypeM (QualName VName)
mapOnQualName = QualName VName -> TermTypeM (QualName VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure,
          mapOnStructType :: StructType -> TermTypeM StructType
mapOnStructType = StructType -> TermTypeM StructType
forall (f :: * -> *) a. Applicative f => a -> f a
pure,
          mapOnPatType :: PatType -> TermTypeM PatType
mapOnPatType = PatType -> TermTypeM PatType
forall (f :: * -> *) a. Applicative f => a -> f a
pure
        }

checkIdent :: IdentBase NoInfo Name -> TermTypeM Ident
checkIdent :: IdentBase NoInfo Name -> TermTypeM Ident
checkIdent (Ident Name
name NoInfo PatType
_ SrcLoc
loc) = do
  (QualName [VName]
_ VName
name', PatType
vt) <- SrcLoc -> QualName Name -> TermTypeM (QualName VName, PatType)
forall (m :: * -> *).
MonadTypeChecker m =>
SrcLoc -> QualName Name -> m (QualName VName, PatType)
lookupVar SrcLoc
loc (Name -> QualName Name
forall v. v -> QualName v
qualName Name
name)
  Ident -> TermTypeM Ident
forall (m :: * -> *) a. Monad m => a -> m a
return (Ident -> TermTypeM Ident) -> Ident -> TermTypeM Ident
forall a b. (a -> b) -> a -> b
$ VName -> Info PatType -> SrcLoc -> Ident
forall (f :: * -> *) vn.
vn -> f PatType -> SrcLoc -> IdentBase f vn
Ident VName
name' (PatType -> Info PatType
forall a. a -> Info a
Info PatType
vt) SrcLoc
loc

checkSlice :: UncheckedSlice -> TermTypeM Slice
checkSlice :: SliceBase NoInfo Name -> TermTypeM Slice
checkSlice = (DimIndexBase NoInfo Name -> TermTypeM (DimIndexBase Info VName))
-> SliceBase NoInfo Name -> TermTypeM Slice
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM DimIndexBase NoInfo Name -> TermTypeM (DimIndexBase Info VName)
checkDimIndex
  where
    checkDimIndex :: DimIndexBase NoInfo Name -> TermTypeM (DimIndexBase Info VName)
checkDimIndex (DimFix UncheckedExp
i) =
      Exp -> DimIndexBase Info VName
forall (f :: * -> *) vn. ExpBase f vn -> DimIndexBase f vn
DimFix (Exp -> DimIndexBase Info VName)
-> TermTypeM Exp -> TermTypeM (DimIndexBase Info VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (String -> [PrimType] -> Exp -> TermTypeM Exp
require String
"use as index" [PrimType]
anySignedType (Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
i)
    checkDimIndex (DimSlice Maybe UncheckedExp
i Maybe UncheckedExp
j Maybe UncheckedExp
s) =
      Maybe Exp -> Maybe Exp -> Maybe Exp -> DimIndexBase Info VName
forall (f :: * -> *) vn.
Maybe (ExpBase f vn)
-> Maybe (ExpBase f vn)
-> Maybe (ExpBase f vn)
-> DimIndexBase f vn
DimSlice (Maybe Exp -> Maybe Exp -> Maybe Exp -> DimIndexBase Info VName)
-> TermTypeM (Maybe Exp)
-> TermTypeM (Maybe Exp -> Maybe Exp -> DimIndexBase Info VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe UncheckedExp -> TermTypeM (Maybe Exp)
check Maybe UncheckedExp
i TermTypeM (Maybe Exp -> Maybe Exp -> DimIndexBase Info VName)
-> TermTypeM (Maybe Exp)
-> TermTypeM (Maybe Exp -> DimIndexBase Info VName)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> Maybe UncheckedExp -> TermTypeM (Maybe Exp)
check Maybe UncheckedExp
j TermTypeM (Maybe Exp -> DimIndexBase Info VName)
-> TermTypeM (Maybe Exp) -> TermTypeM (DimIndexBase Info VName)
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> Maybe UncheckedExp -> TermTypeM (Maybe Exp)
check Maybe UncheckedExp
s

    check :: Maybe UncheckedExp -> TermTypeM (Maybe Exp)
check =
      TermTypeM (Maybe Exp)
-> (UncheckedExp -> TermTypeM (Maybe Exp))
-> Maybe UncheckedExp
-> TermTypeM (Maybe Exp)
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (Maybe Exp -> TermTypeM (Maybe Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe Exp
forall a. Maybe a
Nothing) ((UncheckedExp -> TermTypeM (Maybe Exp))
 -> Maybe UncheckedExp -> TermTypeM (Maybe Exp))
-> (UncheckedExp -> TermTypeM (Maybe Exp))
-> Maybe UncheckedExp
-> TermTypeM (Maybe Exp)
forall a b. (a -> b) -> a -> b
$
        (Exp -> Maybe Exp) -> TermTypeM Exp -> TermTypeM (Maybe Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Exp -> Maybe Exp
forall a. a -> Maybe a
Just (TermTypeM Exp -> TermTypeM (Maybe Exp))
-> (Exp -> TermTypeM Exp) -> Exp -> TermTypeM (Maybe Exp)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> StructType -> Exp -> TermTypeM Exp
unifies String
"use as index" (ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) ())
-> PrimType -> ScalarTypeBase (DimDecl VName) ()
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int64) (Exp -> TermTypeM (Maybe Exp))
-> (UncheckedExp -> TermTypeM Exp)
-> UncheckedExp
-> TermTypeM (Maybe Exp)
forall (m :: * -> *) b c a.
Monad m =>
(b -> m c) -> (a -> m b) -> a -> m c
<=< UncheckedExp -> TermTypeM Exp
checkExp

-- The number of dimensions affected by this slice (so the minimum
-- rank of the array we are slicing).
sliceDims :: Slice -> Int
sliceDims :: Slice -> Int
sliceDims = Slice -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length

sequentially :: TermTypeM a -> (a -> Occurences -> TermTypeM b) -> TermTypeM b
sequentially :: TermTypeM a -> (a -> [Occurence] -> TermTypeM b) -> TermTypeM b
sequentially TermTypeM a
m1 a -> [Occurence] -> TermTypeM b
m2 = do
  (a
a, [Occurence]
m1flow) <- TermTypeM a -> TermTypeM (a, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences TermTypeM a
m1
  (b
b, [Occurence]
m2flow) <- TermTypeM b -> TermTypeM (b, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences (TermTypeM b -> TermTypeM (b, [Occurence]))
-> TermTypeM b -> TermTypeM (b, [Occurence])
forall a b. (a -> b) -> a -> b
$ a -> [Occurence] -> TermTypeM b
m2 a
a [Occurence]
m1flow
  [Occurence] -> TermTypeM ()
occur ([Occurence] -> TermTypeM ()) -> [Occurence] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ [Occurence]
m1flow [Occurence] -> [Occurence] -> [Occurence]
`seqOccurences` [Occurence]
m2flow
  b -> TermTypeM b
forall (m :: * -> *) a. Monad m => a -> m a
return b
b

type Arg = (Exp, PatType, Occurences, SrcLoc)

argExp :: Arg -> Exp
argExp :: Arg -> Exp
argExp (Exp
e, PatType
_, [Occurence]
_, SrcLoc
_) = Exp
e

argType :: Arg -> PatType
argType :: Arg -> PatType
argType (Exp
_, PatType
t, [Occurence]
_, SrcLoc
_) = PatType
t

checkArg :: UncheckedExp -> TermTypeM Arg
checkArg :: UncheckedExp -> TermTypeM Arg
checkArg UncheckedExp
arg = do
  (Exp
arg', [Occurence]
dflow) <- TermTypeM Exp -> TermTypeM (Exp, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences (TermTypeM Exp -> TermTypeM (Exp, [Occurence]))
-> TermTypeM Exp -> TermTypeM (Exp, [Occurence])
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
arg
  PatType
arg_t <- Exp -> TermTypeM PatType
expType Exp
arg'
  Arg -> TermTypeM Arg
forall (m :: * -> *) a. Monad m => a -> m a
return (Exp
arg', PatType
arg_t, [Occurence]
dflow, Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
arg')

instantiateDimsInType ::
  SrcLoc ->
  RigidSource ->
  TypeBase (DimDecl VName) als ->
  TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInType :: SrcLoc
-> RigidSource
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInType SrcLoc
tloc RigidSource
rsrc =
  SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
tloc String
"d" (Rigidity
 -> TypeBase (DimDecl VName) als
 -> TermTypeM (TypeBase (DimDecl VName) als, [VName]))
-> Rigidity
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
forall a b. (a -> b) -> a -> b
$ RigidSource -> Rigidity
Rigid RigidSource
rsrc

instantiateDimsInReturnType ::
  SrcLoc ->
  Maybe (QualName VName) ->
  TypeBase (DimDecl VName) als ->
  TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInReturnType :: SrcLoc
-> Maybe (QualName VName)
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInReturnType SrcLoc
tloc Maybe (QualName VName)
fname =
  SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
tloc String
"ret" (Rigidity
 -> TypeBase (DimDecl VName) als
 -> TermTypeM (TypeBase (DimDecl VName) als, [VName]))
-> Rigidity
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
forall a b. (a -> b) -> a -> b
$ RigidSource -> Rigidity
Rigid (RigidSource -> Rigidity) -> RigidSource -> Rigidity
forall a b. (a -> b) -> a -> b
$ Maybe (QualName VName) -> RigidSource
RigidRet Maybe (QualName VName)
fname

-- Some information about the function/operator we are trying to
-- apply, and how many arguments it has previously accepted.  Used for
-- generating nicer type errors.
type ApplyOp = (Maybe (QualName VName), Int)

checkApply ::
  SrcLoc ->
  ApplyOp ->
  PatType ->
  Arg ->
  TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply :: SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply
  SrcLoc
loc
  (Maybe (QualName VName)
fname, Int
_)
  (Scalar (Arrow Aliasing
as PName
pname PatType
tp1 PatType
tp2))
  (Exp
argexp, PatType
argtype, [Occurence]
dflow, SrcLoc
argloc) =
    Checking
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (Maybe (QualName VName)
-> Exp -> StructType -> StructType -> Checking
CheckingApply Maybe (QualName VName)
fname Exp
argexp (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
tp1) (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
argtype)) (TermTypeM (PatType, PatType, Maybe VName, [VName])
 -> TermTypeM (PatType, PatType, Maybe VName, [VName]))
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
forall a b. (a -> b) -> a -> b
$ do
      Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
expect (SrcLoc -> String -> Usage
mkUsage SrcLoc
argloc String
"use as function argument") (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
tp1) (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
argtype)

      -- Perform substitutions of instantiated variables in the types.
      PatType
tp1' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
tp1
      (PatType
tp2', [VName]
ext) <- SrcLoc
-> Maybe (QualName VName)
-> PatType
-> TermTypeM (PatType, [VName])
forall als.
SrcLoc
-> Maybe (QualName VName)
-> TypeBase (DimDecl VName) als
-> TermTypeM (TypeBase (DimDecl VName) als, [VName])
instantiateDimsInReturnType SrcLoc
loc Maybe (QualName VName)
fname (PatType -> TermTypeM (PatType, [VName]))
-> TermTypeM PatType -> TermTypeM (PatType, [VName])
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
tp2
      PatType
argtype' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
argtype

      -- Check whether this would produce an impossible return type.
      let (Names
_, Names
tp2_paramdims, Names
_) = StructType -> (Names, Names, Names)
dimUses (StructType -> (Names, Names, Names))
-> StructType -> (Names, Names, Names)
forall a b. (a -> b) -> a -> b
$ PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
tp2'
      case (VName -> Bool) -> [VName] -> [VName]
forall a. (a -> Bool) -> [a] -> [a]
filter (VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
tp2_paramdims) [VName]
ext of
        [] -> () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
        [VName]
ext_paramdims -> do
          let onDim :: DimDecl VName -> DimDecl VName
onDim (NamedDim QualName VName
qn)
                | QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
qn VName -> [VName] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [VName]
ext_paramdims = Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim (Maybe VName -> DimDecl VName) -> Maybe VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> Maybe VName
forall a. a -> Maybe a
Just (VName -> Maybe VName) -> VName -> Maybe VName
forall a b. (a -> b) -> a -> b
$ QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
qn
              onDim DimDecl VName
d = DimDecl VName
d
          SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
            Doc
"Anonymous size would appear in function parameter of return type:"
              Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (PatType -> Doc
forall a. Pretty a => a -> Doc
ppr ((DimDecl VName -> DimDecl VName) -> PatType -> PatType
forall (p :: * -> * -> *) a b c.
Bifunctor p =>
(a -> b) -> p a c -> p b c
first DimDecl VName -> DimDecl VName
onDim PatType
tp2'))
              Doc -> Doc -> Doc
</> String -> Doc
textwrap String
"This is usually because a higher-order function is used with functional arguments that return anonymous sizes, which are then used as parameters of other function arguments."

      [Occurence] -> TermTypeM ()
occur [Aliasing -> SrcLoc -> Occurence
observation Aliasing
as SrcLoc
loc]

      [Occurence] -> TermTypeM ()
checkOccurences [Occurence]
dflow

      case [Occurence] -> Maybe Occurence
anyConsumption [Occurence]
dflow of
        Just Occurence
c ->
          let msg :: String
msg = String
"type of expression with consumption at " String -> ShowS
forall a. [a] -> [a] -> [a]
++ SrcLoc -> String
forall a. Located a => a -> String
locStr (Occurence -> SrcLoc
location Occurence
c)
           in Usage -> String -> PatType -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> String -> TypeBase dim as -> m ()
zeroOrderType (SrcLoc -> String -> Usage
mkUsage SrcLoc
argloc String
"potential consumption in expression") String
msg PatType
tp1
        Maybe Occurence
_ -> () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

      [Occurence]
occurs <- ([Occurence]
dflow [Occurence] -> [Occurence] -> [Occurence]
`seqOccurences`) ([Occurence] -> [Occurence])
-> TermTypeM [Occurence] -> TermTypeM [Occurence]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> SrcLoc -> PatType -> Diet -> TermTypeM [Occurence]
consumeArg SrcLoc
argloc PatType
argtype' (PatType -> Diet
forall shape as. TypeBase shape as -> Diet
diet PatType
tp1')

      SrcLoc -> Aliasing -> TermTypeM ()
checkIfConsumable SrcLoc
loc (Aliasing -> TermTypeM ()) -> Aliasing -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ (VName -> Alias) -> Names -> Aliasing
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map VName -> Alias
AliasBound (Names -> Aliasing) -> Names -> Aliasing
forall a b. (a -> b) -> a -> b
$ [Occurence] -> Names
allConsumed [Occurence]
occurs
      [Occurence] -> TermTypeM ()
occur [Occurence]
occurs

      (Maybe VName
argext, TypeSubs
parsubst) <-
        case PName
pname of
          Named VName
pname' -> do
            (DimDecl VName
d, Maybe VName
argext) <- PatType -> Exp -> TermTypeM (DimDecl VName, Maybe VName)
forall dim as.
TypeBase dim as -> Exp -> TermTypeM (DimDecl VName, Maybe VName)
sizeSubst PatType
tp1' Exp
argexp
            (Maybe VName, TypeSubs) -> TermTypeM (Maybe VName, TypeSubs)
forall (m :: * -> *) a. Monad m => a -> m a
return
              ( Maybe VName
argext,
                (VName -> Map VName (Subst StructType) -> Maybe (Subst StructType)
forall k a. Ord k => k -> Map k a -> Maybe a
`M.lookup` VName -> Subst StructType -> Map VName (Subst StructType)
forall k a. k -> a -> Map k a
M.singleton VName
pname' (DimDecl VName -> Subst StructType
forall t. DimDecl VName -> Subst t
SizeSubst DimDecl VName
d))
              )
          PName
_ -> (Maybe VName, TypeSubs) -> TermTypeM (Maybe VName, TypeSubs)
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe VName
forall a. Maybe a
Nothing, Maybe (Subst StructType) -> TypeSubs
forall a b. a -> b -> a
const Maybe (Subst StructType)
forall a. Maybe a
Nothing)
      let tp2'' :: PatType
tp2'' = TypeSubs -> PatType -> PatType
forall a. Substitutable a => TypeSubs -> a -> a
applySubst TypeSubs
parsubst (PatType -> PatType) -> PatType -> PatType
forall a b. (a -> b) -> a -> b
$ PatType -> Diet -> PatType -> PatType
returnType PatType
tp2' (PatType -> Diet
forall shape as. TypeBase shape as -> Diet
diet PatType
tp1') PatType
argtype'

      (PatType, PatType, Maybe VName, [VName])
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return (PatType
tp1', PatType
tp2'', Maybe VName
argext, [VName]
ext)
    where
      sizeSubst :: TypeBase dim as -> Exp -> TermTypeM (DimDecl VName, Maybe VName)
sizeSubst (Scalar (Prim (Signed IntType
Int64))) Exp
e = Maybe (QualName VName)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromArg Maybe (QualName VName)
fname Exp
e
      sizeSubst TypeBase dim as
_ Exp
_ = (DimDecl VName, Maybe VName)
-> TermTypeM (DimDecl VName, Maybe VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim Maybe VName
forall a. Maybe a
Nothing, Maybe VName
forall a. Maybe a
Nothing)
checkApply SrcLoc
loc ApplyOp
fname tfun :: PatType
tfun@(Scalar TypeVar {}) Arg
arg = do
  StructType
tv <- SrcLoc -> String -> TermTypeM StructType
forall (m :: * -> *) als dim.
(MonadUnify m, Monoid als) =>
SrcLoc -> String -> m (TypeBase dim als)
newTypeVar SrcLoc
loc String
"b"
  -- Change the uniqueness of the argument type because we never want
  -- to infer that a function is consuming.
  Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"use as function") (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
tfun) (StructType -> TermTypeM ()) -> StructType -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ ()
-> PName
-> StructType
-> StructType
-> ScalarTypeBase (DimDecl VName) ()
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow ()
forall a. Monoid a => a
mempty PName
Unnamed (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (Arg -> PatType
argType Arg
arg) StructType -> Uniqueness -> StructType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique) StructType
tv
  PatType
tfun' <- PatType -> TermTypeM PatType
forall (m :: * -> *). MonadUnify m => PatType -> m PatType
normPatType PatType
tfun
  SrcLoc
-> ApplyOp
-> PatType
-> Arg
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
checkApply SrcLoc
loc ApplyOp
fname PatType
tfun' Arg
arg
checkApply SrcLoc
loc (Maybe (QualName VName)
fname, Int
prev_applied) PatType
ftype (Exp
argexp, PatType
_, [Occurence]
_, SrcLoc
_) = do
  let fname' :: Doc
fname' = Doc -> (QualName VName -> Doc) -> Maybe (QualName VName) -> Doc
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Doc
"expression" (Doc -> Doc
pquote (Doc -> Doc) -> (QualName VName -> Doc) -> QualName VName -> Doc
forall b c a. (b -> c) -> (a -> b) -> a -> c
. QualName VName -> Doc
forall a. Pretty a => a -> Doc
ppr) Maybe (QualName VName)
fname

  SrcLoc
-> Notes
-> Doc
-> TermTypeM (PatType, PatType, Maybe VName, [VName])
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM (PatType, PatType, Maybe VName, [VName]))
-> Doc -> TermTypeM (PatType, PatType, Maybe VName, [VName])
forall a b. (a -> b) -> a -> b
$
    if Int
prev_applied Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0
      then
        Doc
"Cannot apply" Doc -> Doc -> Doc
<+> Doc
fname' Doc -> Doc -> Doc
<+> Doc
"as function, as it has type:"
          Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (PatType -> Doc
forall a. Pretty a => a -> Doc
ppr PatType
ftype)
      else
        Doc
"Cannot apply" Doc -> Doc -> Doc
<+> Doc
fname' Doc -> Doc -> Doc
<+> Doc
"to argument #" Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Int -> Doc
forall a. Pretty a => a -> Doc
ppr (Int
prev_applied Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
          Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (String -> Doc
forall a. Pretty a => a -> Doc
shorten (String -> Doc) -> String -> Doc
forall a b. (a -> b) -> a -> b
$ Doc -> String
forall a. Pretty a => a -> String
pretty (Doc -> String) -> Doc -> String
forall a b. (a -> b) -> a -> b
$ Doc -> Doc
flatten (Doc -> Doc) -> Doc -> Doc
forall a b. (a -> b) -> a -> b
$ Exp -> Doc
forall a. Pretty a => a -> Doc
ppr Exp
argexp) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
","
          Doc -> Doc -> Doc
<+/> Doc
"as"
          Doc -> Doc -> Doc
<+> Doc
fname'
          Doc -> Doc -> Doc
<+> Doc
"only takes"
          Doc -> Doc -> Doc
<+> Int -> Doc
forall a. Pretty a => a -> Doc
ppr Int
prev_applied
          Doc -> Doc -> Doc
<+> Doc
arguments Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
  where
    arguments :: Doc
arguments
      | Int
prev_applied Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
1 = Doc
"argument"
      | Bool
otherwise = Doc
"arguments"

isInt64 :: Exp -> Maybe Int64
isInt64 :: Exp -> Maybe Int64
isInt64 (Literal (SignedValue (Int64Value Int64
k')) SrcLoc
_) = Int64 -> Maybe Int64
forall a. a -> Maybe a
Just (Int64 -> Maybe Int64) -> Int64 -> Maybe Int64
forall a b. (a -> b) -> a -> b
$ Int64 -> Int64
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int64
k'
isInt64 (IntLit Integer
k' Info PatType
_ SrcLoc
_) = Int64 -> Maybe Int64
forall a. a -> Maybe a
Just (Int64 -> Maybe Int64) -> Int64 -> Maybe Int64
forall a b. (a -> b) -> a -> b
$ Integer -> Int64
forall a. Num a => Integer -> a
fromInteger Integer
k'
isInt64 (Negate Exp
x SrcLoc
_) = Int64 -> Int64
forall a. Num a => a -> a
negate (Int64 -> Int64) -> Maybe Int64 -> Maybe Int64
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Exp -> Maybe Int64
isInt64 Exp
x
isInt64 Exp
_ = Maybe Int64
forall a. Maybe a
Nothing

maybeDimFromExp :: Exp -> Maybe (DimDecl VName)
maybeDimFromExp :: Exp -> Maybe (DimDecl VName)
maybeDimFromExp (Var QualName VName
v Info PatType
_ SrcLoc
_) = DimDecl VName -> Maybe (DimDecl VName)
forall a. a -> Maybe a
Just (DimDecl VName -> Maybe (DimDecl VName))
-> DimDecl VName -> Maybe (DimDecl VName)
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim QualName VName
v
maybeDimFromExp (Parens Exp
e SrcLoc
_) = Exp -> Maybe (DimDecl VName)
maybeDimFromExp Exp
e
maybeDimFromExp (QualParens (QualName VName, SrcLoc)
_ Exp
e SrcLoc
_) = Exp -> Maybe (DimDecl VName)
maybeDimFromExp Exp
e
maybeDimFromExp Exp
e = Int -> DimDecl VName
forall vn. Int -> DimDecl vn
ConstDim (Int -> DimDecl VName) -> (Int64 -> Int) -> Int64 -> DimDecl VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int64 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int64 -> DimDecl VName) -> Maybe Int64 -> Maybe (DimDecl VName)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Exp -> Maybe Int64
isInt64 Exp
e

dimFromExp :: (Exp -> SizeSource) -> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromExp :: (Exp -> SizeSource)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromExp Exp -> SizeSource
rf (Parens Exp
e SrcLoc
_) = (Exp -> SizeSource)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromExp Exp -> SizeSource
rf Exp
e
dimFromExp Exp -> SizeSource
rf (QualParens (QualName VName, SrcLoc)
_ Exp
e SrcLoc
_) = (Exp -> SizeSource)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromExp Exp -> SizeSource
rf Exp
e
dimFromExp Exp -> SizeSource
rf Exp
e
  | Just DimDecl VName
d <- Exp -> Maybe (DimDecl VName)
maybeDimFromExp Exp
e =
    (DimDecl VName, Maybe VName)
-> TermTypeM (DimDecl VName, Maybe VName)
forall (m :: * -> *) a. Monad m => a -> m a
return (DimDecl VName
d, Maybe VName
forall a. Maybe a
Nothing)
  | Bool
otherwise =
    SrcLoc -> SizeSource -> TermTypeM (DimDecl VName, Maybe VName)
extSize (Exp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Exp
e) (SizeSource -> TermTypeM (DimDecl VName, Maybe VName))
-> SizeSource -> TermTypeM (DimDecl VName, Maybe VName)
forall a b. (a -> b) -> a -> b
$ Exp -> SizeSource
rf Exp
e

dimFromArg :: Maybe (QualName VName) -> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromArg :: Maybe (QualName VName)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromArg Maybe (QualName VName)
fname = (Exp -> SizeSource)
-> Exp -> TermTypeM (DimDecl VName, Maybe VName)
dimFromExp ((Exp -> SizeSource)
 -> Exp -> TermTypeM (DimDecl VName, Maybe VName))
-> (Exp -> SizeSource)
-> Exp
-> TermTypeM (DimDecl VName, Maybe VName)
forall a b. (a -> b) -> a -> b
$ FName -> ExpBase NoInfo VName -> SizeSource
SourceArg (Maybe (QualName VName) -> FName
FName Maybe (QualName VName)
fname) (ExpBase NoInfo VName -> SizeSource)
-> (Exp -> ExpBase NoInfo VName) -> Exp -> SizeSource
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> ExpBase NoInfo VName
bareExp

-- | @returnType ret_type arg_diet arg_type@ gives result of applying
-- an argument the given types to a function with the given return
-- type, consuming the argument with the given diet.
returnType ::
  PatType ->
  Diet ->
  PatType ->
  PatType
returnType :: PatType -> Diet -> PatType -> PatType
returnType (Array Aliasing
_ Uniqueness
Unique ScalarTypeBase (DimDecl VName) ()
et ShapeDecl (DimDecl VName)
shape) Diet
_ PatType
_ =
  Aliasing
-> Uniqueness
-> ScalarTypeBase (DimDecl VName) ()
-> ShapeDecl (DimDecl VName)
-> PatType
forall dim as.
as
-> Uniqueness
-> ScalarTypeBase dim ()
-> ShapeDecl dim
-> TypeBase dim as
Array Aliasing
forall a. Monoid a => a
mempty Uniqueness
Unique ScalarTypeBase (DimDecl VName) ()
et ShapeDecl (DimDecl VName)
shape
returnType (Array Aliasing
als Uniqueness
Nonunique ScalarTypeBase (DimDecl VName) ()
et ShapeDecl (DimDecl VName)
shape) Diet
d PatType
arg =
  Aliasing
-> Uniqueness
-> ScalarTypeBase (DimDecl VName) ()
-> ShapeDecl (DimDecl VName)
-> PatType
forall dim as.
as
-> Uniqueness
-> ScalarTypeBase dim ()
-> ShapeDecl dim
-> TypeBase dim as
Array (Aliasing
als Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> Aliasing
arg_als) Uniqueness
Unique ScalarTypeBase (DimDecl VName) ()
et ShapeDecl (DimDecl VName)
shape -- Intentional!
  where
    arg_als :: Aliasing
arg_als = PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases (PatType -> Aliasing) -> PatType -> Aliasing
forall a b. (a -> b) -> a -> b
$ PatType -> Diet -> PatType
forall as shape.
Monoid as =>
TypeBase shape as -> Diet -> TypeBase shape as
maskAliases PatType
arg Diet
d
returnType (Scalar (Record Map Name PatType
fs)) Diet
d PatType
arg =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. Map Name (TypeBase dim as) -> ScalarTypeBase dim as
Record (Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ (PatType -> PatType) -> Map Name PatType -> Map Name PatType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (\PatType
et -> PatType -> Diet -> PatType -> PatType
returnType PatType
et Diet
d PatType
arg) Map Name PatType
fs
returnType (Scalar (Prim PrimType
t)) Diet
_ PatType
_ =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. PrimType -> ScalarTypeBase dim as
Prim PrimType
t
returnType (Scalar (TypeVar Aliasing
_ Uniqueness
Unique TypeName
t [TypeArg (DimDecl VName)]
targs)) Diet
_ PatType
_ =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar Aliasing
forall a. Monoid a => a
mempty Uniqueness
Unique TypeName
t [TypeArg (DimDecl VName)]
targs
returnType (Scalar (TypeVar Aliasing
als Uniqueness
Nonunique TypeName
t [TypeArg (DimDecl VName)]
targs)) Diet
d PatType
arg =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar (Aliasing
als Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> Aliasing
arg_als) Uniqueness
Unique TypeName
t [TypeArg (DimDecl VName)]
targs -- Intentional!
  where
    arg_als :: Aliasing
arg_als = PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases (PatType -> Aliasing) -> PatType -> Aliasing
forall a b. (a -> b) -> a -> b
$ PatType -> Diet -> PatType
forall as shape.
Monoid as =>
TypeBase shape as -> Diet -> TypeBase shape as
maskAliases PatType
arg Diet
d
returnType (Scalar (Arrow Aliasing
old_als PName
v PatType
t1 PatType
t2)) Diet
d PatType
arg =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Aliasing
-> PName
-> PatType
-> PatType
-> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as.
as
-> PName
-> TypeBase dim as
-> TypeBase dim as
-> ScalarTypeBase dim as
Arrow Aliasing
als PName
v (PatType
t1 PatType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` Aliasing
forall a. Monoid a => a
mempty) (PatType
t2 PatType -> Aliasing -> PatType
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` Aliasing
als)
  where
    -- Make sure to propagate the aliases of an existing closure.
    als :: Aliasing
als = Aliasing
old_als Aliasing -> Aliasing -> Aliasing
forall a. Semigroup a => a -> a -> a
<> PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases (PatType -> Diet -> PatType
forall as shape.
Monoid as =>
TypeBase shape as -> Diet -> TypeBase shape as
maskAliases PatType
arg Diet
d)
returnType (Scalar (Sum Map Name [PatType]
cs)) Diet
d PatType
arg =
  ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Map Name [PatType] -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. Map Name [TypeBase dim as] -> ScalarTypeBase dim as
Sum (Map Name [PatType] -> ScalarTypeBase (DimDecl VName) Aliasing)
-> Map Name [PatType] -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ (([PatType] -> [PatType])
-> Map Name [PatType] -> Map Name [PatType]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (([PatType] -> [PatType])
 -> Map Name [PatType] -> Map Name [PatType])
-> ((PatType -> PatType) -> [PatType] -> [PatType])
-> (PatType -> PatType)
-> Map Name [PatType]
-> Map Name [PatType]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (PatType -> PatType) -> [PatType] -> [PatType]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap) (\PatType
et -> PatType -> Diet -> PatType -> PatType
returnType PatType
et Diet
d PatType
arg) Map Name [PatType]
cs

-- | @t `maskAliases` d@ removes aliases (sets them to 'mempty') from
-- the parts of @t@ that are denoted as consumed by the 'Diet' @d@.
maskAliases ::
  Monoid as =>
  TypeBase shape as ->
  Diet ->
  TypeBase shape as
maskAliases :: TypeBase shape as -> Diet -> TypeBase shape as
maskAliases TypeBase shape as
t Diet
Consume = TypeBase shape as
t TypeBase shape as -> as -> TypeBase shape as
forall dim asf ast. TypeBase dim asf -> ast -> TypeBase dim ast
`setAliases` as
forall a. Monoid a => a
mempty
maskAliases TypeBase shape as
t Diet
Observe = TypeBase shape as
t
maskAliases (Scalar (Record Map Name (TypeBase shape as)
ets)) (RecordDiet Map Name Diet
ds) =
  ScalarTypeBase shape as -> TypeBase shape as
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase shape as -> TypeBase shape as)
-> ScalarTypeBase shape as -> TypeBase shape as
forall a b. (a -> b) -> a -> b
$ Map Name (TypeBase shape as) -> ScalarTypeBase shape as
forall dim as. Map Name (TypeBase dim as) -> ScalarTypeBase dim as
Record (Map Name (TypeBase shape as) -> ScalarTypeBase shape as)
-> Map Name (TypeBase shape as) -> ScalarTypeBase shape as
forall a b. (a -> b) -> a -> b
$ (TypeBase shape as -> Diet -> TypeBase shape as)
-> Map Name (TypeBase shape as)
-> Map Name Diet
-> Map Name (TypeBase shape as)
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
M.intersectionWith TypeBase shape as -> Diet -> TypeBase shape as
forall as shape.
Monoid as =>
TypeBase shape as -> Diet -> TypeBase shape as
maskAliases Map Name (TypeBase shape as)
ets Map Name Diet
ds
maskAliases TypeBase shape as
t FuncDiet {} = TypeBase shape as
t
maskAliases TypeBase shape as
_ Diet
_ = String -> TypeBase shape as
forall a. HasCallStack => String -> a
error String
"Invalid arguments passed to maskAliases."

consumeArg :: SrcLoc -> PatType -> Diet -> TermTypeM [Occurence]
consumeArg :: SrcLoc -> PatType -> Diet -> TermTypeM [Occurence]
consumeArg SrcLoc
loc (Scalar (Record Map Name PatType
ets)) (RecordDiet Map Name Diet
ds) =
  [[Occurence]] -> [Occurence]
forall (t :: * -> *) a. Foldable t => t [a] -> [a]
concat ([[Occurence]] -> [Occurence])
-> (Map Name [Occurence] -> [[Occurence]])
-> Map Name [Occurence]
-> [Occurence]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Name [Occurence] -> [[Occurence]]
forall k a. Map k a -> [a]
M.elems (Map Name [Occurence] -> [Occurence])
-> TermTypeM (Map Name [Occurence]) -> TermTypeM [Occurence]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ((PatType, Diet) -> TermTypeM [Occurence])
-> Map Name (PatType, Diet) -> TermTypeM (Map Name [Occurence])
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse ((PatType -> Diet -> TermTypeM [Occurence])
-> (PatType, Diet) -> TermTypeM [Occurence]
forall a b c. (a -> b -> c) -> (a, b) -> c
uncurry ((PatType -> Diet -> TermTypeM [Occurence])
 -> (PatType, Diet) -> TermTypeM [Occurence])
-> (PatType -> Diet -> TermTypeM [Occurence])
-> (PatType, Diet)
-> TermTypeM [Occurence]
forall a b. (a -> b) -> a -> b
$ SrcLoc -> PatType -> Diet -> TermTypeM [Occurence]
consumeArg SrcLoc
loc) ((PatType -> Diet -> (PatType, Diet))
-> Map Name PatType -> Map Name Diet -> Map Name (PatType, Diet)
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
M.intersectionWith (,) Map Name PatType
ets Map Name Diet
ds)
consumeArg SrcLoc
loc (Array Aliasing
_ Uniqueness
Nonunique ScalarTypeBase (DimDecl VName) ()
_ ShapeDecl (DimDecl VName)
_) Diet
Consume =
  SrcLoc -> Notes -> Doc -> TermTypeM [Occurence]
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty Doc
"Consuming parameter passed non-unique argument."
consumeArg SrcLoc
loc (Scalar (TypeVar Aliasing
_ Uniqueness
Nonunique TypeName
_ [TypeArg (DimDecl VName)]
_)) Diet
Consume =
  SrcLoc -> Notes -> Doc -> TermTypeM [Occurence]
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty Doc
"Consuming parameter passed non-unique argument."
consumeArg SrcLoc
loc (Scalar (Arrow Aliasing
_ PName
_ PatType
t1 PatType
_)) (FuncDiet Diet
d Diet
_)
  | Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ PatType -> Diet -> Bool
forall dim as. TypeBase dim as -> Diet -> Bool
contravariantArg PatType
t1 Diet
d =
    SrcLoc -> Notes -> Doc -> TermTypeM [Occurence]
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty Doc
"Non-consuming higher-order parameter passed consuming argument."
  where
    contravariantArg :: TypeBase dim as -> Diet -> Bool
contravariantArg (Array as
_ Uniqueness
Unique ScalarTypeBase dim ()
_ ShapeDecl dim
_) Diet
Observe =
      Bool
False
    contravariantArg (Scalar (TypeVar as
_ Uniqueness
Unique TypeName
_ [TypeArg dim]
_)) Diet
Observe =
      Bool
False
    contravariantArg (Scalar (Record Map Name (TypeBase dim as)
ets)) (RecordDiet Map Name Diet
ds) =
      Map Name Bool -> Bool
forall (t :: * -> *). Foldable t => t Bool -> Bool
and ((TypeBase dim as -> Diet -> Bool)
-> Map Name (TypeBase dim as) -> Map Name Diet -> Map Name Bool
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
M.intersectionWith TypeBase dim as -> Diet -> Bool
contravariantArg Map Name (TypeBase dim as)
ets Map Name Diet
ds)
    contravariantArg (Scalar (Arrow as
_ PName
_ TypeBase dim as
tp TypeBase dim as
tr)) (FuncDiet Diet
dp Diet
dr) =
      TypeBase dim as -> Diet -> Bool
contravariantArg TypeBase dim as
tp Diet
dp Bool -> Bool -> Bool
&& TypeBase dim as -> Diet -> Bool
contravariantArg TypeBase dim as
tr Diet
dr
    contravariantArg TypeBase dim as
_ Diet
_ =
      Bool
True
consumeArg SrcLoc
loc PatType
at Diet
Consume = [Occurence] -> TermTypeM [Occurence]
forall (m :: * -> *) a. Monad m => a -> m a
return [Aliasing -> SrcLoc -> Occurence
consumption (PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
at) SrcLoc
loc]
consumeArg SrcLoc
loc PatType
at Diet
_ = [Occurence] -> TermTypeM [Occurence]
forall (m :: * -> *) a. Monad m => a -> m a
return [Aliasing -> SrcLoc -> Occurence
observation (PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
at) SrcLoc
loc]

-- | Type-check a single expression in isolation.  This expression may
-- turn out to be polymorphic, in which case the list of type
-- parameters will be non-empty.
checkOneExp :: UncheckedExp -> TypeM ([TypeParam], Exp)
checkOneExp :: UncheckedExp -> TypeM ([TypeParam], Exp)
checkOneExp UncheckedExp
e = ((([TypeParam], Exp), [Occurence]) -> ([TypeParam], Exp))
-> TypeM (([TypeParam], Exp), [Occurence])
-> TypeM ([TypeParam], Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (([TypeParam], Exp), [Occurence]) -> ([TypeParam], Exp)
forall a b. (a, b) -> a
fst (TypeM (([TypeParam], Exp), [Occurence])
 -> TypeM ([TypeParam], Exp))
-> (TermTypeM ([TypeParam], Exp)
    -> TypeM (([TypeParam], Exp), [Occurence]))
-> TermTypeM ([TypeParam], Exp)
-> TypeM ([TypeParam], Exp)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM ([TypeParam], Exp)
-> TypeM (([TypeParam], Exp), [Occurence])
forall a. TermTypeM a -> TypeM (a, [Occurence])
runTermTypeM (TermTypeM ([TypeParam], Exp) -> TypeM ([TypeParam], Exp))
-> TermTypeM ([TypeParam], Exp) -> TypeM ([TypeParam], Exp)
forall a b. (a -> b) -> a -> b
$ do
  Exp
e' <- UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
e
  let t :: StructType
t = PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType) -> PatType -> StructType
forall a b. (a -> b) -> a -> b
$ Exp -> PatType
typeOf Exp
e'
  ([TypeParam]
tparams, [Pat]
_, StructType
_, [VName]
_) <-
    Name
-> SrcLoc
-> [TypeParam]
-> [Pat]
-> StructType
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
letGeneralise (String -> Name
nameFromString String
"<exp>") (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
e) [] [] StructType
t
  Names -> TermTypeM ()
fixOverloadedTypes (Names -> TermTypeM ()) -> Names -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ StructType -> Names
forall as dim. Monoid as => TypeBase dim as -> Names
typeVars StructType
t
  Exp
e'' <- Exp -> TermTypeM Exp
forall e. ASTMappable e => e -> TermTypeM e
updateTypes Exp
e'
  Exp -> TermTypeM ()
checkUnmatched Exp
e''
  Exp -> TermTypeM ()
causalityCheck Exp
e''
  Exp -> TermTypeM ()
literalOverflowCheck Exp
e''
  ([TypeParam], Exp) -> TermTypeM ([TypeParam], Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return ([TypeParam]
tparams, Exp
e'')

-- Verify that all sum type constructors and empty array literals have
-- a size that is known (rigid or a type parameter).  This is to
-- ensure that we can actually determine their shape at run-time.
causalityCheck :: Exp -> TermTypeM ()
causalityCheck :: Exp -> TermTypeM ()
causalityCheck Exp
binding_body = do
  Constraints
constraints <- TermTypeM Constraints
forall (m :: * -> *). MonadUnify m => m Constraints
getConstraints

  let checkCausality :: Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality Doc
what Names
known TypeBase (DimDecl VName) as
t SrcLoc
loc
        | (VName
d, SrcLoc
dloc) : [(VName, SrcLoc)]
_ <-
            (VName -> Maybe (VName, SrcLoc)) -> [VName] -> [(VName, SrcLoc)]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe (Constraints -> Names -> VName -> Maybe (VName, SrcLoc)
forall a a.
Ord a =>
Map a (a, Constraint) -> Set a -> a -> Maybe (a, SrcLoc)
unknown Constraints
constraints Names
known) ([VName] -> [(VName, SrcLoc)]) -> [VName] -> [(VName, SrcLoc)]
forall a b. (a -> b) -> a -> b
$
              Names -> [VName]
forall a. Set a -> [a]
S.toList (Names -> [VName]) -> Names -> [VName]
forall a b. (a -> b) -> a -> b
$ StructType -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNames (StructType -> Names) -> StructType -> Names
forall a b. (a -> b) -> a -> b
$ TypeBase (DimDecl VName) as -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct TypeBase (DimDecl VName) as
t =
          t (Either TypeError) a -> Maybe (t (Either TypeError) a)
forall a. a -> Maybe a
Just (t (Either TypeError) a -> Maybe (t (Either TypeError) a))
-> t (Either TypeError) a -> Maybe (t (Either TypeError) a)
forall a b. (a -> b) -> a -> b
$ Either TypeError a -> t (Either TypeError) a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (Either TypeError a -> t (Either TypeError) a)
-> Either TypeError a -> t (Either TypeError) a
forall a b. (a -> b) -> a -> b
$ Doc
-> SrcLoc
-> VName
-> SrcLoc
-> TypeBase (DimDecl VName) as
-> Either TypeError a
forall v a b b.
(IsName v, Pretty a, Located b) =>
Doc -> SrcLoc -> v -> b -> a -> Either TypeError b
causality Doc
what SrcLoc
loc VName
d SrcLoc
dloc TypeBase (DimDecl VName) as
t
        | Bool
otherwise = Maybe (t (Either TypeError) a)
forall a. Maybe a
Nothing

      checkParamCausality :: Names -> Pat -> Maybe (t (Either TypeError) a)
checkParamCausality Names
known Pat
p =
        Doc -> Names -> PatType -> SrcLoc -> Maybe (t (Either TypeError) a)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality (Pat -> Doc
forall a. Pretty a => a -> Doc
ppr Pat
p) Names
known (Pat -> PatType
patternType Pat
p) (Pat -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Pat
p)

      onExp ::
        S.Set VName ->
        Exp ->
        StateT (S.Set VName) (Either TypeError) Exp

      onExp :: Names -> Exp -> StateT Names (Either TypeError) Exp
onExp Names
known (Var QualName VName
v (Info PatType
t) SrcLoc
loc)
        | Just StateT Names (Either TypeError) Exp
bad <- Doc
-> Names
-> PatType
-> SrcLoc
-> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality (Doc -> Doc
pquote (QualName VName -> Doc
forall a. Pretty a => a -> Doc
ppr QualName VName
v)) Names
known PatType
t SrcLoc
loc =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known (ProjectSection [Name]
_ (Info PatType
t) SrcLoc
loc)
        | Just StateT Names (Either TypeError) Exp
bad <- Doc
-> Names
-> PatType
-> SrcLoc
-> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality Doc
"projection section" Names
known PatType
t SrcLoc
loc =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known (IndexSection Slice
_ (Info PatType
t) SrcLoc
loc)
        | Just StateT Names (Either TypeError) Exp
bad <- Doc
-> Names
-> PatType
-> SrcLoc
-> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality Doc
"projection section" Names
known PatType
t SrcLoc
loc =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known (OpSectionRight QualName VName
_ (Info PatType
t) Exp
_ (Info (PName, StructType), Info (PName, StructType, Maybe VName))
_ Info PatType
_ SrcLoc
loc)
        | Just StateT Names (Either TypeError) Exp
bad <- Doc
-> Names
-> PatType
-> SrcLoc
-> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality Doc
"operator section" Names
known PatType
t SrcLoc
loc =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known (OpSectionLeft QualName VName
_ (Info PatType
t) Exp
_ (Info (PName, StructType, Maybe VName), Info (PName, StructType))
_ (Info PatType, Info [VName])
_ SrcLoc
loc)
        | Just StateT Names (Either TypeError) Exp
bad <- Doc
-> Names
-> PatType
-> SrcLoc
-> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality Doc
"operator section" Names
known PatType
t SrcLoc
loc =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known (ArrayLit [] (Info PatType
t) SrcLoc
loc)
        | Just StateT Names (Either TypeError) Exp
bad <- Doc
-> Names
-> PatType
-> SrcLoc
-> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) as a.
MonadTrans t =>
Doc
-> Names
-> TypeBase (DimDecl VName) as
-> SrcLoc
-> Maybe (t (Either TypeError) a)
checkCausality Doc
"empty array" Names
known PatType
t SrcLoc
loc =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known (Lambda [Pat]
params Exp
_ Maybe (TypeExp VName)
_ Info (Aliasing, StructType)
_ SrcLoc
_)
        | StateT Names (Either TypeError) Exp
bad : [StateT Names (Either TypeError) Exp]
_ <- (Pat -> Maybe (StateT Names (Either TypeError) Exp))
-> [Pat] -> [StateT Names (Either TypeError) Exp]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe (Names -> Pat -> Maybe (StateT Names (Either TypeError) Exp)
forall (t :: (* -> *) -> * -> *) a.
MonadTrans t =>
Names -> Pat -> Maybe (t (Either TypeError) a)
checkParamCausality Names
known) [Pat]
params =
          StateT Names (Either TypeError) Exp
bad
      onExp Names
known e :: Exp
e@(AppExp (LetPat [SizeBinder VName]
_ Pat
_ Exp
bindee_e Exp
body_e SrcLoc
_) (Info AppRes
res)) = do
        Names
-> Exp -> Exp -> [VName] -> StateT Names (Either TypeError) ()
sequencePoint Names
known Exp
bindee_e Exp
body_e ([VName] -> StateT Names (Either TypeError) ())
-> [VName] -> StateT Names (Either TypeError) ()
forall a b. (a -> b) -> a -> b
$ AppRes -> [VName]
appResExt AppRes
res
        Exp -> StateT Names (Either TypeError) Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e
      onExp Names
known e :: Exp
e@(AppExp (Apply Exp
f Exp
arg (Info (Diet
_, Maybe VName
p)) SrcLoc
_) (Info AppRes
res)) = do
        Names
-> Exp -> Exp -> [VName] -> StateT Names (Either TypeError) ()
sequencePoint Names
known Exp
arg Exp
f ([VName] -> StateT Names (Either TypeError) ())
-> [VName] -> StateT Names (Either TypeError) ()
forall a b. (a -> b) -> a -> b
$ Maybe VName -> [VName]
forall a. Maybe a -> [a]
maybeToList Maybe VName
p [VName] -> [VName] -> [VName]
forall a. [a] -> [a] -> [a]
++ AppRes -> [VName]
appResExt AppRes
res
        Exp -> StateT Names (Either TypeError) Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e
      onExp
        Names
known
        e :: Exp
e@(AppExp (BinOp (QualName VName
f, SrcLoc
floc) Info PatType
ft (Exp
x, Info (StructType
_, Maybe VName
xp)) (Exp
y, Info (StructType
_, Maybe VName
yp)) SrcLoc
_) (Info AppRes
res)) = do
          Names
args_known <-
            Either TypeError Names -> StateT Names (Either TypeError) Names
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (Either TypeError Names -> StateT Names (Either TypeError) Names)
-> Either TypeError Names -> StateT Names (Either TypeError) Names
forall a b. (a -> b) -> a -> b
$
              StateT Names (Either TypeError) ()
-> Names -> Either TypeError Names
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m s
execStateT (Names
-> Exp -> Exp -> [VName] -> StateT Names (Either TypeError) ()
sequencePoint Names
known Exp
x Exp
y ([VName] -> StateT Names (Either TypeError) ())
-> [VName] -> StateT Names (Either TypeError) ()
forall a b. (a -> b) -> a -> b
$ [Maybe VName] -> [VName]
forall a. [Maybe a] -> [a]
catMaybes [Maybe VName
xp, Maybe VName
yp]) Names
forall a. Monoid a => a
mempty
          StateT Names (Either TypeError) Exp
-> StateT Names (Either TypeError) ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (StateT Names (Either TypeError) Exp
 -> StateT Names (Either TypeError) ())
-> StateT Names (Either TypeError) Exp
-> StateT Names (Either TypeError) ()
forall a b. (a -> b) -> a -> b
$ Names -> Exp -> StateT Names (Either TypeError) Exp
onExp (Names
args_known Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> Names
known) (QualName VName -> Info PatType -> SrcLoc -> Exp
forall (f :: * -> *) vn.
QualName vn -> f PatType -> SrcLoc -> ExpBase f vn
Var QualName VName
f Info PatType
ft SrcLoc
floc)
          (Names -> Names) -> StateT Names (Either TypeError) ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Names
args_known Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList (AppRes -> [VName]
appResExt AppRes
res)) Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<>)
          Exp -> StateT Names (Either TypeError) Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
e
      onExp Names
known e :: Exp
e@(AppExp AppExpBase Info VName
e' (Info AppRes
res)) = do
        Names
-> AppExpBase Info VName -> StateT Names (Either TypeError) ()
forall a.
ASTMappable a =>
Names -> a -> StateT Names (Either TypeError) ()
recurse Names
known AppExpBase Info VName
e'
        (Names -> Names) -> StateT Names (Either TypeError) ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify (Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList (AppRes -> [VName]
appResExt AppRes
res))
        Exp -> StateT Names (Either TypeError) Exp
forall (f :: * -> *) a. Applicative f => a -> f a
pure Exp
e
      onExp Names
known Exp
e = do
        Names -> Exp -> StateT Names (Either TypeError) ()
forall a.
ASTMappable a =>
Names -> a -> StateT Names (Either TypeError) ()
recurse Names
known Exp
e
        Exp -> StateT Names (Either TypeError) Exp
forall (f :: * -> *) a. Applicative f => a -> f a
pure Exp
e

      recurse :: Names -> a -> StateT Names (Either TypeError) ()
recurse Names
known = StateT Names (Either TypeError) a
-> StateT Names (Either TypeError) ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (StateT Names (Either TypeError) a
 -> StateT Names (Either TypeError) ())
-> (a -> StateT Names (Either TypeError) a)
-> a
-> StateT Names (Either TypeError) ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ASTMapper (StateT Names (Either TypeError))
-> a -> StateT Names (Either TypeError) a
forall x (m :: * -> *).
(ASTMappable x, Monad m) =>
ASTMapper m -> x -> m x
astMap ASTMapper (StateT Names (Either TypeError))
mapper
        where
          mapper :: ASTMapper (StateT Names (Either TypeError))
mapper = ASTMapper (StateT Names (Either TypeError))
forall (m :: * -> *). Monad m => ASTMapper m
identityMapper {mapOnExp :: Exp -> StateT Names (Either TypeError) Exp
mapOnExp = Names -> Exp -> StateT Names (Either TypeError) Exp
onExp Names
known}

      sequencePoint :: Names
-> Exp -> Exp -> [VName] -> StateT Names (Either TypeError) ()
sequencePoint Names
known Exp
x Exp
y [VName]
ext = do
        Names
new_known <- Either TypeError Names -> StateT Names (Either TypeError) Names
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (Either TypeError Names -> StateT Names (Either TypeError) Names)
-> Either TypeError Names -> StateT Names (Either TypeError) Names
forall a b. (a -> b) -> a -> b
$ StateT Names (Either TypeError) Exp
-> Names -> Either TypeError Names
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m s
execStateT (Names -> Exp -> StateT Names (Either TypeError) Exp
onExp Names
known Exp
x) Names
forall a. Monoid a => a
mempty
        StateT Names (Either TypeError) Exp
-> StateT Names (Either TypeError) ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (StateT Names (Either TypeError) Exp
 -> StateT Names (Either TypeError) ())
-> StateT Names (Either TypeError) Exp
-> StateT Names (Either TypeError) ()
forall a b. (a -> b) -> a -> b
$ Names -> Exp -> StateT Names (Either TypeError) Exp
onExp (Names
new_known Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> Names
known) Exp
y
        (Names -> Names) -> StateT Names (Either TypeError) ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Names
new_known Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList [VName]
ext) Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<>)

  (TypeError -> TermTypeM ())
-> (Exp -> TermTypeM ()) -> Either TypeError Exp -> TermTypeM ()
forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either TypeError -> TermTypeM ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (TermTypeM () -> Exp -> TermTypeM ()
forall a b. a -> b -> a
const (TermTypeM () -> Exp -> TermTypeM ())
-> TermTypeM () -> Exp -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()) (Either TypeError Exp -> TermTypeM ())
-> Either TypeError Exp -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    StateT Names (Either TypeError) Exp
-> Names -> Either TypeError Exp
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m a
evalStateT (Names -> Exp -> StateT Names (Either TypeError) Exp
onExp Names
forall a. Monoid a => a
mempty Exp
binding_body) Names
forall a. Monoid a => a
mempty
  where
    unknown :: Map a (a, Constraint) -> Set a -> a -> Maybe (a, SrcLoc)
unknown Map a (a, Constraint)
constraints Set a
known a
v = do
      Bool -> Maybe ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard (Bool -> Maybe ()) -> Bool -> Maybe ()
forall a b. (a -> b) -> a -> b
$ a
v a -> Set a -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.notMember` Set a
known
      SrcLoc
loc <- Map a (a, Constraint) -> a -> Maybe SrcLoc
forall k a. Ord k => Map k (a, Constraint) -> k -> Maybe SrcLoc
unknowable Map a (a, Constraint)
constraints a
v
      (a, SrcLoc) -> Maybe (a, SrcLoc)
forall (m :: * -> *) a. Monad m => a -> m a
return (a
v, SrcLoc
loc)

    unknowable :: Map k (a, Constraint) -> k -> Maybe SrcLoc
unknowable Map k (a, Constraint)
constraints k
v =
      case (a, Constraint) -> Constraint
forall a b. (a, b) -> b
snd ((a, Constraint) -> Constraint)
-> Maybe (a, Constraint) -> Maybe Constraint
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> k -> Map k (a, Constraint) -> Maybe (a, Constraint)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup k
v Map k (a, Constraint)
constraints of
        Just (UnknowableSize SrcLoc
loc RigidSource
_) -> SrcLoc -> Maybe SrcLoc
forall a. a -> Maybe a
Just SrcLoc
loc
        Maybe Constraint
_ -> Maybe SrcLoc
forall a. Maybe a
Nothing

    causality :: Doc -> SrcLoc -> v -> b -> a -> Either TypeError b
causality Doc
what SrcLoc
loc v
d b
dloc a
t =
      TypeError -> Either TypeError b
forall a b. a -> Either a b
Left (TypeError -> Either TypeError b)
-> TypeError -> Either TypeError b
forall a b. (a -> b) -> a -> b
$
        SrcLoc -> Notes -> Doc -> TypeError
TypeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TypeError) -> (Doc -> Doc) -> Doc -> TypeError
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Doc -> Doc -> Doc
withIndexLink Doc
"causality-check" (Doc -> TypeError) -> Doc -> TypeError
forall a b. (a -> b) -> a -> b
$
          Doc
"Causality check: size" Doc -> Doc -> Doc
<+/> Doc -> Doc
pquote (v -> Doc
forall v. IsName v => v -> Doc
pprName v
d)
            Doc -> Doc -> Doc
<+/> Doc
"needed for type of"
            Doc -> Doc -> Doc
<+> Doc
what Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
colon
            Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (a -> Doc
forall a. Pretty a => a -> Doc
ppr a
t)
            Doc -> Doc -> Doc
</> Doc
"But"
            Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (v -> Doc
forall v. IsName v => v -> Doc
pprName v
d)
            Doc -> Doc -> Doc
<+> Doc
"is computed at"
            Doc -> Doc -> Doc
<+/> String -> Doc
text (SrcLoc -> b -> String
forall a b. (Located a, Located b) => a -> b -> String
locStrRel SrcLoc
loc b
dloc) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
            Doc -> Doc -> Doc
</> Doc
""
            Doc -> Doc -> Doc
</> Doc
"Hint:"
            Doc -> Doc -> Doc
<+> Doc -> Doc
align
              ( String -> Doc
textwrap String
"Bind the expression producing" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (v -> Doc
forall v. IsName v => v -> Doc
pprName v
d)
                  Doc -> Doc -> Doc
<+> Doc
"with 'let' beforehand."
              )

-- | Traverse the expression, emitting warnings if any of the literals overflow
-- their inferred types
--
-- Note: currently unable to detect float underflow (such as 1e-400 -> 0)
literalOverflowCheck :: Exp -> TermTypeM ()
literalOverflowCheck :: Exp -> TermTypeM ()
literalOverflowCheck = TermTypeM Exp -> TermTypeM ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (TermTypeM Exp -> TermTypeM ())
-> (Exp -> TermTypeM Exp) -> Exp -> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp -> TermTypeM Exp
forall (f :: * -> *). MonadTypeChecker f => Exp -> f Exp
check
  where
    check :: Exp -> f Exp
check e :: Exp
e@(IntLit Integer
x Info PatType
ty SrcLoc
loc) =
      Exp
e Exp -> f () -> f Exp
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ case Info PatType
ty of
        Info (Scalar (Prim PrimType
t)) -> Bool -> Integer -> PrimType -> SrcLoc -> f ()
forall (f :: * -> *) loc a a.
(MonadTypeChecker f, Located loc, Pretty a, Pretty a) =>
Bool -> a -> a -> loc -> f ()
warnBounds (Integer -> PrimType -> Bool
forall a. Integral a => a -> PrimType -> Bool
inBoundsI Integer
x PrimType
t) Integer
x PrimType
t SrcLoc
loc
        Info PatType
_ -> String -> f ()
forall a. HasCallStack => String -> a
error String
"Inferred type of int literal is not a number"
    check e :: Exp
e@(FloatLit Double
x Info PatType
ty SrcLoc
loc) =
      Exp
e Exp -> f () -> f Exp
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ case Info PatType
ty of
        Info (Scalar (Prim (FloatType FloatType
t))) -> Bool -> Double -> FloatType -> SrcLoc -> f ()
forall (f :: * -> *) loc a a.
(MonadTypeChecker f, Located loc, Pretty a, Pretty a) =>
Bool -> a -> a -> loc -> f ()
warnBounds (Double -> FloatType -> Bool
forall a. RealFloat a => a -> FloatType -> Bool
inBoundsF Double
x FloatType
t) Double
x FloatType
t SrcLoc
loc
        Info PatType
_ -> String -> f ()
forall a. HasCallStack => String -> a
error String
"Inferred type of float literal is not a float"
    check e :: Exp
e@(Negate (IntLit Integer
x Info PatType
ty SrcLoc
loc1) SrcLoc
loc2) =
      Exp
e Exp -> f () -> f Exp
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ case Info PatType
ty of
        Info (Scalar (Prim PrimType
t)) -> Bool -> Integer -> PrimType -> SrcLoc -> f ()
forall (f :: * -> *) loc a a.
(MonadTypeChecker f, Located loc, Pretty a, Pretty a) =>
Bool -> a -> a -> loc -> f ()
warnBounds (Integer -> PrimType -> Bool
forall a. Integral a => a -> PrimType -> Bool
inBoundsI (- Integer
x) PrimType
t) (- Integer
x) PrimType
t (SrcLoc
loc1 SrcLoc -> SrcLoc -> SrcLoc
forall a. Semigroup a => a -> a -> a
<> SrcLoc
loc2)
        Info PatType
_ -> String -> f ()
forall a. HasCallStack => String -> a
error String
"Inferred type of int literal is not a number"
    check Exp
e = ASTMapper f -> Exp -> f Exp
forall x (m :: * -> *).
(ASTMappable x, Monad m) =>
ASTMapper m -> x -> m x
astMap ASTMapper f
forall (m :: * -> *). Monad m => ASTMapper m
identityMapper {mapOnExp :: Exp -> f Exp
mapOnExp = Exp -> f Exp
check} Exp
e
    bitWidth :: IntType -> Int
bitWidth IntType
ty = Int
8 Int -> Int -> Int
forall a. Num a => a -> a -> a
* IntType -> Int
forall a. Num a => IntType -> a
intByteSize IntType
ty :: Int
    inBoundsI :: a -> PrimType -> Bool
inBoundsI a
x (Signed IntType
t) = a
x a -> a -> Bool
forall a. Ord a => a -> a -> Bool
>= -a
2 a -> Int -> a
forall a b. (Num a, Integral b) => a -> b -> a
^ (IntType -> Int
bitWidth IntType
t Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) Bool -> Bool -> Bool
&& a
x a -> a -> Bool
forall a. Ord a => a -> a -> Bool
< a
2 a -> Int -> a
forall a b. (Num a, Integral b) => a -> b -> a
^ (IntType -> Int
bitWidth IntType
t Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)
    inBoundsI a
x (Unsigned IntType
t) = a
x a -> a -> Bool
forall a. Ord a => a -> a -> Bool
>= a
0 Bool -> Bool -> Bool
&& a
x a -> a -> Bool
forall a. Ord a => a -> a -> Bool
< a
2 a -> Int -> a
forall a b. (Num a, Integral b) => a -> b -> a
^ IntType -> Int
bitWidth IntType
t
    inBoundsI a
x (FloatType FloatType
Float16) = Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Half -> Bool
forall a. RealFloat a => a -> Bool
isInfinite (a -> Half
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
x :: Half)
    inBoundsI a
x (FloatType FloatType
Float32) = Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Float -> Bool
forall a. RealFloat a => a -> Bool
isInfinite (a -> Float
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
x :: Float)
    inBoundsI a
x (FloatType FloatType
Float64) = Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Double -> Bool
forall a. RealFloat a => a -> Bool
isInfinite (a -> Double
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
x :: Double)
    inBoundsI a
_ PrimType
Bool = String -> Bool
forall a. HasCallStack => String -> a
error String
"Inferred type of int literal is not a number"
    inBoundsF :: a -> FloatType -> Bool
inBoundsF a
x FloatType
Float16 = Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Float -> Bool
forall a. RealFloat a => a -> Bool
isInfinite (a -> Float
forall a b. (Real a, Fractional b) => a -> b
realToFrac a
x :: Float)
    inBoundsF a
x FloatType
Float32 = Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Float -> Bool
forall a. RealFloat a => a -> Bool
isInfinite (a -> Float
forall a b. (Real a, Fractional b) => a -> b
realToFrac a
x :: Float)
    inBoundsF a
x FloatType
Float64 = Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ a -> Bool
forall a. RealFloat a => a -> Bool
isInfinite a
x
    warnBounds :: Bool -> a -> a -> loc -> f ()
warnBounds Bool
inBounds a
x a
ty loc
loc =
      Bool -> f () -> f ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless Bool
inBounds (f () -> f ()) -> f () -> f ()
forall a b. (a -> b) -> a -> b
$
        loc -> Notes -> Doc -> f ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError loc
loc Notes
forall a. Monoid a => a
mempty (Doc -> f ()) -> Doc -> f ()
forall a b. (a -> b) -> a -> b
$
          Doc
"Literal " Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> a -> Doc
forall a. Pretty a => a -> Doc
ppr a
x
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
" out of bounds for inferred type "
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> a -> Doc
forall a. Pretty a => a -> Doc
ppr a
ty
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."

-- | Type-check a top-level (or module-level) function definition.
-- Despite the name, this is also used for checking constant
-- definitions, by treating them as 0-ary functions.
checkFunDef ::
  ( Name,
    Maybe UncheckedTypeExp,
    [UncheckedTypeParam],
    [UncheckedPat],
    UncheckedExp,
    SrcLoc
  ) ->
  TypeM
    ( VName,
      [TypeParam],
      [Pat],
      Maybe (TypeExp VName),
      StructType,
      [VName],
      Exp
    )
checkFunDef :: (Name, Maybe (TypeExp Name), [UncheckedTypeParam], [UncheckedPat],
 UncheckedExp, SrcLoc)
-> TypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
checkFunDef (Name
fname, Maybe (TypeExp Name)
maybe_retdecl, [UncheckedTypeParam]
tparams, [UncheckedPat]
params, UncheckedExp
body, SrcLoc
loc) =
  (((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
   [VName], Exp),
  [Occurence])
 -> (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
     [VName], Exp))
-> TypeM
     ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
       [VName], Exp),
      [Occurence])
-> TypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
  [VName], Exp),
 [Occurence])
-> (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
    [VName], Exp)
forall a b. (a, b) -> a
fst (TypeM
   ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
     [VName], Exp),
    [Occurence])
 -> TypeM
      (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
       [VName], Exp))
-> TypeM
     ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
       [VName], Exp),
      [Occurence])
-> TypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
forall a b. (a -> b) -> a -> b
$
    TermTypeM
  (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
   [VName], Exp)
-> TypeM
     ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
       [VName], Exp),
      [Occurence])
forall a. TermTypeM a -> TypeM (a, [Occurence])
runTermTypeM (TermTypeM
   (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
    [VName], Exp)
 -> TypeM
      ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
        [VName], Exp),
       [Occurence]))
-> TermTypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
-> TypeM
     ((VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
       [VName], Exp),
      [Occurence])
forall a b. (a -> b) -> a -> b
$ do
      ([TypeParam]
tparams', [Pat]
params', Maybe (TypeExp VName)
maybe_retdecl', StructType
rettype', [VName]
retext, Exp
body') <-
        (Name, Maybe (TypeExp Name), [UncheckedTypeParam], [UncheckedPat],
 UncheckedExp, SrcLoc)
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
checkBinding (Name
fname, Maybe (TypeExp Name)
maybe_retdecl, [UncheckedTypeParam]
tparams, [UncheckedPat]
params, UncheckedExp
body, SrcLoc
loc)

      -- Since this is a top-level function, we also resolve overloaded
      -- types, using either defaults or complaining about ambiguities.
      Names -> TermTypeM ()
fixOverloadedTypes (Names -> TermTypeM ()) -> Names -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        StructType -> Names
forall as dim. Monoid as => TypeBase dim as -> Names
typeVars StructType
rettype' Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> (Pat -> Names) -> [Pat] -> Names
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap (PatType -> Names
forall as dim. Monoid as => TypeBase dim as -> Names
typeVars (PatType -> Names) -> (Pat -> PatType) -> Pat -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Pat -> PatType
patternType) [Pat]
params'

      -- Then replace all inferred types in the body and parameters.
      Exp
body'' <- Exp -> TermTypeM Exp
forall e. ASTMappable e => e -> TermTypeM e
updateTypes Exp
body'
      [Pat]
params'' <- [Pat] -> TermTypeM [Pat]
forall e. ASTMappable e => e -> TermTypeM e
updateTypes [Pat]
params'
      Maybe (TypeExp VName)
maybe_retdecl'' <- (TypeExp VName -> TermTypeM (TypeExp VName))
-> Maybe (TypeExp VName) -> TermTypeM (Maybe (TypeExp VName))
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
traverse TypeExp VName -> TermTypeM (TypeExp VName)
forall e. ASTMappable e => e -> TermTypeM e
updateTypes Maybe (TypeExp VName)
maybe_retdecl'
      StructType
rettype'' <- StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully StructType
rettype'

      -- Check if pattern matches are exhaustive and yield
      -- errors if not.
      Exp -> TermTypeM ()
checkUnmatched Exp
body''

      -- Check if the function body can actually be evaluated.
      Exp -> TermTypeM ()
causalityCheck Exp
body''

      Exp -> TermTypeM ()
literalOverflowCheck Exp
body''

      [(Namespace, Name)]
-> TermTypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
-> TermTypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
forall (m :: * -> *) a.
MonadTypeChecker m =>
[(Namespace, Name)] -> m a -> m a
bindSpaced [(Namespace
Term, Name
fname)] (TermTypeM
   (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
    [VName], Exp)
 -> TermTypeM
      (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
       [VName], Exp))
-> TermTypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
-> TermTypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
forall a b. (a -> b) -> a -> b
$ do
        VName
fname' <- Namespace -> Name -> SrcLoc -> TermTypeM VName
forall (m :: * -> *).
MonadTypeChecker m =>
Namespace -> Name -> SrcLoc -> m VName
checkName Namespace
Term Name
fname SrcLoc
loc
        Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Name -> String
nameToString Name
fname String -> [String] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [String]
doNotShadow) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
          SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
            Doc
"The" Doc -> Doc -> Doc
<+> Name -> Doc
forall v. IsName v => v -> Doc
pprName Name
fname Doc -> Doc -> Doc
<+> Doc
"operator may not be redefined."

        (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
 [VName], Exp)
-> TermTypeM
     (VName, [TypeParam], [Pat], Maybe (TypeExp VName), StructType,
      [VName], Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return (VName
fname', [TypeParam]
tparams', [Pat]
params'', Maybe (TypeExp VName)
maybe_retdecl'', StructType
rettype'', [VName]
retext, Exp
body'')

-- | This is "fixing" as in "setting them", not "correcting them".  We
-- only make very conservative fixing.
fixOverloadedTypes :: Names -> TermTypeM ()
fixOverloadedTypes :: Names -> TermTypeM ()
fixOverloadedTypes Names
tyvars_at_toplevel =
  TermTypeM Constraints
forall (m :: * -> *). MonadUnify m => m Constraints
getConstraints TermTypeM Constraints
-> (Constraints -> TermTypeM ()) -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ((VName, Constraint) -> TermTypeM ())
-> [(VName, Constraint)] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ (VName, Constraint) -> TermTypeM ()
forall (m :: * -> *).
(MonadUnify m, MonadTypeChecker m) =>
(VName, Constraint) -> m ()
fixOverloaded ([(VName, Constraint)] -> TermTypeM ())
-> (Constraints -> [(VName, Constraint)])
-> Constraints
-> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map VName Constraint -> [(VName, Constraint)]
forall k a. Map k a -> [(k, a)]
M.toList (Map VName Constraint -> [(VName, Constraint)])
-> (Constraints -> Map VName Constraint)
-> Constraints
-> [(VName, Constraint)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((Int, Constraint) -> Constraint)
-> Constraints -> Map VName Constraint
forall a b k. (a -> b) -> Map k a -> Map k b
M.map (Int, Constraint) -> Constraint
forall a b. (a, b) -> b
snd
  where
    fixOverloaded :: (VName, Constraint) -> m ()
fixOverloaded (VName
v, Overloaded [PrimType]
ots Usage
usage)
      | IntType -> PrimType
Signed IntType
Int32 PrimType -> [PrimType] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [PrimType]
ots = do
        Usage -> StructType -> StructType -> m ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify Usage
usage (ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (()
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) ()
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar () Uniqueness
Nonunique (VName -> TypeName
typeName VName
v) [])) (StructType -> m ()) -> StructType -> m ()
forall a b. (a -> b) -> a -> b
$
          ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) ())
-> PrimType -> ScalarTypeBase (DimDecl VName) ()
forall a b. (a -> b) -> a -> b
$ IntType -> PrimType
Signed IntType
Int32
        Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (VName
v VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
tyvars_at_toplevel) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$
          Usage -> Doc -> m ()
forall (m :: * -> *) loc.
(MonadTypeChecker m, Located loc) =>
loc -> Doc -> m ()
warn Usage
usage Doc
"Defaulting ambiguous type to i32."
      | FloatType -> PrimType
FloatType FloatType
Float64 PrimType -> [PrimType] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [PrimType]
ots = do
        Usage -> StructType -> StructType -> m ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
unify Usage
usage (ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (()
-> Uniqueness
-> TypeName
-> [TypeArg (DimDecl VName)]
-> ScalarTypeBase (DimDecl VName) ()
forall dim as.
as
-> Uniqueness -> TypeName -> [TypeArg dim] -> ScalarTypeBase dim as
TypeVar () Uniqueness
Nonunique (VName -> TypeName
typeName VName
v) [])) (StructType -> m ()) -> StructType -> m ()
forall a b. (a -> b) -> a -> b
$
          ScalarTypeBase (DimDecl VName) () -> StructType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) () -> StructType)
-> ScalarTypeBase (DimDecl VName) () -> StructType
forall a b. (a -> b) -> a -> b
$ PrimType -> ScalarTypeBase (DimDecl VName) ()
forall dim as. PrimType -> ScalarTypeBase dim as
Prim (PrimType -> ScalarTypeBase (DimDecl VName) ())
-> PrimType -> ScalarTypeBase (DimDecl VName) ()
forall a b. (a -> b) -> a -> b
$ FloatType -> PrimType
FloatType FloatType
Float64
        Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (VName
v VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
tyvars_at_toplevel) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$
          Usage -> Doc -> m ()
forall (m :: * -> *) loc.
(MonadTypeChecker m, Located loc) =>
loc -> Doc -> m ()
warn Usage
usage Doc
"Defaulting ambiguous type to f64."
      | Bool
otherwise =
        Usage -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Usage
usage Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
          Doc
"Type is ambiguous (could be one of" Doc -> Doc -> Doc
<+> [Doc] -> Doc
commasep ((PrimType -> Doc) -> [PrimType] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map PrimType -> Doc
forall a. Pretty a => a -> Doc
ppr [PrimType]
ots) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
")."
            Doc -> Doc -> Doc
</> Doc
"Add a type annotation to disambiguate the type."
    fixOverloaded (VName
_, NoConstraint Liftedness
_ Usage
usage) =
      Usage -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Usage
usage Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
        Doc
"Type of expression is ambiguous."
          Doc -> Doc -> Doc
</> Doc
"Add a type annotation to disambiguate the type."
    fixOverloaded (VName
_, Equality Usage
usage) =
      Usage -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Usage
usage Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
        Doc
"Type is ambiguous (must be equality type)."
          Doc -> Doc -> Doc
</> Doc
"Add a type annotation to disambiguate the type."
    fixOverloaded (VName
_, HasFields Map Name StructType
fs Usage
usage) =
      Usage -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Usage
usage Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
        Doc
"Type is ambiguous.  Must be record with fields:"
          Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 ([Doc] -> Doc
stack ([Doc] -> Doc) -> [Doc] -> Doc
forall a b. (a -> b) -> a -> b
$ ((Name, StructType) -> Doc) -> [(Name, StructType)] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map (Name, StructType) -> Doc
forall a a. (Pretty a, Pretty a) => (a, a) -> Doc
field ([(Name, StructType)] -> [Doc]) -> [(Name, StructType)] -> [Doc]
forall a b. (a -> b) -> a -> b
$ Map Name StructType -> [(Name, StructType)]
forall k a. Map k a -> [(k, a)]
M.toList Map Name StructType
fs)
          Doc -> Doc -> Doc
</> Doc
"Add a type annotation to disambiguate the type."
      where
        field :: (a, a) -> Doc
field (a
l, a
t) = a -> Doc
forall a. Pretty a => a -> Doc
ppr a
l Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
colon Doc -> Doc -> Doc
<+> Doc -> Doc
align (a -> Doc
forall a. Pretty a => a -> Doc
ppr a
t)
    fixOverloaded (VName
_, HasConstrs Map Name [StructType]
cs Usage
usage) =
      Usage -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Usage
usage Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
        Doc
"Type is ambiguous (must be a sum type with constructors:"
          Doc -> Doc -> Doc
<+> ScalarTypeBase (DimDecl VName) () -> Doc
forall a. Pretty a => a -> Doc
ppr (Map Name [StructType] -> ScalarTypeBase (DimDecl VName) ()
forall dim as. Map Name [TypeBase dim as] -> ScalarTypeBase dim as
Sum Map Name [StructType]
cs) Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
")."
          Doc -> Doc -> Doc
</> Doc
"Add a type annotation to disambiguate the type."
    fixOverloaded (VName
v, Size Maybe (DimDecl VName)
Nothing Usage
usage) =
      Usage -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Usage
usage Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$ Doc
"Size" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
v) Doc -> Doc -> Doc
<+> Doc
"is ambiguous.\n"
    fixOverloaded (VName, Constraint)
_ = () -> m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

hiddenParamNames :: [Pat] -> Names
hiddenParamNames :: [Pat] -> Names
hiddenParamNames [Pat]
params = Names
hidden
  where
    param_all_names :: Names
param_all_names = [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ([Names] -> Names) -> [Names] -> Names
forall a b. (a -> b) -> a -> b
$ (Pat -> Names) -> [Pat] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames [Pat]
params
    named :: (PName, b) -> Maybe VName
named (Named VName
x, b
_) = VName -> Maybe VName
forall a. a -> Maybe a
Just VName
x
    named (PName
Unnamed, b
_) = Maybe VName
forall a. Maybe a
Nothing
    param_names :: Names
param_names =
      [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList ([VName] -> Names) -> [VName] -> Names
forall a b. (a -> b) -> a -> b
$ (Pat -> Maybe VName) -> [Pat] -> [VName]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe ((PName, StructType) -> Maybe VName
forall b. (PName, b) -> Maybe VName
named ((PName, StructType) -> Maybe VName)
-> (Pat -> (PName, StructType)) -> Pat -> Maybe VName
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Pat -> (PName, StructType)
patternParam) [Pat]
params
    hidden :: Names
hidden = Names
param_all_names Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` Names
param_names

inferredReturnType :: SrcLoc -> [Pat] -> PatType -> TermTypeM StructType
inferredReturnType :: SrcLoc -> [Pat] -> PatType -> TermTypeM StructType
inferredReturnType SrcLoc
loc [Pat]
params PatType
t =
  -- The inferred type may refer to names that are bound by the
  -- parameter patterns, but which will not be visible in the type.
  -- These we must turn into fresh type variables, which will be
  -- existential in the return type.
  ((PatType, [VName]) -> StructType)
-> TermTypeM (PatType, [VName]) -> TermTypeM StructType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType)
-> ((PatType, [VName]) -> PatType)
-> (PatType, [VName])
-> StructType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (PatType, [VName]) -> PatType
forall a b. (a, b) -> a
fst) (TermTypeM (PatType, [VName]) -> TermTypeM StructType)
-> TermTypeM (PatType, [VName]) -> TermTypeM StructType
forall a b. (a -> b) -> a -> b
$
    SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType
      SrcLoc
loc
      ((VName -> Ident -> Bool) -> Map VName Ident -> Map VName Ident
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey (Bool -> Ident -> Bool
forall a b. a -> b -> a
const (Bool -> Ident -> Bool)
-> (VName -> Bool) -> VName -> Ident -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
hidden)) (Map VName Ident -> Map VName Ident)
-> Map VName Ident -> Map VName Ident
forall a b. (a -> b) -> a -> b
$ (Pat -> Map VName Ident) -> [Pat] -> Map VName Ident
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap Pat -> Map VName Ident
forall (f :: * -> *).
Functor f =>
PatBase f VName -> Map VName (IdentBase f VName)
patternMap [Pat]
params)
      (PatType -> TermTypeM (PatType, [VName]))
-> PatType -> TermTypeM (PatType, [VName])
forall a b. (a -> b) -> a -> b
$ [Pat] -> PatType -> PatType
inferReturnUniqueness [Pat]
params PatType
t
  where
    hidden :: Names
hidden = [Pat] -> Names
hiddenParamNames [Pat]
params

checkBinding ::
  ( Name,
    Maybe UncheckedTypeExp,
    [UncheckedTypeParam],
    [UncheckedPat],
    UncheckedExp,
    SrcLoc
  ) ->
  TermTypeM
    ( [TypeParam],
      [Pat],
      Maybe (TypeExp VName),
      StructType,
      [VName],
      Exp
    )
checkBinding :: (Name, Maybe (TypeExp Name), [UncheckedTypeParam], [UncheckedPat],
 UncheckedExp, SrcLoc)
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
checkBinding (Name
fname, Maybe (TypeExp Name)
maybe_retdecl, [UncheckedTypeParam]
tparams, [UncheckedPat]
params, UncheckedExp
body, SrcLoc
loc) =
  TermTypeM
  ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
   Exp)
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall b. TermTypeM b -> TermTypeM b
noUnique (TermTypeM
   ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
    Exp)
 -> TermTypeM
      ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
       Exp))
-> (([TypeParam]
     -> [Pat]
     -> TermTypeM
          ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
           Exp))
    -> TermTypeM
         ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
          Exp))
-> ([TypeParam]
    -> [Pat]
    -> TermTypeM
         ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
          Exp))
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeM
  ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
   Exp)
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall b. TermTypeM b -> TermTypeM b
incLevel (TermTypeM
   ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
    Exp)
 -> TermTypeM
      ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
       Exp))
-> (([TypeParam]
     -> [Pat]
     -> TermTypeM
          ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
           Exp))
    -> TermTypeM
         ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
          Exp))
-> ([TypeParam]
    -> [Pat]
    -> TermTypeM
         ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
          Exp))
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [UncheckedTypeParam]
-> [UncheckedPat]
-> ([TypeParam]
    -> [Pat]
    -> TermTypeM
         ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
          Exp))
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall a.
[UncheckedTypeParam]
-> [UncheckedPat]
-> ([TypeParam] -> [Pat] -> TermTypeM a)
-> TermTypeM a
bindingParams [UncheckedTypeParam]
tparams [UncheckedPat]
params (([TypeParam]
  -> [Pat]
  -> TermTypeM
       ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
        Exp))
 -> TermTypeM
      ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
       Exp))
-> ([TypeParam]
    -> [Pat]
    -> TermTypeM
         ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
          Exp))
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall a b. (a -> b) -> a -> b
$ \[TypeParam]
tparams' [Pat]
params' -> do
    Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when ([UncheckedPat] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [UncheckedPat]
params Bool -> Bool -> Bool
&& (UncheckedTypeParam -> Bool) -> [UncheckedTypeParam] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any UncheckedTypeParam -> Bool
forall vn. TypeParamBase vn -> Bool
isSizeParam [UncheckedTypeParam]
tparams) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
      SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError
        SrcLoc
loc
        Notes
forall a. Monoid a => a
mempty
        Doc
"Size parameters are only allowed on bindings that also have value parameters."

    Maybe (TypeExp VName, StructType)
maybe_retdecl' <- Maybe (TypeExp Name)
-> (TypeExp Name -> TermTypeM (TypeExp VName, StructType))
-> TermTypeM (Maybe (TypeExp VName, StructType))
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
t a -> (a -> m b) -> m (t b)
forM Maybe (TypeExp Name)
maybe_retdecl ((TypeExp Name -> TermTypeM (TypeExp VName, StructType))
 -> TermTypeM (Maybe (TypeExp VName, StructType)))
-> (TypeExp Name -> TermTypeM (TypeExp VName, StructType))
-> TermTypeM (Maybe (TypeExp VName, StructType))
forall a b. (a -> b) -> a -> b
$ \TypeExp Name
retdecl -> do
      (TypeExp VName
retdecl', StructType
ret_nodims, Liftedness
_) <- TypeExp Name -> TermTypeM (TypeExp VName, StructType, Liftedness)
forall (m :: * -> *).
MonadTypeChecker m =>
TypeExp Name -> m (TypeExp VName, StructType, Liftedness)
checkTypeExp TypeExp Name
retdecl
      (StructType
ret, [VName]
_) <- SrcLoc
-> String
-> Rigidity
-> StructType
-> TermTypeM (StructType, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"funret" Rigidity
Nonrigid StructType
ret_nodims
      (TypeExp VName, StructType)
-> TermTypeM (TypeExp VName, StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return (TypeExp VName
retdecl', StructType
ret)

    Exp
body' <-
      [Pat]
-> UncheckedExp -> Maybe StructType -> SrcLoc -> TermTypeM Exp
checkFunBody
        [Pat]
params'
        UncheckedExp
body
        ((TypeExp VName, StructType) -> StructType
forall a b. (a, b) -> b
snd ((TypeExp VName, StructType) -> StructType)
-> Maybe (TypeExp VName, StructType) -> Maybe StructType
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe (TypeExp VName, StructType)
maybe_retdecl')
        (SrcLoc
-> (TypeExp Name -> SrcLoc) -> Maybe (TypeExp Name) -> SrcLoc
forall b a. b -> (a -> b) -> Maybe a -> b
maybe SrcLoc
loc TypeExp Name -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Maybe (TypeExp Name)
maybe_retdecl)

    [Pat]
params'' <- (Pat -> TermTypeM Pat) -> [Pat] -> TermTypeM [Pat]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM Pat -> TermTypeM Pat
forall e. ASTMappable e => e -> TermTypeM e
updateTypes [Pat]
params'
    PatType
body_t <- Exp -> TermTypeM PatType
expTypeFully Exp
body'

    (Maybe (TypeExp VName)
maybe_retdecl'', StructType
rettype) <- case Maybe (TypeExp VName, StructType)
maybe_retdecl' of
      Just (TypeExp VName
retdecl', StructType
ret) -> do
        let rettype_structural :: TypeBase () ()
rettype_structural = StructType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural StructType
ret
        TypeBase () () -> [Pat] -> PatType -> TermTypeM ()
forall (t :: * -> *) shape as shape.
Foldable t =>
TypeBase shape as
-> t Pat -> TypeBase shape Aliasing -> TermTypeM ()
checkReturnAlias TypeBase () ()
rettype_structural [Pat]
params'' PatType
body_t

        Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when ([UncheckedPat] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [UncheckedPat]
params) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ SrcLoc -> TypeBase () () -> TermTypeM ()
nothingMustBeUnique SrcLoc
loc TypeBase () ()
rettype_structural

        StructType
ret' <- StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully StructType
ret

        (Maybe (TypeExp VName), StructType)
-> TermTypeM (Maybe (TypeExp VName), StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return (TypeExp VName -> Maybe (TypeExp VName)
forall a. a -> Maybe a
Just TypeExp VName
retdecl', StructType
ret')
      Maybe (TypeExp VName, StructType)
Nothing
        | [UncheckedPat] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [UncheckedPat]
params ->
          (Maybe (TypeExp VName), StructType)
-> TermTypeM (Maybe (TypeExp VName), StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (TypeExp VName)
forall a. Maybe a
Nothing, PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct (PatType -> StructType) -> PatType -> StructType
forall a b. (a -> b) -> a -> b
$ PatType
body_t PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique)
        | Bool
otherwise -> do
          StructType
body_t' <- SrcLoc -> [Pat] -> PatType -> TermTypeM StructType
inferredReturnType SrcLoc
loc [Pat]
params'' PatType
body_t
          (Maybe (TypeExp VName), StructType)
-> TermTypeM (Maybe (TypeExp VName), StructType)
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (TypeExp VName)
forall a. Maybe a
Nothing, StructType
body_t')

    Maybe Name -> [Pat] -> TermTypeM ()
verifyFunctionParams (Name -> Maybe Name
forall a. a -> Maybe a
Just Name
fname) [Pat]
params''

    ([TypeParam]
tparams'', [Pat]
params''', StructType
rettype'', [VName]
retext) <-
      Name
-> SrcLoc
-> [TypeParam]
-> [Pat]
-> StructType
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
letGeneralise Name
fname SrcLoc
loc [TypeParam]
tparams' [Pat]
params'' StructType
rettype

    [Pat] -> PatType -> SrcLoc -> TermTypeM ()
checkGlobalAliases [Pat]
params'' PatType
body_t SrcLoc
loc

    ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
 Exp)
-> TermTypeM
     ([TypeParam], [Pat], Maybe (TypeExp VName), StructType, [VName],
      Exp)
forall (m :: * -> *) a. Monad m => a -> m a
return ([TypeParam]
tparams'', [Pat]
params''', Maybe (TypeExp VName)
maybe_retdecl'', StructType
rettype'', [VName]
retext, Exp
body')
  where
    checkReturnAlias :: TypeBase shape as
-> t Pat -> TypeBase shape Aliasing -> TermTypeM ()
checkReturnAlias TypeBase shape as
rettp t Pat
params' =
      (Set (Uniqueness, VName)
 -> (Uniqueness, Names) -> TermTypeM (Set (Uniqueness, VName)))
-> Set (Uniqueness, VName) -> [(Uniqueness, Names)] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m ()
foldM_ (t Pat
-> Set (Uniqueness, VName)
-> (Uniqueness, Names)
-> TermTypeM (Set (Uniqueness, VName))
forall (t :: * -> *).
Foldable t =>
t Pat
-> Set (Uniqueness, VName)
-> (Uniqueness, Names)
-> TermTypeM (Set (Uniqueness, VName))
checkReturnAlias' t Pat
params') Set (Uniqueness, VName)
forall a. Set a
S.empty ([(Uniqueness, Names)] -> TermTypeM ())
-> (TypeBase shape Aliasing -> [(Uniqueness, Names)])
-> TypeBase shape Aliasing
-> TermTypeM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TypeBase shape as
-> TypeBase shape Aliasing -> [(Uniqueness, Names)]
forall shape as shape.
TypeBase shape as
-> TypeBase shape Aliasing -> [(Uniqueness, Names)]
returnAliasing TypeBase shape as
rettp
    checkReturnAlias' :: t Pat
-> Set (Uniqueness, VName)
-> (Uniqueness, Names)
-> TermTypeM (Set (Uniqueness, VName))
checkReturnAlias' t Pat
params' Set (Uniqueness, VName)
seen (Uniqueness
Unique, Names
names)
      | (VName -> Bool) -> [VName] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any (VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` ((Uniqueness, VName) -> VName) -> Set (Uniqueness, VName) -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map (Uniqueness, VName) -> VName
forall a b. (a, b) -> b
snd Set (Uniqueness, VName)
seen) ([VName] -> Bool) -> [VName] -> Bool
forall a b. (a -> b) -> a -> b
$ Names -> [VName]
forall a. Set a -> [a]
S.toList Names
names =
        Name -> SrcLoc -> TermTypeM (Set (Uniqueness, VName))
forall a. Name -> SrcLoc -> TermTypeM a
uniqueReturnAliased Name
fname SrcLoc
loc
      | Bool
otherwise = do
        t Pat -> Names -> TermTypeM ()
forall (t :: * -> *). Foldable t => t Pat -> Names -> TermTypeM ()
notAliasingParam t Pat
params' Names
names
        Set (Uniqueness, VName) -> TermTypeM (Set (Uniqueness, VName))
forall (m :: * -> *) a. Monad m => a -> m a
return (Set (Uniqueness, VName) -> TermTypeM (Set (Uniqueness, VName)))
-> Set (Uniqueness, VName) -> TermTypeM (Set (Uniqueness, VName))
forall a b. (a -> b) -> a -> b
$ Set (Uniqueness, VName)
seen Set (Uniqueness, VName)
-> Set (Uniqueness, VName) -> Set (Uniqueness, VName)
forall a. Ord a => Set a -> Set a -> Set a
`S.union` Uniqueness -> Names -> Set (Uniqueness, VName)
forall t t. (Ord t, Ord t) => t -> Set t -> Set (t, t)
tag Uniqueness
Unique Names
names
    checkReturnAlias' t Pat
_ Set (Uniqueness, VName)
seen (Uniqueness
Nonunique, Names
names)
      | ((Uniqueness, VName) -> Bool) -> [(Uniqueness, VName)] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any ((Uniqueness, VName) -> Set (Uniqueness, VName) -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Set (Uniqueness, VName)
seen) ([(Uniqueness, VName)] -> Bool) -> [(Uniqueness, VName)] -> Bool
forall a b. (a -> b) -> a -> b
$ Set (Uniqueness, VName) -> [(Uniqueness, VName)]
forall a. Set a -> [a]
S.toList (Set (Uniqueness, VName) -> [(Uniqueness, VName)])
-> Set (Uniqueness, VName) -> [(Uniqueness, VName)]
forall a b. (a -> b) -> a -> b
$ Uniqueness -> Names -> Set (Uniqueness, VName)
forall t t. (Ord t, Ord t) => t -> Set t -> Set (t, t)
tag Uniqueness
Unique Names
names =
        Name -> SrcLoc -> TermTypeM (Set (Uniqueness, VName))
forall a. Name -> SrcLoc -> TermTypeM a
uniqueReturnAliased Name
fname SrcLoc
loc
      | Bool
otherwise = Set (Uniqueness, VName) -> TermTypeM (Set (Uniqueness, VName))
forall (m :: * -> *) a. Monad m => a -> m a
return (Set (Uniqueness, VName) -> TermTypeM (Set (Uniqueness, VName)))
-> Set (Uniqueness, VName) -> TermTypeM (Set (Uniqueness, VName))
forall a b. (a -> b) -> a -> b
$ Set (Uniqueness, VName)
seen Set (Uniqueness, VName)
-> Set (Uniqueness, VName) -> Set (Uniqueness, VName)
forall a. Ord a => Set a -> Set a -> Set a
`S.union` Uniqueness -> Names -> Set (Uniqueness, VName)
forall t t. (Ord t, Ord t) => t -> Set t -> Set (t, t)
tag Uniqueness
Nonunique Names
names

    notAliasingParam :: t Pat -> Names -> TermTypeM ()
notAliasingParam t Pat
params' Names
names =
      t Pat -> (Pat -> TermTypeM ()) -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
t a -> (a -> m b) -> m ()
forM_ t Pat
params' ((Pat -> TermTypeM ()) -> TermTypeM ())
-> (Pat -> TermTypeM ()) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \Pat
p ->
        let consumedNonunique :: Ident -> Bool
consumedNonunique Ident
p' =
              Bool -> Bool
not (PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique (PatType -> Bool) -> PatType -> Bool
forall a b. (a -> b) -> a -> b
$ Info PatType -> PatType
forall a. Info a -> a
unInfo (Info PatType -> PatType) -> Info PatType -> PatType
forall a b. (a -> b) -> a -> b
$ Ident -> Info PatType
forall (f :: * -> *) vn. IdentBase f vn -> f PatType
identType Ident
p') Bool -> Bool -> Bool
&& (Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
p' VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
names)
         in case (Ident -> Bool) -> [Ident] -> Maybe Ident
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Maybe a
find Ident -> Bool
consumedNonunique ([Ident] -> Maybe Ident) -> [Ident] -> Maybe Ident
forall a b. (a -> b) -> a -> b
$ Set Ident -> [Ident]
forall a. Set a -> [a]
S.toList (Set Ident -> [Ident]) -> Set Ident -> [Ident]
forall a b. (a -> b) -> a -> b
$ Pat -> Set Ident
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set (IdentBase f vn)
patIdents Pat
p of
              Just Ident
p' ->
                Name -> Name -> SrcLoc -> TermTypeM ()
returnAliased Name
fname (VName -> Name
baseName (VName -> Name) -> VName -> Name
forall a b. (a -> b) -> a -> b
$ Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
p') SrcLoc
loc
              Maybe Ident
Nothing ->
                () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

    tag :: t -> Set t -> Set (t, t)
tag t
u = (t -> (t, t)) -> Set t -> Set (t, t)
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map (t
u,)

    returnAliasing :: TypeBase shape as
-> TypeBase shape Aliasing -> [(Uniqueness, Names)]
returnAliasing (Scalar (Record Map Name (TypeBase shape as)
ets1)) (Scalar (Record Map Name (TypeBase shape Aliasing)
ets2)) =
      [[(Uniqueness, Names)]] -> [(Uniqueness, Names)]
forall (t :: * -> *) a. Foldable t => t [a] -> [a]
concat ([[(Uniqueness, Names)]] -> [(Uniqueness, Names)])
-> [[(Uniqueness, Names)]] -> [(Uniqueness, Names)]
forall a b. (a -> b) -> a -> b
$ Map Name [(Uniqueness, Names)] -> [[(Uniqueness, Names)]]
forall k a. Map k a -> [a]
M.elems (Map Name [(Uniqueness, Names)] -> [[(Uniqueness, Names)]])
-> Map Name [(Uniqueness, Names)] -> [[(Uniqueness, Names)]]
forall a b. (a -> b) -> a -> b
$ (TypeBase shape as
 -> TypeBase shape Aliasing -> [(Uniqueness, Names)])
-> Map Name (TypeBase shape as)
-> Map Name (TypeBase shape Aliasing)
-> Map Name [(Uniqueness, Names)]
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
M.intersectionWith TypeBase shape as
-> TypeBase shape Aliasing -> [(Uniqueness, Names)]
returnAliasing Map Name (TypeBase shape as)
ets1 Map Name (TypeBase shape Aliasing)
ets2
    returnAliasing TypeBase shape as
expected TypeBase shape Aliasing
got =
      [(TypeBase shape as -> Uniqueness
forall shape as. TypeBase shape as -> Uniqueness
uniqueness TypeBase shape as
expected, (Alias -> VName) -> Aliasing -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> VName
aliasVar (Aliasing -> Names) -> Aliasing -> Names
forall a b. (a -> b) -> a -> b
$ TypeBase shape Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase shape Aliasing
got)]

-- | Extract all the shape names that occur in positive position
-- (roughly, left side of an arrow) in a given type.
typeDimNamesPos :: TypeBase (DimDecl VName) als -> S.Set VName
typeDimNamesPos :: TypeBase (DimDecl VName) als -> Names
typeDimNamesPos (Scalar (Arrow als
_ PName
_ TypeBase (DimDecl VName) als
t1 TypeBase (DimDecl VName) als
t2)) = TypeBase (DimDecl VName) als -> Names
forall als. TypeBase (DimDecl VName) als -> Names
onParam TypeBase (DimDecl VName) als
t1 Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> TypeBase (DimDecl VName) als -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNamesPos TypeBase (DimDecl VName) als
t2
  where
    onParam :: TypeBase (DimDecl VName) als -> S.Set VName
    onParam :: TypeBase (DimDecl VName) als -> Names
onParam (Scalar Arrow {}) = Names
forall a. Monoid a => a
mempty
    onParam (Scalar (Record Map Name (TypeBase (DimDecl VName) als)
fs)) = [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ([Names] -> Names) -> [Names] -> Names
forall a b. (a -> b) -> a -> b
$ (TypeBase (DimDecl VName) als -> Names)
-> [TypeBase (DimDecl VName) als] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map TypeBase (DimDecl VName) als -> Names
forall als. TypeBase (DimDecl VName) als -> Names
onParam ([TypeBase (DimDecl VName) als] -> [Names])
-> [TypeBase (DimDecl VName) als] -> [Names]
forall a b. (a -> b) -> a -> b
$ Map Name (TypeBase (DimDecl VName) als)
-> [TypeBase (DimDecl VName) als]
forall k a. Map k a -> [a]
M.elems Map Name (TypeBase (DimDecl VName) als)
fs
    onParam (Scalar (TypeVar als
_ Uniqueness
_ TypeName
_ [TypeArg (DimDecl VName)]
targs)) = [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ([Names] -> Names) -> [Names] -> Names
forall a b. (a -> b) -> a -> b
$ (TypeArg (DimDecl VName) -> Names)
-> [TypeArg (DimDecl VName)] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map TypeArg (DimDecl VName) -> Names
onTypeArg [TypeArg (DimDecl VName)]
targs
    onParam TypeBase (DimDecl VName) als
t = TypeBase (DimDecl VName) als -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNames TypeBase (DimDecl VName) als
t
    onTypeArg :: TypeArg (DimDecl VName) -> Names
onTypeArg (TypeArgDim (NamedDim QualName VName
d) SrcLoc
_) = VName -> Names
forall a. a -> Set a
S.singleton (VName -> Names) -> VName -> Names
forall a b. (a -> b) -> a -> b
$ QualName VName -> VName
forall vn. QualName vn -> vn
qualLeaf QualName VName
d
    onTypeArg (TypeArgDim DimDecl VName
_ SrcLoc
_) = Names
forall a. Monoid a => a
mempty
    onTypeArg (TypeArgType StructType
t SrcLoc
_) = StructType -> Names
forall als. TypeBase (DimDecl VName) als -> Names
onParam StructType
t
typeDimNamesPos TypeBase (DimDecl VName) als
_ = Names
forall a. Monoid a => a
mempty

checkGlobalAliases :: [Pat] -> PatType -> SrcLoc -> TermTypeM ()
checkGlobalAliases :: [Pat] -> PatType -> SrcLoc -> TermTypeM ()
checkGlobalAliases [Pat]
params PatType
body_t SrcLoc
loc = do
  Map VName ValBinding
vtable <- (TermEnv -> Map VName ValBinding)
-> TermTypeM (Map VName ValBinding)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((TermEnv -> Map VName ValBinding)
 -> TermTypeM (Map VName ValBinding))
-> (TermEnv -> Map VName ValBinding)
-> TermTypeM (Map VName ValBinding)
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable (TermScope -> Map VName ValBinding)
-> (TermEnv -> TermScope) -> TermEnv -> Map VName ValBinding
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermEnv -> TermScope
termScope
  let isLocal :: VName -> Bool
isLocal VName
v = case VName
v VName -> Map VName ValBinding -> Maybe ValBinding
forall k a. Ord k => k -> Map k a -> Maybe a
`M.lookup` Map VName ValBinding
vtable of
        Just (BoundV Locality
Local [TypeParam]
_ PatType
_) -> Bool
True
        Maybe ValBinding
_ -> Bool
False
  let als :: [VName]
als =
        (VName -> Bool) -> [VName] -> [VName]
forall a. (a -> Bool) -> [a] -> [a]
filter (Bool -> Bool
not (Bool -> Bool) -> (VName -> Bool) -> VName -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> Bool
isLocal) ([VName] -> [VName]) -> [VName] -> [VName]
forall a b. (a -> b) -> a -> b
$
          Names -> [VName]
forall a. Set a -> [a]
S.toList (Names -> [VName]) -> Names -> [VName]
forall a b. (a -> b) -> a -> b
$
            PatType -> Names
boundArrayAliases PatType
body_t
              Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` (Pat -> Names) -> [Pat] -> Names
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames [Pat]
params
  case [VName]
als of
    VName
v : [VName]
_
      | Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ [Pat] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [Pat]
params ->
        SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
          Doc
"Function result aliases the free variable "
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
v)
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
            Doc -> Doc -> Doc
</> Doc
"Use" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote Doc
"copy" Doc -> Doc -> Doc
<+> Doc
"to break the aliasing."
    [VName]
_ ->
      () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

inferReturnUniqueness :: [Pat] -> PatType -> PatType
inferReturnUniqueness :: [Pat] -> PatType -> PatType
inferReturnUniqueness [Pat]
params PatType
t =
  let forbidden :: Names
forbidden = PatType -> Names
aliasesMultipleTimes PatType
t
      uniques :: Names
uniques = [Pat] -> Names
uniqueParamNames [Pat]
params
      delve :: PatType -> PatType
delve (Scalar (Record Map Name PatType
fs)) =
        ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall dim as. ScalarTypeBase dim as -> TypeBase dim as
Scalar (ScalarTypeBase (DimDecl VName) Aliasing -> PatType)
-> ScalarTypeBase (DimDecl VName) Aliasing -> PatType
forall a b. (a -> b) -> a -> b
$ Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall dim as. Map Name (TypeBase dim as) -> ScalarTypeBase dim as
Record (Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing)
-> Map Name PatType -> ScalarTypeBase (DimDecl VName) Aliasing
forall a b. (a -> b) -> a -> b
$ (PatType -> PatType) -> Map Name PatType -> Map Name PatType
forall a b k. (a -> b) -> Map k a -> Map k b
M.map PatType -> PatType
delve Map Name PatType
fs
      delve PatType
t'
        | (VName -> Bool) -> Names -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all (VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
uniques) (PatType -> Names
boundArrayAliases PatType
t'),
          Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ (Alias -> Bool) -> Aliasing -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any ((VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
forbidden) (VName -> Bool) -> (Alias -> VName) -> Alias -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Alias -> VName
aliasVar) (PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
t') =
          PatType
t'
        | Bool
otherwise =
          PatType
t' PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique
   in PatType -> PatType
delve PatType
t

-- An alias inhibits uniqueness if it is used in disjoint values.
aliasesMultipleTimes :: PatType -> Names
aliasesMultipleTimes :: PatType -> Names
aliasesMultipleTimes = [VName] -> Names
forall a. Ord a => [a] -> Set a
S.fromList ([VName] -> Names) -> (PatType -> [VName]) -> PatType -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((VName, Int) -> VName) -> [(VName, Int)] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map (VName, Int) -> VName
forall a b. (a, b) -> a
fst ([(VName, Int)] -> [VName])
-> (PatType -> [(VName, Int)]) -> PatType -> [VName]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((VName, Int) -> Bool) -> [(VName, Int)] -> [(VName, Int)]
forall a. (a -> Bool) -> [a] -> [a]
filter ((Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
1) (Int -> Bool) -> ((VName, Int) -> Int) -> (VName, Int) -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (VName, Int) -> Int
forall a b. (a, b) -> b
snd) ([(VName, Int)] -> [(VName, Int)])
-> (PatType -> [(VName, Int)]) -> PatType -> [(VName, Int)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map VName Int -> [(VName, Int)]
forall k a. Map k a -> [(k, a)]
M.toList (Map VName Int -> [(VName, Int)])
-> (PatType -> Map VName Int) -> PatType -> [(VName, Int)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PatType -> Map VName Int
forall shape. TypeBase shape Aliasing -> Map VName Int
delve
  where
    delve :: TypeBase shape Aliasing -> Map VName Int
delve (Scalar (Record Map Name (TypeBase shape Aliasing)
fs)) =
      (Map VName Int -> Map VName Int -> Map VName Int)
-> Map VName Int -> [Map VName Int] -> Map VName Int
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' ((Int -> Int -> Int)
-> Map VName Int -> Map VName Int -> Map VName Int
forall k a. Ord k => (a -> a -> a) -> Map k a -> Map k a -> Map k a
M.unionWith Int -> Int -> Int
forall a. Num a => a -> a -> a
(+)) Map VName Int
forall a. Monoid a => a
mempty ([Map VName Int] -> Map VName Int)
-> [Map VName Int] -> Map VName Int
forall a b. (a -> b) -> a -> b
$ (TypeBase shape Aliasing -> Map VName Int)
-> [TypeBase shape Aliasing] -> [Map VName Int]
forall a b. (a -> b) -> [a] -> [b]
map TypeBase shape Aliasing -> Map VName Int
delve ([TypeBase shape Aliasing] -> [Map VName Int])
-> [TypeBase shape Aliasing] -> [Map VName Int]
forall a b. (a -> b) -> a -> b
$ Map Name (TypeBase shape Aliasing) -> [TypeBase shape Aliasing]
forall k a. Map k a -> [a]
M.elems Map Name (TypeBase shape Aliasing)
fs
    delve TypeBase shape Aliasing
t =
      [(VName, Int)] -> Map VName Int
forall k a. Ord k => [(k, a)] -> Map k a
M.fromList ([(VName, Int)] -> Map VName Int)
-> [(VName, Int)] -> Map VName Int
forall a b. (a -> b) -> a -> b
$ [VName] -> [Int] -> [(VName, Int)]
forall a b. [a] -> [b] -> [(a, b)]
zip ((Alias -> VName) -> [Alias] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map Alias -> VName
aliasVar ([Alias] -> [VName]) -> [Alias] -> [VName]
forall a b. (a -> b) -> a -> b
$ Aliasing -> [Alias]
forall a. Set a -> [a]
S.toList (TypeBase shape Aliasing -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases TypeBase shape Aliasing
t)) ([Int] -> [(VName, Int)]) -> [Int] -> [(VName, Int)]
forall a b. (a -> b) -> a -> b
$ Int -> [Int]
forall a. a -> [a]
repeat (Int
1 :: Int)

uniqueParamNames :: [Pat] -> Names
uniqueParamNames :: [Pat] -> Names
uniqueParamNames =
  (Ident -> VName) -> Set Ident -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName
    (Set Ident -> Names) -> ([Pat] -> Set Ident) -> [Pat] -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Ident -> Bool) -> Set Ident -> Set Ident
forall a. (a -> Bool) -> Set a -> Set a
S.filter (PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique (PatType -> Bool) -> (Ident -> PatType) -> Ident -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Info PatType -> PatType
forall a. Info a -> a
unInfo (Info PatType -> PatType)
-> (Ident -> Info PatType) -> Ident -> PatType
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Ident -> Info PatType
forall (f :: * -> *) vn. IdentBase f vn -> f PatType
identType)
    (Set Ident -> Set Ident)
-> ([Pat] -> Set Ident) -> [Pat] -> Set Ident
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Pat -> Set Ident) -> [Pat] -> Set Ident
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap Pat -> Set Ident
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set (IdentBase f vn)
patIdents

boundArrayAliases :: PatType -> S.Set VName
boundArrayAliases :: PatType -> Names
boundArrayAliases (Array Aliasing
als Uniqueness
_ ScalarTypeBase (DimDecl VName) ()
_ ShapeDecl (DimDecl VName)
_) = Aliasing -> Names
boundAliases Aliasing
als
boundArrayAliases (Scalar Prim {}) = Names
forall a. Monoid a => a
mempty
boundArrayAliases (Scalar (Record Map Name PatType
fs)) = (PatType -> Names) -> Map Name PatType -> Names
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap PatType -> Names
boundArrayAliases Map Name PatType
fs
boundArrayAliases (Scalar (TypeVar Aliasing
als Uniqueness
_ TypeName
_ [TypeArg (DimDecl VName)]
_)) = Aliasing -> Names
boundAliases Aliasing
als
boundArrayAliases (Scalar Arrow {}) = Names
forall a. Monoid a => a
mempty
boundArrayAliases (Scalar (Sum Map Name [PatType]
fs)) =
  [Names] -> Names
forall a. Monoid a => [a] -> a
mconcat ([Names] -> Names) -> [Names] -> Names
forall a b. (a -> b) -> a -> b
$ ([PatType] -> [Names]) -> [[PatType]] -> [Names]
forall (t :: * -> *) a b. Foldable t => (a -> [b]) -> t a -> [b]
concatMap ((PatType -> Names) -> [PatType] -> [Names]
forall a b. (a -> b) -> [a] -> [b]
map PatType -> Names
boundArrayAliases) ([[PatType]] -> [Names]) -> [[PatType]] -> [Names]
forall a b. (a -> b) -> a -> b
$ Map Name [PatType] -> [[PatType]]
forall k a. Map k a -> [a]
M.elems Map Name [PatType]
fs

-- | The set of in-scope variables that are being aliased.
boundAliases :: Aliasing -> S.Set VName
boundAliases :: Aliasing -> Names
boundAliases = (Alias -> VName) -> Aliasing -> Names
forall b a. Ord b => (a -> b) -> Set a -> Set b
S.map Alias -> VName
aliasVar (Aliasing -> Names) -> (Aliasing -> Aliasing) -> Aliasing -> Names
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Alias -> Bool) -> Aliasing -> Aliasing
forall a. (a -> Bool) -> Set a -> Set a
S.filter Alias -> Bool
bound
  where
    bound :: Alias -> Bool
bound AliasBound {} = Bool
True
    bound AliasFree {} = Bool
False

nothingMustBeUnique :: SrcLoc -> TypeBase () () -> TermTypeM ()
nothingMustBeUnique :: SrcLoc -> TypeBase () () -> TermTypeM ()
nothingMustBeUnique SrcLoc
loc = TypeBase () () -> TermTypeM ()
forall dim as. TypeBase dim as -> TermTypeM ()
check
  where
    check :: TypeBase dim as -> TermTypeM ()
check (Array as
_ Uniqueness
Unique ScalarTypeBase dim ()
_ ShapeDecl dim
_) = TermTypeM ()
forall a. TermTypeM a
bad
    check (Scalar (TypeVar as
_ Uniqueness
Unique TypeName
_ [TypeArg dim]
_)) = TermTypeM ()
forall a. TermTypeM a
bad
    check (Scalar (Record Map Name (TypeBase dim as)
fs)) = (TypeBase dim as -> TermTypeM ())
-> Map Name (TypeBase dim as) -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ TypeBase dim as -> TermTypeM ()
check Map Name (TypeBase dim as)
fs
    check (Scalar (Sum Map Name [TypeBase dim as]
fs)) = ([TypeBase dim as] -> TermTypeM ())
-> Map Name [TypeBase dim as] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ ((TypeBase dim as -> TermTypeM ())
-> [TypeBase dim as] -> TermTypeM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ TypeBase dim as -> TermTypeM ()
check) Map Name [TypeBase dim as]
fs
    check TypeBase dim as
_ = () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
    bad :: TermTypeM a
bad = SrcLoc -> Notes -> Doc -> TermTypeM a
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
loc Notes
forall a. Monoid a => a
mempty Doc
"A top-level constant cannot have a unique type."

-- | Verify certain restrictions on function parameters, and bail out
-- on dubious constructions.
--
-- These restrictions apply to all functions (anonymous or otherwise).
-- Top-level functions have further restrictions that are checked
-- during let-generalisation.
verifyFunctionParams :: Maybe Name -> [Pat] -> TermTypeM ()
verifyFunctionParams :: Maybe Name -> [Pat] -> TermTypeM ()
verifyFunctionParams Maybe Name
fname [Pat]
params =
  Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (Maybe Name -> Checking
CheckingParams Maybe Name
fname) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
    Names -> [Pat] -> TermTypeM ()
forall (m :: * -> *). MonadTypeChecker m => Names -> [Pat] -> m ()
verifyParams ((Pat -> Names) -> [Pat] -> Names
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap Pat -> Names
forall (f :: * -> *) vn.
(Functor f, Ord vn) =>
PatBase f vn -> Set vn
patNames [Pat]
params) ([Pat] -> TermTypeM ()) -> TermTypeM [Pat] -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< (Pat -> TermTypeM Pat) -> [Pat] -> TermTypeM [Pat]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM Pat -> TermTypeM Pat
forall e. ASTMappable e => e -> TermTypeM e
updateTypes [Pat]
params
  where
    verifyParams :: Names -> [Pat] -> m ()
verifyParams Names
forbidden (Pat
p : [Pat]
ps)
      | VName
d : [VName]
_ <- Names -> [VName]
forall a. Set a -> [a]
S.toList (Names -> [VName]) -> Names -> [VName]
forall a b. (a -> b) -> a -> b
$ Pat -> Names
patternDimNames Pat
p Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.intersection` Names
forbidden =
        Pat -> Notes -> Doc -> m ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError Pat
p Notes
forall a. Monoid a => a
mempty (Doc -> m ()) -> Doc -> m ()
forall a b. (a -> b) -> a -> b
$
          Doc
"Parameter" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Pat -> Doc
forall a. Pretty a => a -> Doc
ppr Pat
p)
            Doc -> Doc -> Doc
<+/> Doc
"refers to size" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
d)
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
comma
            Doc -> Doc -> Doc
<+/> String -> Doc
textwrap String
"which will not be accessible to the caller"
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
comma
            Doc -> Doc -> Doc
<+/> String -> Doc
textwrap String
"possibly because it is nested in a tuple or record."
            Doc -> Doc -> Doc
<+/> String -> Doc
textwrap String
"Consider ascribing an explicit type that does not reference "
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
d)
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
"."
      | Bool
otherwise = Names -> [Pat] -> m ()
verifyParams Names
forbidden' [Pat]
ps
      where
        forbidden' :: Names
forbidden' =
          case Pat -> (PName, StructType)
patternParam Pat
p of
            (Named VName
v, StructType
_) -> Names
forbidden Names -> Names -> Names
forall a. Ord a => Set a -> Set a -> Set a
`S.difference` VName -> Names
forall a. a -> Set a
S.singleton VName
v
            (PName, StructType)
_ -> Names
forbidden
    verifyParams Names
_ [] = () -> m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

-- Returns the sizes of the immediate type produced,
-- the sizes of parameter types, and the sizes of return types.
dimUses :: StructType -> (Names, Names, Names)
dimUses :: StructType -> (Names, Names, Names)
dimUses = (State (Names, Names, Names) (TypeBase () ())
-> (Names, Names, Names) -> (Names, Names, Names)
forall s a. State s a -> s -> s
`execState` (Names, Names, Names)
forall a. Monoid a => a
mempty) (State (Names, Names, Names) (TypeBase () ())
 -> (Names, Names, Names))
-> (StructType -> State (Names, Names, Names) (TypeBase () ()))
-> StructType
-> (Names, Names, Names)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Names
 -> DimPos
 -> DimDecl VName
 -> StateT (Names, Names, Names) Identity ())
-> StructType -> State (Names, Names, Names) (TypeBase () ())
forall (f :: * -> *) fdim tdim als.
Applicative f =>
(Names -> DimPos -> fdim -> f tdim)
-> TypeBase fdim als -> f (TypeBase tdim als)
traverseDims Names
-> DimPos
-> DimDecl VName
-> StateT (Names, Names, Names) Identity ()
forall a (m :: * -> *) p.
(Ord a, MonadState (Set a, Set a, Set a) m) =>
p -> DimPos -> DimDecl a -> m ()
f
  where
    f :: p -> DimPos -> DimDecl a -> m ()
f p
_ DimPos
PosImmediate (NamedDim QualName a
v) =
      ((Set a, Set a, Set a) -> (Set a, Set a, Set a)) -> m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Set a, Set a, Set a)
-> (Set a, Set a, Set a) -> (Set a, Set a, Set a)
forall a. Semigroup a => a -> a -> a
<> (a -> Set a
forall a. a -> Set a
S.singleton (QualName a -> a
forall vn. QualName vn -> vn
qualLeaf QualName a
v), Set a
forall a. Monoid a => a
mempty, Set a
forall a. Monoid a => a
mempty))
    f p
_ DimPos
PosParam (NamedDim QualName a
v) =
      ((Set a, Set a, Set a) -> (Set a, Set a, Set a)) -> m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Set a, Set a, Set a)
-> (Set a, Set a, Set a) -> (Set a, Set a, Set a)
forall a. Semigroup a => a -> a -> a
<> (Set a
forall a. Monoid a => a
mempty, a -> Set a
forall a. a -> Set a
S.singleton (QualName a -> a
forall vn. QualName vn -> vn
qualLeaf QualName a
v), Set a
forall a. Monoid a => a
mempty))
    f p
_ DimPos
PosReturn (NamedDim QualName a
v) =
      ((Set a, Set a, Set a) -> (Set a, Set a, Set a)) -> m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((Set a, Set a, Set a)
-> (Set a, Set a, Set a) -> (Set a, Set a, Set a)
forall a. Semigroup a => a -> a -> a
<> (Set a
forall a. Monoid a => a
mempty, Set a
forall a. Monoid a => a
mempty, a -> Set a
forall a. a -> Set a
S.singleton (QualName a -> a
forall vn. QualName vn -> vn
qualLeaf QualName a
v)))
    f p
_ DimPos
_ DimDecl a
_ = () -> m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

-- | Find all type variables in the given type that are covered by the
-- constraints, and produce type parameters that close over them.
--
-- The passed-in list of type parameters is always prepended to the
-- produced list of type parameters.
closeOverTypes ::
  Name ->
  SrcLoc ->
  [TypeParam] ->
  [StructType] ->
  StructType ->
  Constraints ->
  TermTypeM ([TypeParam], StructType, [VName])
closeOverTypes :: Name
-> SrcLoc
-> [TypeParam]
-> [StructType]
-> StructType
-> Constraints
-> TermTypeM ([TypeParam], StructType, [VName])
closeOverTypes Name
defname SrcLoc
defloc [TypeParam]
tparams [StructType]
paramts StructType
ret Constraints
substs = do
  ([TypeParam]
more_tparams, [VName]
retext) <-
    [Either TypeParam VName] -> ([TypeParam], [VName])
forall a b. [Either a b] -> ([a], [b])
partitionEithers ([Either TypeParam VName] -> ([TypeParam], [VName]))
-> ([Maybe (Either TypeParam VName)] -> [Either TypeParam VName])
-> [Maybe (Either TypeParam VName)]
-> ([TypeParam], [VName])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Maybe (Either TypeParam VName)] -> [Either TypeParam VName]
forall a. [Maybe a] -> [a]
catMaybes
      ([Maybe (Either TypeParam VName)] -> ([TypeParam], [VName]))
-> TermTypeM [Maybe (Either TypeParam VName)]
-> TermTypeM ([TypeParam], [VName])
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ((VName, Constraint) -> TermTypeM (Maybe (Either TypeParam VName)))
-> [(VName, Constraint)]
-> TermTypeM [Maybe (Either TypeParam VName)]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM (VName, Constraint) -> TermTypeM (Maybe (Either TypeParam VName))
forall (m :: * -> *).
(MonadUnify m, MonadTypeChecker m) =>
(VName, Constraint) -> m (Maybe (Either TypeParam VName))
closeOver (Map VName Constraint -> [(VName, Constraint)]
forall k a. Map k a -> [(k, a)]
M.toList (Map VName Constraint -> [(VName, Constraint)])
-> Map VName Constraint -> [(VName, Constraint)]
forall a b. (a -> b) -> a -> b
$ ((Int, Constraint) -> Constraint)
-> Constraints -> Map VName Constraint
forall a b k. (a -> b) -> Map k a -> Map k b
M.map (Int, Constraint) -> Constraint
forall a b. (a, b) -> b
snd Constraints
to_close_over)
  let retToAnyDim :: VName -> Maybe (Subst t)
retToAnyDim VName
v = do
        Bool -> Maybe ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard (Bool -> Maybe ()) -> Bool -> Maybe ()
forall a b. (a -> b) -> a -> b
$ VName
v VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
ret_sizes
        UnknowableSize {} <- (Int, Constraint) -> Constraint
forall a b. (a, b) -> b
snd ((Int, Constraint) -> Constraint)
-> Maybe (Int, Constraint) -> Maybe Constraint
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> VName -> Constraints -> Maybe (Int, Constraint)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
v Constraints
substs
        Subst t -> Maybe (Subst t)
forall a. a -> Maybe a
Just (Subst t -> Maybe (Subst t)) -> Subst t -> Maybe (Subst t)
forall a b. (a -> b) -> a -> b
$ DimDecl VName -> Subst t
forall t. DimDecl VName -> Subst t
SizeSubst (DimDecl VName -> Subst t) -> DimDecl VName -> Subst t
forall a b. (a -> b) -> a -> b
$ Maybe VName -> DimDecl VName
forall vn. Maybe vn -> DimDecl vn
AnyDim (Maybe VName -> DimDecl VName) -> Maybe VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> Maybe VName
forall a. a -> Maybe a
Just VName
v
  ([TypeParam], StructType, [VName])
-> TermTypeM ([TypeParam], StructType, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return
    ( [TypeParam]
tparams [TypeParam] -> [TypeParam] -> [TypeParam]
forall a. [a] -> [a] -> [a]
++ [TypeParam]
more_tparams,
      TypeSubs -> StructType -> StructType
forall a. Substitutable a => TypeSubs -> a -> a
applySubst TypeSubs
forall t. VName -> Maybe (Subst t)
retToAnyDim StructType
ret,
      [VName]
retext
    )
  where
    t :: StructType
t = [StructType] -> StructType -> StructType
forall as dim.
Monoid as =>
[TypeBase dim as] -> TypeBase dim as -> TypeBase dim as
foldFunType [StructType]
paramts StructType
ret
    to_close_over :: Constraints
to_close_over = (VName -> (Int, Constraint) -> Bool) -> Constraints -> Constraints
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey (\VName
k (Int, Constraint)
_ -> VName
k VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
visible) Constraints
substs
    visible :: Names
visible = StructType -> Names
forall as dim. Monoid as => TypeBase dim as -> Names
typeVars StructType
t Names -> Names -> Names
forall a. Semigroup a => a -> a -> a
<> StructType -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNames StructType
t

    (Names
produced_sizes, Names
param_sizes, Names
ret_sizes) = StructType -> (Names, Names, Names)
dimUses StructType
t

    -- Avoid duplicate type parameters.
    closeOver :: (VName, Constraint) -> m (Maybe (Either TypeParam VName))
closeOver (VName
k, Constraint
_)
      | VName
k VName -> [VName] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` (TypeParam -> VName) -> [TypeParam] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName [TypeParam]
tparams =
        Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe (Either TypeParam VName)
forall a. Maybe a
Nothing
    closeOver (VName
k, NoConstraint Liftedness
l Usage
usage) =
      Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (Either TypeParam VName)
 -> m (Maybe (Either TypeParam VName)))
-> Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall a b. (a -> b) -> a -> b
$ Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a. a -> Maybe a
Just (Either TypeParam VName -> Maybe (Either TypeParam VName))
-> Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a b. (a -> b) -> a -> b
$ TypeParam -> Either TypeParam VName
forall a b. a -> Either a b
Left (TypeParam -> Either TypeParam VName)
-> TypeParam -> Either TypeParam VName
forall a b. (a -> b) -> a -> b
$ Liftedness -> VName -> SrcLoc -> TypeParam
forall vn. Liftedness -> vn -> SrcLoc -> TypeParamBase vn
TypeParamType Liftedness
l VName
k (SrcLoc -> TypeParam) -> SrcLoc -> TypeParam
forall a b. (a -> b) -> a -> b
$ Usage -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Usage
usage
    closeOver (VName
k, ParamType Liftedness
l SrcLoc
loc) =
      Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (Either TypeParam VName)
 -> m (Maybe (Either TypeParam VName)))
-> Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall a b. (a -> b) -> a -> b
$ Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a. a -> Maybe a
Just (Either TypeParam VName -> Maybe (Either TypeParam VName))
-> Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a b. (a -> b) -> a -> b
$ TypeParam -> Either TypeParam VName
forall a b. a -> Either a b
Left (TypeParam -> Either TypeParam VName)
-> TypeParam -> Either TypeParam VName
forall a b. (a -> b) -> a -> b
$ Liftedness -> VName -> SrcLoc -> TypeParam
forall vn. Liftedness -> vn -> SrcLoc -> TypeParamBase vn
TypeParamType Liftedness
l VName
k SrcLoc
loc
    closeOver (VName
k, Size Maybe (DimDecl VName)
Nothing Usage
usage) =
      Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (Either TypeParam VName)
 -> m (Maybe (Either TypeParam VName)))
-> Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall a b. (a -> b) -> a -> b
$ Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a. a -> Maybe a
Just (Either TypeParam VName -> Maybe (Either TypeParam VName))
-> Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a b. (a -> b) -> a -> b
$ TypeParam -> Either TypeParam VName
forall a b. a -> Either a b
Left (TypeParam -> Either TypeParam VName)
-> TypeParam -> Either TypeParam VName
forall a b. (a -> b) -> a -> b
$ VName -> SrcLoc -> TypeParam
forall vn. vn -> SrcLoc -> TypeParamBase vn
TypeParamDim VName
k (SrcLoc -> TypeParam) -> SrcLoc -> TypeParam
forall a b. (a -> b) -> a -> b
$ Usage -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Usage
usage
    closeOver (VName
k, UnknowableSize SrcLoc
_ RigidSource
_)
      | VName
k VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
param_sizes = do
        Notes
notes <- SrcLoc -> DimDecl VName -> m Notes
forall a (m :: * -> *).
(Located a, MonadUnify m) =>
a -> DimDecl VName -> m Notes
dimNotes SrcLoc
defloc (DimDecl VName -> m Notes) -> DimDecl VName -> m Notes
forall a b. (a -> b) -> a -> b
$ QualName VName -> DimDecl VName
forall vn. QualName vn -> DimDecl vn
NamedDim (QualName VName -> DimDecl VName)
-> QualName VName -> DimDecl VName
forall a b. (a -> b) -> a -> b
$ VName -> QualName VName
forall v. v -> QualName v
qualName VName
k
        SrcLoc -> Notes -> Doc -> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError SrcLoc
defloc Notes
notes (Doc -> m (Maybe (Either TypeParam VName)))
-> Doc -> m (Maybe (Either TypeParam VName))
forall a b. (a -> b) -> a -> b
$
          Doc
"Unknowable size" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
k)
            Doc -> Doc -> Doc
<+> Doc
"imposes constraint on type of"
            Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (Name -> Doc
forall v. IsName v => v -> Doc
pprName Name
defname)
            Doc -> Doc -> Doc
forall a. Semigroup a => a -> a -> a
<> Doc
", which is inferred as:"
            Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
t)
      | VName
k VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
produced_sizes =
        Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe (Either TypeParam VName)
 -> m (Maybe (Either TypeParam VName)))
-> Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall a b. (a -> b) -> a -> b
$ Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a. a -> Maybe a
Just (Either TypeParam VName -> Maybe (Either TypeParam VName))
-> Either TypeParam VName -> Maybe (Either TypeParam VName)
forall a b. (a -> b) -> a -> b
$ VName -> Either TypeParam VName
forall a b. b -> Either a b
Right VName
k
    closeOver (VName
_, Constraint
_) =
      Maybe (Either TypeParam VName)
-> m (Maybe (Either TypeParam VName))
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe (Either TypeParam VName)
forall a. Maybe a
Nothing

letGeneralise ::
  Name ->
  SrcLoc ->
  [TypeParam] ->
  [Pat] ->
  StructType ->
  TermTypeM ([TypeParam], [Pat], StructType, [VName])
letGeneralise :: Name
-> SrcLoc
-> [TypeParam]
-> [Pat]
-> StructType
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
letGeneralise Name
defname SrcLoc
defloc [TypeParam]
tparams [Pat]
params StructType
rettype =
  Checking
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (Name -> Checking
CheckingLetGeneralise Name
defname) (TermTypeM ([TypeParam], [Pat], StructType, [VName])
 -> TermTypeM ([TypeParam], [Pat], StructType, [VName]))
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
forall a b. (a -> b) -> a -> b
$ do
    Constraints
now_substs <- TermTypeM Constraints
forall (m :: * -> *). MonadUnify m => m Constraints
getConstraints

    -- Candidates for let-generalisation are those type variables that
    --
    -- (1) were not known before we checked this function, and
    --
    -- (2) are not used in the (new) definition of any type variables
    -- known before we checked this function.
    --
    -- (3) are not referenced from an overloaded type (for example,
    -- are the element types of an incompletely resolved record type).
    -- This is a bit more restrictive than I'd like, and SML for
    -- example does not have this restriction.
    --
    -- Criteria (1) and (2) is implemented by looking at the binding
    -- level of the type variables.
    let keep_type_vars :: Names
keep_type_vars = Constraints -> Names
overloadedTypeVars Constraints
now_substs

    Int
cur_lvl <- TermTypeM Int
forall (m :: * -> *). MonadUnify m => m Int
curLevel
    let candidate :: VName -> (Int, b) -> Bool
candidate VName
k (Int
lvl, b
_) = (VName
k VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.notMember` Names
keep_type_vars) Bool -> Bool -> Bool
&& Int
lvl Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
cur_lvl
        new_substs :: Constraints
new_substs = (VName -> (Int, Constraint) -> Bool) -> Constraints -> Constraints
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey VName -> (Int, Constraint) -> Bool
forall b. VName -> (Int, b) -> Bool
candidate Constraints
now_substs

    ([TypeParam]
tparams', StructType
rettype', [VName]
retext) <-
      Name
-> SrcLoc
-> [TypeParam]
-> [StructType]
-> StructType
-> Constraints
-> TermTypeM ([TypeParam], StructType, [VName])
closeOverTypes
        Name
defname
        SrcLoc
defloc
        [TypeParam]
tparams
        ((Pat -> StructType) -> [Pat] -> [StructType]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> StructType
patternStructType [Pat]
params)
        StructType
rettype
        Constraints
new_substs

    StructType
rettype'' <- StructType -> TermTypeM StructType
forall e. ASTMappable e => e -> TermTypeM e
updateTypes StructType
rettype'

    let used_sizes :: Names
used_sizes =
          (StructType -> Names) -> [StructType] -> Names
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap StructType -> Names
forall als. TypeBase (DimDecl VName) als -> Names
typeDimNames ([StructType] -> Names) -> [StructType] -> Names
forall a b. (a -> b) -> a -> b
$
            StructType
rettype'' StructType -> [StructType] -> [StructType]
forall a. a -> [a] -> [a]
: (Pat -> StructType) -> [Pat] -> [StructType]
forall a b. (a -> b) -> [a] -> [b]
map Pat -> StructType
patternStructType [Pat]
params
    case (TypeParam -> Bool) -> [TypeParam] -> [TypeParam]
forall a. (a -> Bool) -> [a] -> [a]
filter ((VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.notMember` Names
used_sizes) (VName -> Bool) -> (TypeParam -> VName) -> TypeParam -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName) ([TypeParam] -> [TypeParam]) -> [TypeParam] -> [TypeParam]
forall a b. (a -> b) -> a -> b
$
      (TypeParam -> Bool) -> [TypeParam] -> [TypeParam]
forall a. (a -> Bool) -> [a] -> [a]
filter TypeParam -> Bool
forall vn. TypeParamBase vn -> Bool
isSizeParam [TypeParam]
tparams' of
      [] -> () -> TermTypeM ()
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
      TypeParam
tp : [TypeParam]
_ -> SizeBinder VName -> TermTypeM ()
forall (m :: * -> *) a.
MonadTypeChecker m =>
SizeBinder VName -> m a
unusedSize (SizeBinder VName -> TermTypeM ())
-> SizeBinder VName -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ VName -> SrcLoc -> SizeBinder VName
forall vn. vn -> SrcLoc -> SizeBinder vn
SizeBinder (TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName TypeParam
tp) (TypeParam -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf TypeParam
tp)

    -- We keep those type variables that were not closed over by
    -- let-generalisation.
    (Constraints -> Constraints) -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
(Constraints -> Constraints) -> m ()
modifyConstraints ((Constraints -> Constraints) -> TermTypeM ())
-> (Constraints -> Constraints) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ (VName -> (Int, Constraint) -> Bool) -> Constraints -> Constraints
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey ((VName -> (Int, Constraint) -> Bool)
 -> Constraints -> Constraints)
-> (VName -> (Int, Constraint) -> Bool)
-> Constraints
-> Constraints
forall a b. (a -> b) -> a -> b
$ \VName
k (Int, Constraint)
_ -> VName
k VName -> [VName] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`notElem` (TypeParam -> VName) -> [TypeParam] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map TypeParam -> VName
forall vn. TypeParamBase vn -> vn
typeParamName [TypeParam]
tparams'

    ([TypeParam], [Pat], StructType, [VName])
-> TermTypeM ([TypeParam], [Pat], StructType, [VName])
forall (m :: * -> *) a. Monad m => a -> m a
return ([TypeParam]
tparams', [Pat]
params, StructType
rettype'', [VName]
retext)

checkFunBody ::
  [Pat] ->
  UncheckedExp ->
  Maybe StructType ->
  SrcLoc ->
  TermTypeM Exp
checkFunBody :: [Pat]
-> UncheckedExp -> Maybe StructType -> SrcLoc -> TermTypeM Exp
checkFunBody [Pat]
params UncheckedExp
body Maybe StructType
maybe_rettype SrcLoc
loc = do
  Exp
body' <- TermTypeM Exp -> TermTypeM Exp
forall b. TermTypeM b -> TermTypeM b
noSizeEscape (TermTypeM Exp -> TermTypeM Exp) -> TermTypeM Exp -> TermTypeM Exp
forall a b. (a -> b) -> a -> b
$ UncheckedExp -> TermTypeM Exp
checkExp UncheckedExp
body

  -- Unify body return type with return annotation, if one exists.
  case Maybe StructType
maybe_rettype of
    Just StructType
rettype -> do
      (StructType
rettype_withdims, [VName]
_) <- SrcLoc
-> String
-> Rigidity
-> StructType
-> TermTypeM (StructType, [VName])
forall (m :: * -> *) als.
MonadUnify m =>
SrcLoc
-> String
-> Rigidity
-> TypeBase (DimDecl VName) als
-> m (TypeBase (DimDecl VName) als, [VName])
instantiateEmptyArrayDims SrcLoc
loc String
"impl" Rigidity
Nonrigid StructType
rettype

      PatType
body_t <- Exp -> TermTypeM PatType
expTypeFully Exp
body'
      -- We need to turn any sizes provided by "hidden" parameter
      -- names into existential sizes instead.
      let hidden :: Names
hidden = [Pat] -> Names
hiddenParamNames [Pat]
params
      (PatType
body_t', [VName]
_) <-
        SrcLoc
-> Map VName Ident -> PatType -> TermTypeM (PatType, [VName])
unscopeType
          SrcLoc
loc
          ( (VName -> Ident -> Bool) -> Map VName Ident -> Map VName Ident
forall k a. (k -> a -> Bool) -> Map k a -> Map k a
M.filterWithKey (Bool -> Ident -> Bool
forall a b. a -> b -> a
const (Bool -> Ident -> Bool)
-> (VName -> Bool) -> VName -> Ident -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` Names
hidden)) (Map VName Ident -> Map VName Ident)
-> Map VName Ident -> Map VName Ident
forall a b. (a -> b) -> a -> b
$
              (Pat -> Map VName Ident) -> [Pat] -> Map VName Ident
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap Pat -> Map VName Ident
forall (f :: * -> *).
Functor f =>
PatBase f VName -> Map VName (IdentBase f VName)
patternMap [Pat]
params
          )
          PatType
body_t

      let usage :: Usage
usage = SrcLoc -> String -> Usage
mkUsage (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
body) String
"return type annotation"
      Checking -> TermTypeM () -> TermTypeM ()
forall a. Checking -> TermTypeM a -> TermTypeM a
onFailure (StructType -> StructType -> Checking
CheckingReturn StructType
rettype (PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
body_t')) (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        Usage -> StructType -> StructType -> TermTypeM ()
forall (m :: * -> *).
MonadUnify m =>
Usage -> StructType -> StructType -> m ()
expect Usage
usage StructType
rettype_withdims (StructType -> TermTypeM ()) -> StructType -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ PatType -> StructType
forall dim as. TypeBase dim as -> TypeBase dim ()
toStruct PatType
body_t'

      -- We also have to make sure that uniqueness matches.  This is done
      -- explicitly, because uniqueness is ignored by unification.
      StructType
rettype' <- StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully StructType
rettype
      StructType
body_t'' <- StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully StructType
rettype -- Substs may have changed.
      Bool -> TermTypeM () -> TermTypeM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (StructType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural StructType
body_t'' TypeBase () () -> TypeBase () () -> Bool
`subtypeOf` StructType -> TypeBase () ()
forall dim as. TypeBase dim as -> TypeBase () ()
toStructural StructType
rettype') (TermTypeM () -> TermTypeM ()) -> TermTypeM () -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
        SrcLoc -> Notes -> Doc -> TermTypeM ()
forall (m :: * -> *) loc a.
(MonadTypeChecker m, Located loc) =>
loc -> Notes -> Doc -> m a
typeError (UncheckedExp -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf UncheckedExp
body) Notes
forall a. Monoid a => a
mempty (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$
          Doc
"Body type" Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
body_t'')
            Doc -> Doc -> Doc
</> Doc
"is not a subtype of annotated type"
            Doc -> Doc -> Doc
</> Int -> Doc -> Doc
indent Int
2 (StructType -> Doc
forall a. Pretty a => a -> Doc
ppr StructType
rettype')
    Maybe StructType
Nothing -> () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

  Exp -> TermTypeM Exp
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
body'

--- Consumption

occur :: Occurences -> TermTypeM ()
occur :: [Occurence] -> TermTypeM ()
occur [Occurence]
occs = (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateOccs :: [Occurence]
stateOccs = TermTypeState -> [Occurence]
stateOccs TermTypeState
s [Occurence] -> [Occurence] -> [Occurence]
forall a. Semigroup a => a -> a -> a
<> [Occurence]
occs}

-- | Proclaim that we have made read-only use of the given variable.
observe :: Ident -> TermTypeM ()
observe :: Ident -> TermTypeM ()
observe (Ident VName
nm (Info PatType
t) SrcLoc
loc) =
  let als :: Aliasing
als = VName -> Alias
AliasBound VName
nm Alias -> Aliasing -> Aliasing
forall a. Ord a => a -> Set a -> Set a
`S.insert` PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
t
   in [Occurence] -> TermTypeM ()
occur [Aliasing -> SrcLoc -> Occurence
observation Aliasing
als SrcLoc
loc]

describeVar :: SrcLoc -> VName -> TermTypeM Doc
describeVar :: SrcLoc -> VName -> TermTypeM Doc
describeVar SrcLoc
loc VName
v =
  (TermTypeState -> Doc) -> TermTypeM Doc
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets ((TermTypeState -> Doc) -> TermTypeM Doc)
-> (TermTypeState -> Doc) -> TermTypeM Doc
forall a b. (a -> b) -> a -> b
$
    Doc -> (NameReason -> Doc) -> Maybe NameReason -> Doc
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (Doc
"variable" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName VName
v)) (SrcLoc -> NameReason -> Doc
nameReason SrcLoc
loc)
      (Maybe NameReason -> Doc)
-> (TermTypeState -> Maybe NameReason) -> TermTypeState -> Doc
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> Map VName NameReason -> Maybe NameReason
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
v
      (Map VName NameReason -> Maybe NameReason)
-> (TermTypeState -> Map VName NameReason)
-> TermTypeState
-> Maybe NameReason
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermTypeState -> Map VName NameReason
stateNames

checkIfConsumable :: SrcLoc -> Aliasing -> TermTypeM ()
checkIfConsumable :: SrcLoc -> Aliasing -> TermTypeM ()
checkIfConsumable SrcLoc
loc Aliasing
als = do
  Map VName ValBinding
vtable <- (TermEnv -> Map VName ValBinding)
-> TermTypeM (Map VName ValBinding)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((TermEnv -> Map VName ValBinding)
 -> TermTypeM (Map VName ValBinding))
-> (TermEnv -> Map VName ValBinding)
-> TermTypeM (Map VName ValBinding)
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable (TermScope -> Map VName ValBinding)
-> (TermEnv -> TermScope) -> TermEnv -> Map VName ValBinding
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TermEnv -> TermScope
termScope
  let consumable :: VName -> Bool
consumable VName
v = case VName -> Map VName ValBinding -> Maybe ValBinding
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup VName
v Map VName ValBinding
vtable of
        Just (BoundV Locality
Local [TypeParam]
_ PatType
t)
          | PatType -> Int
forall dim as. TypeBase dim as -> Int
arrayRank PatType
t Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0 -> PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
t
          | Scalar TypeVar {} <- PatType
t -> PatType -> Bool
forall shape as. TypeBase shape as -> Bool
unique PatType
t
          | Scalar Arrow {} <- PatType
t -> Bool
False
          | Bool
otherwise -> Bool
True
        Just (BoundV Locality
Global [TypeParam]
_ PatType
_) -> Bool
False
        Maybe ValBinding
_ -> Bool
True
  -- The sort ensures that AliasBound vars are shown before AliasFree.
  case (Alias -> VName) -> [Alias] -> [VName]
forall a b. (a -> b) -> [a] -> [b]
map Alias -> VName
aliasVar ([Alias] -> [VName]) -> [Alias] -> [VName]
forall a b. (a -> b) -> a -> b
$ [Alias] -> [Alias]
forall a. Ord a => [a] -> [a]
sort ([Alias] -> [Alias]) -> [Alias] -> [Alias]
forall a b. (a -> b) -> a -> b
$ (Alias -> Bool) -> [Alias] -> [Alias]
forall a. (a -> Bool) -> [a] -> [a]
filter (Bool -> Bool
not (Bool -> Bool) -> (Alias -> Bool) -> Alias -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. VName -> Bool
consumable (VName -> Bool) -> (Alias -> VName) -> Alias -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Alias -> VName
aliasVar) ([Alias] -> [Alias]) -> [Alias] -> [Alias]
forall a b. (a -> b) -> a -> b
$ Aliasing -> [Alias]
forall a. Set a -> [a]
S.toList Aliasing
als of
    VName
v : [VName]
_ -> SrcLoc -> Doc -> TermTypeM ()
forall (m :: * -> *) b. MonadTypeChecker m => SrcLoc -> Doc -> m b
notConsumable SrcLoc
loc (Doc -> TermTypeM ()) -> TermTypeM Doc -> TermTypeM ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< SrcLoc -> VName -> TermTypeM Doc
describeVar SrcLoc
loc VName
v
    [] -> () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

-- | Proclaim that we have written to the given variable.
consume :: SrcLoc -> Aliasing -> TermTypeM ()
consume :: SrcLoc -> Aliasing -> TermTypeM ()
consume SrcLoc
loc Aliasing
als = do
  SrcLoc -> Aliasing -> TermTypeM ()
checkIfConsumable SrcLoc
loc Aliasing
als
  [Occurence] -> TermTypeM ()
occur [Aliasing -> SrcLoc -> Occurence
consumption Aliasing
als SrcLoc
loc]

-- | Proclaim that we have written to the given variable, and mark
-- accesses to it and all of its aliases as invalid inside the given
-- computation.
consuming :: Ident -> TermTypeM a -> TermTypeM a
consuming :: Ident -> TermTypeM a -> TermTypeM a
consuming (Ident VName
name (Info PatType
t) SrcLoc
loc) TermTypeM a
m = do
  PatType
t' <- PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully PatType
t
  SrcLoc -> Aliasing -> TermTypeM ()
consume SrcLoc
loc (Aliasing -> TermTypeM ()) -> Aliasing -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ VName -> Alias
AliasBound VName
name Alias -> Aliasing -> Aliasing
forall a. Ord a => a -> Set a -> Set a
`S.insert` PatType -> Aliasing
forall as shape. Monoid as => TypeBase shape as -> as
aliases PatType
t'
  (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope TermScope -> TermScope
consume' TermTypeM a
m
  where
    consume' :: TermScope -> TermScope
consume' TermScope
scope =
      TermScope
scope {scopeVtable :: Map VName ValBinding
scopeVtable = VName -> ValBinding -> Map VName ValBinding -> Map VName ValBinding
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert VName
name (SrcLoc -> ValBinding
WasConsumed SrcLoc
loc) (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable TermScope
scope}

collectOccurences :: TermTypeM a -> TermTypeM (a, Occurences)
collectOccurences :: TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences TermTypeM a
m = do
  [Occurence]
old <- (TermTypeState -> [Occurence]) -> TermTypeM [Occurence]
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets TermTypeState -> [Occurence]
stateOccs
  (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateOccs :: [Occurence]
stateOccs = [Occurence]
forall a. Monoid a => a
mempty}
  a
x <- TermTypeM a
m
  [Occurence]
new <- (TermTypeState -> [Occurence]) -> TermTypeM [Occurence]
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets TermTypeState -> [Occurence]
stateOccs
  (TermTypeState -> TermTypeState) -> TermTypeM ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify ((TermTypeState -> TermTypeState) -> TermTypeM ())
-> (TermTypeState -> TermTypeState) -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ \TermTypeState
s -> TermTypeState
s {stateOccs :: [Occurence]
stateOccs = [Occurence]
old}
  (a, [Occurence]) -> TermTypeM (a, [Occurence])
forall (f :: * -> *) a. Applicative f => a -> f a
pure (a
x, [Occurence]
new)

tapOccurences :: TermTypeM a -> TermTypeM (a, Occurences)
tapOccurences :: TermTypeM a -> TermTypeM (a, [Occurence])
tapOccurences TermTypeM a
m = do
  (a
x, [Occurence]
occs) <- TermTypeM a -> TermTypeM (a, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences TermTypeM a
m
  [Occurence] -> TermTypeM ()
occur [Occurence]
occs
  (a, [Occurence]) -> TermTypeM (a, [Occurence])
forall (f :: * -> *) a. Applicative f => a -> f a
pure (a
x, [Occurence]
occs)

removeSeminullOccurences :: TermTypeM a -> TermTypeM a
removeSeminullOccurences :: TermTypeM a -> TermTypeM a
removeSeminullOccurences TermTypeM a
m = do
  (a
x, [Occurence]
occs) <- TermTypeM a -> TermTypeM (a, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences TermTypeM a
m
  [Occurence] -> TermTypeM ()
occur ([Occurence] -> TermTypeM ()) -> [Occurence] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ (Occurence -> Bool) -> [Occurence] -> [Occurence]
forall a. (a -> Bool) -> [a] -> [a]
filter (Bool -> Bool
not (Bool -> Bool) -> (Occurence -> Bool) -> Occurence -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Occurence -> Bool
seminullOccurence) [Occurence]
occs
  a -> TermTypeM a
forall (f :: * -> *) a. Applicative f => a -> f a
pure a
x

checkIfUsed :: Occurences -> Ident -> TermTypeM ()
checkIfUsed :: [Occurence] -> Ident -> TermTypeM ()
checkIfUsed [Occurence]
occs Ident
v
  | Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
v VName -> Names -> Bool
forall a. Ord a => a -> Set a -> Bool
`S.member` [Occurence] -> Names
allOccuring [Occurence]
occs,
    Bool -> Bool
not (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$ String
"_" String -> String -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf` VName -> String
forall v. IsName v => v -> String
prettyName (Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
v) =
    SrcLoc -> Doc -> TermTypeM ()
forall (m :: * -> *) loc.
(MonadTypeChecker m, Located loc) =>
loc -> Doc -> m ()
warn (Ident -> SrcLoc
forall a. Located a => a -> SrcLoc
srclocOf Ident
v) (Doc -> TermTypeM ()) -> Doc -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ Doc
"Unused variable" Doc -> Doc -> Doc
<+> Doc -> Doc
pquote (VName -> Doc
forall v. IsName v => v -> Doc
pprName (VName -> Doc) -> VName -> Doc
forall a b. (a -> b) -> a -> b
$ Ident -> VName
forall (f :: * -> *) vn. IdentBase f vn -> vn
identName Ident
v) Doc -> Doc -> Doc
<+> Doc
"."
  | Bool
otherwise =
    () -> TermTypeM ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

alternative :: TermTypeM a -> TermTypeM b -> TermTypeM (a, b)
alternative :: TermTypeM a -> TermTypeM b -> TermTypeM (a, b)
alternative TermTypeM a
m1 TermTypeM b
m2 = do
  (a
x, [Occurence]
occurs1) <- TermTypeM a -> TermTypeM (a, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences (TermTypeM a -> TermTypeM (a, [Occurence]))
-> TermTypeM a -> TermTypeM (a, [Occurence])
forall a b. (a -> b) -> a -> b
$ TermTypeM a -> TermTypeM a
forall b. TermTypeM b -> TermTypeM b
noSizeEscape TermTypeM a
m1
  (b
y, [Occurence]
occurs2) <- TermTypeM b -> TermTypeM (b, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences (TermTypeM b -> TermTypeM (b, [Occurence]))
-> TermTypeM b -> TermTypeM (b, [Occurence])
forall a b. (a -> b) -> a -> b
$ TermTypeM b -> TermTypeM b
forall b. TermTypeM b -> TermTypeM b
noSizeEscape TermTypeM b
m2
  [Occurence] -> TermTypeM ()
checkOccurences [Occurence]
occurs1
  [Occurence] -> TermTypeM ()
checkOccurences [Occurence]
occurs2
  [Occurence] -> TermTypeM ()
occur ([Occurence] -> TermTypeM ()) -> [Occurence] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ [Occurence]
occurs1 [Occurence] -> [Occurence] -> [Occurence]
`altOccurences` [Occurence]
occurs2
  (a, b) -> TermTypeM (a, b)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (a
x, b
y)

-- | Enter a context where nothing outside can be consumed (i.e. the
-- body of a function definition).
noUnique :: TermTypeM a -> TermTypeM a
noUnique :: TermTypeM a -> TermTypeM a
noUnique TermTypeM a
m = do
  (a
x, [Occurence]
occs) <- TermTypeM a -> TermTypeM (a, [Occurence])
forall a. TermTypeM a -> TermTypeM (a, [Occurence])
collectOccurences (TermTypeM a -> TermTypeM (a, [Occurence]))
-> TermTypeM a -> TermTypeM (a, [Occurence])
forall a b. (a -> b) -> a -> b
$ (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope TermScope -> TermScope
f TermTypeM a
m
  [Occurence] -> TermTypeM ()
checkOccurences [Occurence]
occs
  [Occurence] -> TermTypeM ()
occur ([Occurence] -> TermTypeM ()) -> [Occurence] -> TermTypeM ()
forall a b. (a -> b) -> a -> b
$ ([Occurence], [Occurence]) -> [Occurence]
forall a b. (a, b) -> a
fst (([Occurence], [Occurence]) -> [Occurence])
-> ([Occurence], [Occurence]) -> [Occurence]
forall a b. (a -> b) -> a -> b
$ [Occurence] -> ([Occurence], [Occurence])
split [Occurence]
occs
  a -> TermTypeM a
forall (f :: * -> *) a. Applicative f => a -> f a
pure a
x
  where
    f :: TermScope -> TermScope
f TermScope
scope = TermScope
scope {scopeVtable :: Map VName ValBinding
scopeVtable = (ValBinding -> ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b k. (a -> b) -> Map k a -> Map k b
M.map ValBinding -> ValBinding
set (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable TermScope
scope}

    set :: ValBinding -> ValBinding
set (BoundV Locality
l [TypeParam]
tparams PatType
t) = Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
l [TypeParam]
tparams (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$ PatType
t PatType -> Uniqueness -> PatType
forall dim as. TypeBase dim as -> Uniqueness -> TypeBase dim as
`setUniqueness` Uniqueness
Nonunique
    set (OverloadedF [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rt) = [PrimType] -> [Maybe PrimType] -> Maybe PrimType -> ValBinding
OverloadedF [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rt
    set ValBinding
EqualityF = ValBinding
EqualityF
    set (WasConsumed SrcLoc
loc) = SrcLoc -> ValBinding
WasConsumed SrcLoc
loc

    split :: [Occurence] -> ([Occurence], [Occurence])
split = [(Occurence, Occurence)] -> ([Occurence], [Occurence])
forall a b. [(a, b)] -> ([a], [b])
unzip ([(Occurence, Occurence)] -> ([Occurence], [Occurence]))
-> ([Occurence] -> [(Occurence, Occurence)])
-> [Occurence]
-> ([Occurence], [Occurence])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Occurence -> (Occurence, Occurence))
-> [Occurence] -> [(Occurence, Occurence)]
forall a b. (a -> b) -> [a] -> [b]
map (\Occurence
occ -> (Occurence
occ {consumed :: Maybe Names
consumed = Maybe Names
forall a. Monoid a => a
mempty}, Occurence
occ {observed :: Names
observed = Names
forall a. Monoid a => a
mempty}))

onlySelfAliasing :: TermTypeM a -> TermTypeM a
onlySelfAliasing :: TermTypeM a -> TermTypeM a
onlySelfAliasing = (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
forall a. (TermScope -> TermScope) -> TermTypeM a -> TermTypeM a
localScope (\TermScope
scope -> TermScope
scope {scopeVtable :: Map VName ValBinding
scopeVtable = (VName -> ValBinding -> ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall k a b. (k -> a -> b) -> Map k a -> Map k b
M.mapWithKey VName -> ValBinding -> ValBinding
set (Map VName ValBinding -> Map VName ValBinding)
-> Map VName ValBinding -> Map VName ValBinding
forall a b. (a -> b) -> a -> b
$ TermScope -> Map VName ValBinding
scopeVtable TermScope
scope})
  where
    set :: VName -> ValBinding -> ValBinding
set VName
k (BoundV Locality
l [TypeParam]
tparams PatType
t) =
      Locality -> [TypeParam] -> PatType -> ValBinding
BoundV Locality
l [TypeParam]
tparams (PatType -> ValBinding) -> PatType -> ValBinding
forall a b. (a -> b) -> a -> b
$
        PatType
t PatType -> (Aliasing -> Aliasing) -> PatType
forall dim asf ast.
TypeBase dim asf -> (asf -> ast) -> TypeBase dim ast
`addAliases` Aliasing -> Aliasing -> Aliasing
forall a. Ord a => Set a -> Set a -> Set a
S.intersection (Alias -> Aliasing
forall a. a -> Set a
S.singleton (VName -> Alias
AliasBound VName
k))
    set VName
_ (OverloadedF [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rt) = [PrimType] -> [Maybe PrimType] -> Maybe PrimType -> ValBinding
OverloadedF [PrimType]
ts [Maybe PrimType]
pts Maybe PrimType
rt
    set VName
_ ValBinding
EqualityF = ValBinding
EqualityF
    set VName
_ (WasConsumed SrcLoc
loc) = SrcLoc -> ValBinding
WasConsumed SrcLoc
loc

arrayOfM ::
  (Pretty (ShapeDecl dim), Monoid as) =>
  SrcLoc ->
  TypeBase dim as ->
  ShapeDecl dim ->
  Uniqueness ->
  TermTypeM (TypeBase dim as)
arrayOfM :: SrcLoc
-> TypeBase dim as
-> ShapeDecl dim
-> Uniqueness
-> TermTypeM (TypeBase dim as)
arrayOfM SrcLoc
loc TypeBase dim as
t ShapeDecl dim
shape Uniqueness
u = do
  Usage -> String -> TypeBase dim as -> TermTypeM ()
forall (m :: * -> *) dim as.
(MonadUnify m, Pretty (ShapeDecl dim), Monoid as) =>
Usage -> String -> TypeBase dim as -> m ()
arrayElemType (SrcLoc -> String -> Usage
mkUsage SrcLoc
loc String
"use as array element") String
"type used in array" TypeBase dim as
t
  TypeBase dim as -> TermTypeM (TypeBase dim as)
forall (m :: * -> *) a. Monad m => a -> m a
return (TypeBase dim as -> TermTypeM (TypeBase dim as))
-> TypeBase dim as -> TermTypeM (TypeBase dim as)
forall a b. (a -> b) -> a -> b
$ TypeBase dim as -> ShapeDecl dim -> Uniqueness -> TypeBase dim as
forall as dim.
Monoid as =>
TypeBase dim as -> ShapeDecl dim -> Uniqueness -> TypeBase dim as
arrayOf TypeBase dim as
t ShapeDecl dim
shape Uniqueness
u

updateTypes :: ASTMappable e => e -> TermTypeM e
updateTypes :: e -> TermTypeM e
updateTypes = ASTMapper TermTypeM -> e -> TermTypeM e
forall x (m :: * -> *).
(ASTMappable x, Monad m) =>
ASTMapper m -> x -> m x
astMap ASTMapper TermTypeM
tv
  where
    tv :: ASTMapper TermTypeM
tv =
      ASTMapper :: forall (m :: * -> *).
(Exp -> m Exp)
-> (VName -> m VName)
-> (QualName VName -> m (QualName VName))
-> (StructType -> m StructType)
-> (PatType -> m PatType)
-> ASTMapper m
ASTMapper
        { mapOnExp :: Exp -> TermTypeM Exp
mapOnExp = ASTMapper TermTypeM -> Exp -> TermTypeM Exp
forall x (m :: * -> *).
(ASTMappable x, Monad m) =>
ASTMapper m -> x -> m x
astMap ASTMapper TermTypeM
tv,
          mapOnName :: VName -> TermTypeM VName
mapOnName = VName -> TermTypeM VName
forall (f :: * -> *) a. Applicative f => a -> f a
pure,
          mapOnQualName :: QualName VName -> TermTypeM (QualName VName)
mapOnQualName = QualName VName -> TermTypeM (QualName VName)
forall (f :: * -> *) a. Applicative f => a -> f a
pure,
          mapOnStructType :: StructType -> TermTypeM StructType
mapOnStructType = StructType -> TermTypeM StructType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully,
          mapOnPatType :: PatType -> TermTypeM PatType
mapOnPatType = PatType -> TermTypeM PatType
forall a (m :: * -> *). (Substitutable a, MonadUnify m) => a -> m a
normTypeFully
        }