Safe Haskell | None |
---|---|
Language | Haskell2010 |
CoreSyn holds all the main data types for use by for the Glasgow Haskell Compiler midsection
- data Expr b
- type Alt b = (AltCon, [b], Expr b)
- data Bind b
- data AltCon
- type Arg b = Expr b
- data Tickish id
- = ProfNote {
- profNoteCC :: CostCentre
- profNoteCount :: !Bool
- profNoteScope :: !Bool
- | HpcTick {
- tickModule :: Module
- tickId :: !Int
- | Breakpoint {
- breakpointId :: !Int
- breakpointFVs :: [id]
- | SourceNote { }
- = ProfNote {
- data TickishScoping
- data TickishPlacement
- type CoreProgram = [CoreBind]
- type CoreExpr = Expr CoreBndr
- type CoreAlt = Alt CoreBndr
- type CoreBind = Bind CoreBndr
- type CoreArg = Arg CoreBndr
- type CoreBndr = Var
- type TaggedExpr t = Expr (TaggedBndr t)
- type TaggedAlt t = Alt (TaggedBndr t)
- type TaggedBind t = Bind (TaggedBndr t)
- type TaggedArg t = Arg (TaggedBndr t)
- data TaggedBndr t = TB CoreBndr t
- deTagExpr :: TaggedExpr t -> CoreExpr
- type InId = Id
- type InBind = CoreBind
- type InExpr = CoreExpr
- type InAlt = CoreAlt
- type InArg = CoreArg
- type InType = Type
- type InKind = Kind
- type InBndr = CoreBndr
- type InVar = Var
- type InCoercion = Coercion
- type InTyVar = TyVar
- type InCoVar = CoVar
- type OutId = Id
- type OutBind = CoreBind
- type OutExpr = CoreExpr
- type OutAlt = CoreAlt
- type OutArg = CoreArg
- type OutType = Type
- type OutKind = Kind
- type OutBndr = CoreBndr
- type OutVar = Var
- type OutCoercion = Coercion
- type OutTyVar = TyVar
- type OutCoVar = CoVar
- mkLet :: Bind b -> Expr b -> Expr b
- mkLets :: [Bind b] -> Expr b -> Expr b
- mkLams :: [b] -> Expr b -> Expr b
- mkApps :: Expr b -> [Arg b] -> Expr b
- mkTyApps :: Expr b -> [Type] -> Expr b
- mkCoApps :: Expr b -> [Coercion] -> Expr b
- mkVarApps :: Expr b -> [Var] -> Expr b
- mkTyArg :: Type -> Expr b
- mkIntLit :: DynFlags -> Integer -> Expr b
- mkIntLitInt :: DynFlags -> Int -> Expr b
- mkWordLit :: DynFlags -> Integer -> Expr b
- mkWordLitWord :: DynFlags -> Word -> Expr b
- mkWord64LitWord64 :: Word64 -> Expr b
- mkInt64LitInt64 :: Int64 -> Expr b
- mkCharLit :: Char -> Expr b
- mkStringLit :: String -> Expr b
- mkFloatLit :: Rational -> Expr b
- mkFloatLitFloat :: Float -> Expr b
- mkDoubleLit :: Rational -> Expr b
- mkDoubleLitDouble :: Double -> Expr b
- mkConApp :: DataCon -> [Arg b] -> Expr b
- mkConApp2 :: DataCon -> [Type] -> [Var] -> Expr b
- mkTyBind :: TyVar -> Type -> CoreBind
- mkCoBind :: CoVar -> Coercion -> CoreBind
- varToCoreExpr :: CoreBndr -> Expr b
- varsToCoreExprs :: [CoreBndr] -> [Expr b]
- isId :: Var -> Bool
- cmpAltCon :: AltCon -> AltCon -> Ordering
- cmpAlt :: (AltCon, a, b) -> (AltCon, a, b) -> Ordering
- ltAlt :: (AltCon, a, b) -> (AltCon, a, b) -> Bool
- bindersOf :: Bind b -> [b]
- bindersOfBinds :: [Bind b] -> [b]
- rhssOfBind :: Bind b -> [Expr b]
- rhssOfAlts :: [Alt b] -> [Expr b]
- collectBinders :: Expr b -> ([b], Expr b)
- collectTyBinders :: CoreExpr -> ([TyVar], CoreExpr)
- collectTyAndValBinders :: CoreExpr -> ([TyVar], [Id], CoreExpr)
- collectNBinders :: Int -> Expr b -> ([b], Expr b)
- collectArgs :: Expr b -> (Expr b, [Arg b])
- collectArgsTicks :: (Tickish Id -> Bool) -> Expr b -> (Expr b, [Arg b], [Tickish Id])
- flattenBinds :: [Bind b] -> [(b, Expr b)]
- exprToType :: CoreExpr -> Type
- exprToCoercion_maybe :: CoreExpr -> Maybe Coercion
- applyTypeToArg :: Type -> CoreExpr -> Type
- isValArg :: Expr b -> Bool
- isTypeArg :: Expr b -> Bool
- isTyCoArg :: Expr b -> Bool
- valArgCount :: [Arg b] -> Int
- valBndrCount :: [CoreBndr] -> Int
- isRuntimeArg :: CoreExpr -> Bool
- isRuntimeVar :: Var -> Bool
- tickishCounts :: Tickish id -> Bool
- tickishScoped :: Tickish id -> TickishScoping
- tickishScopesLike :: Tickish id -> TickishScoping -> Bool
- tickishFloatable :: Tickish id -> Bool
- tickishCanSplit :: Tickish id -> Bool
- mkNoCount :: Tickish id -> Tickish id
- mkNoScope :: Tickish id -> Tickish id
- tickishIsCode :: Tickish id -> Bool
- tickishPlace :: Tickish id -> TickishPlacement
- tickishContains :: Eq b => Tickish b -> Tickish b -> Bool
- data Unfolding
- = NoUnfolding
- | BootUnfolding
- | OtherCon [AltCon]
- | DFunUnfolding { }
- | CoreUnfolding { }
- data UnfoldingGuidance
- = UnfWhen {
- ug_arity :: Arity
- ug_unsat_ok :: Bool
- ug_boring_ok :: Bool
- | UnfIfGoodArgs { }
- | UnfNever
- = UnfWhen {
- data UnfoldingSource
- noUnfolding :: Unfolding
- bootUnfolding :: Unfolding
- evaldUnfolding :: Unfolding
- mkOtherCon :: [AltCon] -> Unfolding
- unSaturatedOk :: Bool
- needSaturated :: Bool
- boringCxtOk :: Bool
- boringCxtNotOk :: Bool
- unfoldingTemplate :: Unfolding -> CoreExpr
- expandUnfolding_maybe :: Unfolding -> Maybe CoreExpr
- maybeUnfoldingTemplate :: Unfolding -> Maybe CoreExpr
- otherCons :: Unfolding -> [AltCon]
- isValueUnfolding :: Unfolding -> Bool
- isEvaldUnfolding :: Unfolding -> Bool
- isCheapUnfolding :: Unfolding -> Bool
- isExpandableUnfolding :: Unfolding -> Bool
- isConLikeUnfolding :: Unfolding -> Bool
- isCompulsoryUnfolding :: Unfolding -> Bool
- isStableUnfolding :: Unfolding -> Bool
- isFragileUnfolding :: Unfolding -> Bool
- hasSomeUnfolding :: Unfolding -> Bool
- isBootUnfolding :: Unfolding -> Bool
- canUnfold :: Unfolding -> Bool
- neverUnfoldGuidance :: UnfoldingGuidance -> Bool
- isStableSource :: UnfoldingSource -> Bool
- type AnnExpr bndr annot = (annot, AnnExpr' bndr annot)
- data AnnExpr' bndr annot
- = AnnVar Id
- | AnnLit Literal
- | AnnLam bndr (AnnExpr bndr annot)
- | AnnApp (AnnExpr bndr annot) (AnnExpr bndr annot)
- | AnnCase (AnnExpr bndr annot) bndr Type [AnnAlt bndr annot]
- | AnnLet (AnnBind bndr annot) (AnnExpr bndr annot)
- | AnnCast (AnnExpr bndr annot) (annot, Coercion)
- | AnnTick (Tickish Id) (AnnExpr bndr annot)
- | AnnType Type
- | AnnCoercion Coercion
- data AnnBind bndr annot
- type AnnAlt bndr annot = (AltCon, [bndr], AnnExpr bndr annot)
- collectAnnArgs :: AnnExpr b a -> (AnnExpr b a, [AnnExpr b a])
- collectAnnArgsTicks :: (Tickish Var -> Bool) -> AnnExpr b a -> (AnnExpr b a, [AnnExpr b a], [Tickish Var])
- deAnnotate :: AnnExpr bndr annot -> Expr bndr
- deAnnotate' :: AnnExpr' bndr annot -> Expr bndr
- deAnnAlt :: AnnAlt bndr annot -> Alt bndr
- collectAnnBndrs :: AnnExpr bndr annot -> ([bndr], AnnExpr bndr annot)
- collectNAnnBndrs :: Int -> AnnExpr bndr annot -> ([bndr], AnnExpr bndr annot)
- data IsOrphan
- isOrphan :: IsOrphan -> Bool
- notOrphan :: IsOrphan -> Bool
- chooseOrphanAnchor :: NameSet -> IsOrphan
- data CoreRule
- = Rule { }
- | BuiltinRule { }
- type RuleBase = NameEnv [CoreRule]
- type RuleName = FastString
- type RuleFun = DynFlags -> InScopeEnv -> Id -> [CoreExpr] -> Maybe CoreExpr
- type IdUnfoldingFun = Id -> Unfolding
- type InScopeEnv = (InScopeSet, IdUnfoldingFun)
- data RuleEnv = RuleEnv {}
- mkRuleEnv :: RuleBase -> [Module] -> RuleEnv
- emptyRuleEnv :: RuleEnv
- ruleArity :: CoreRule -> Int
- ruleName :: CoreRule -> RuleName
- ruleIdName :: CoreRule -> Name
- ruleActivation :: CoreRule -> Activation
- setRuleIdName :: Name -> CoreRule -> CoreRule
- isBuiltinRule :: CoreRule -> Bool
- isLocalRule :: CoreRule -> Bool
- isAutoRule :: CoreRule -> Bool
- data CoreVect
Main data types
This is the data type that represents GHCs core intermediate language. Currently GHC uses System FC http://research.microsoft.com/~simonpj/papers/ext-f/ for this purpose, which is closely related to the simpler and better known System F http://en.wikipedia.org/wiki/System_F.
We get from Haskell source to this Core language in a number of stages:
- The source code is parsed into an abstract syntax tree, which is represented
by the data type
HsExpr
with the names beingRdrNames
- This syntax tree is renamed, which attaches a
Unique
to everyRdrName
(yielding aName
) to disambiguate identifiers which are lexically identical. For example, this program:
f x = let f x = x + 1 in f (x - 2)
Would be renamed by having Unique
s attached so it looked something like this:
f_1 x_2 = let f_3 x_4 = x_4 + 1 in f_3 (x_2 - 2)
But see Note [Shadowing] below.
- The resulting syntax tree undergoes type checking (which also deals with instantiating
type class arguments) to yield a
HsExpr
type that hasId
as it's names. - Finally the syntax tree is desugared from the expressive
HsExpr
type into thisExpr
type, which has far fewer constructors and hence is easier to perform optimization, analysis and code generation on.
The type parameter b
is for the type of binders in the expression tree.
The language consists of the following elements:
- Variables
- Primitive literals
- Applications: note that the argument may be a
Type
. See Note [CoreSyn let/app invariant] See Note [Levity polymorphism invariants] - Lambda abstraction See Note [Levity polymorphism invariants]
- Recursive and non recursive
let
s. Operationally this corresponds to allocating a thunk for the things bound and then executing the sub-expression.
The right hand sides of all top-level and recursive let
s
must be of lifted type (see Type for
the meaning of lifted vs. unlifted). There is one exception
to this rule, top-level let
s are allowed to bind primitive
string literals, see Note [CoreSyn top-level string literals].
See Note [CoreSyn let/app invariant] See Note [Levity polymorphism invariants]
We allow a non-recursive let to bind a type variable, thus:
Let (NonRec tv (Type ty)) body
This can be very convenient for postponing type substitutions until the next run of the simplifier.
At the moment, the rest of the compiler only deals with type-let in a Let expression, rather than at top level. We may want to revist this choice.
- Case expression. Operationally this corresponds to evaluating the scrutinee (expression examined) to weak head normal form and then examining at most one level of resulting constructor (i.e. you cannot do nested pattern matching directly with this).
The binder gets bound to the value of the scrutinee,
and the Type
must be that of all the case alternatives
This is one of the more complicated elements of the Core language, and comes with a number of restrictions:
- The list of alternatives may be empty; See Note [Empty case alternatives]
- The
DEFAULT
case alternative must be first in the list, if it occurs at all. - The remaining cases are in order of increasing
tag (for
DataAlts
) or lit (forLitAlts
). This makes finding the relevant constructor easy, and makes comparison easier too. - The list of alternatives must be exhaustive. An exhaustive case does not necessarily mention all constructors:
data Foo = Red | Green | Blue ... case x of Red -> True other -> f (case x of Green -> ... Blue -> ... ) ...
The inner case does not need a Red
alternative, because x
can't be Red
at that program point.
- Floating-point values must not be scrutinised against literals. See Trac #9238 and Note [Rules for floating-point comparisons] in PrelRules for rationale.
- Cast an expression to a particular type.
This is used to implement
newtype
s (anewtype
constructor or destructor just becomes aCast
in Core) and GADTs. - Notes. These allow general information to be added to expressions in the syntax tree
- A type: this should only show up at the top level of an Arg
- A coercion
type Alt b = (AltCon, [b], Expr b) Source #
A case split alternative. Consists of the constructor leading to the alternative,
the variables bound from the constructor, and the expression to be executed given that binding.
The default alternative is (DEFAULT, [], rhs)
Binding, used for top level bindings in a module and local bindings in a let
.
A case alternative constructor (i.e. pattern match)
Allows attaching extra information to points in expressions
ProfNote | An |
| |
HpcTick | A "tick" used by HPC to track the execution of each subexpression in the original source code. |
| |
Breakpoint | A breakpoint for the GHCi debugger. This behaves like an HPC tick, but has a list of free variables which will be available for inspection in GHCi when the program stops at the breakpoint. NB. we must take account of these Ids when (a) counting free variables, and (b) substituting (don't substitute for them) |
| |
SourceNote | A source note. Source notes are pure annotations: Their presence should neither influence compilation nor execution. The semantics are given by causality: The presence of a source note means that a local change in the referenced source code span will possibly provoke the generated code to change. On the flip-side, the functionality of annotated code *must* be invariant against changes to all source code *except* the spans referenced in the source notes (see "Causality of optimized Haskell" paper for details). Therefore extending the scope of any given source note is always valid. Note that it is still undesirable though, as this reduces their usefulness for debugging and profiling. Therefore we will generally try only to make use of this property where it is necessary to enable optimizations. |
|
data TickishScoping Source #
Specifies the scoping behaviour of ticks. This governs the behaviour of ticks that care about the covered code and the cost associated with it. Important for ticks relating to profiling.
NoScope | No scoping: The tick does not care about what code it covers. Transformations can freely move code inside as well as outside without any additional annotation obligations |
SoftScope | Soft scoping: We want all code that is covered to stay covered. Note that this scope type does not forbid transformations from happening, as as long as all results of the transformations are still covered by this tick or a copy of it. For example let x = tick... (let y = foo in bar) in baz ===> let x = tick... bar; y = tick... foo in baz Is a valid transformation as far as "bar" and "foo" is concerned, because both still are scoped over by the tick. Note though that one might object to the "let" not being covered by the tick any more. However, we are generally lax with this - constant costs don't matter too much, and given that the "let" was effectively merged we can view it as having lost its identity anyway. Also note that this scoping behaviour allows floating a tick "upwards" in pretty much any situation. For example: case foo of x -> tick... bar ==> tick... case foo of x -> bar While this is always leagl, we want to make a best effort to only make us of this where it exposes transformation opportunities. |
CostCentreScope | Cost centre scoping: We don't want any costs to move to other cost-centre stacks. This means we not only want no code or cost to get moved out of their cost centres, but we also object to code getting associated with new cost-centre ticks - or changing the order in which they get applied. A rule of thumb is that we don't want any code to gain new annotations. However, there are notable exceptions, for example: let f = y -> foo in tick... ... (f x) ... ==> tick... ... foo[x/y] ... In-lining lambdas like this is always legal, because inlining a function does not change the cost-centre stack when the function is called. |
data TickishPlacement Source #
Governs the kind of expression that the tick gets placed on when
annotating for example using mkTick
. If we find that we want to
put a tickish on an expression ruled out here, we try to float it
inwards until we find a suitable expression.
PlaceRuntime | Place ticks exactly on run-time expressions. We can still move the tick through pure compile-time constructs such as other ticks, casts or type lambdas. This is the most restrictive placement rule for ticks, as all tickishs have in common that they want to track runtime processes. The only legal placement rule for counting ticks. |
PlaceNonLam | As |
PlaceCostCentre | In addition to floating through lambdas, cost-centre style tickishs can also be moved from constructors, non-function variables and literals. For example: let x = scc... C (scc... y) (scc... 3) in ... Neither the constructor application, the variable or the literal are likely to have any cost worth mentioning. And even if y names a thunk, the call would not care about the evaluation context. Therefore removing all annotations in the above example is safe. |
type CoreProgram = [CoreBind] Source #
The common case for the type of binders and variables when we are manipulating the Core language within GHC
type TaggedExpr t = Expr (TaggedBndr t) Source #
type TaggedAlt t = Alt (TaggedBndr t) Source #
type TaggedBind t = Bind (TaggedBndr t) Source #
type TaggedArg t = Arg (TaggedBndr t) Source #
deTagExpr :: TaggedExpr t -> CoreExpr Source #
In/Out type synonyms
type InCoercion = Coercion Source #
type OutCoercion = Coercion Source #
Expr
construction
mkLets :: [Bind b] -> Expr b -> Expr b Source #
Bind all supplied binding groups over an expression in a nested let expression. Assumes
that the rhs satisfies the let/app invariant. Prefer to use mkCoreLets
if
possible, which does guarantee the invariant
mkLams :: [b] -> Expr b -> Expr b Source #
Bind all supplied binders over an expression in a nested lambda expression. Prefer to
use mkCoreLams
if possible
mkApps :: Expr b -> [Arg b] -> Expr b infixl 4 Source #
Apply a list of argument expressions to a function expression in a nested fashion. Prefer to
use mkCoreApps
if possible
mkTyApps :: Expr b -> [Type] -> Expr b infixl 4 Source #
Apply a list of type argument expressions to a function expression in a nested fashion
mkCoApps :: Expr b -> [Coercion] -> Expr b infixl 4 Source #
Apply a list of coercion argument expressions to a function expression in a nested fashion
mkVarApps :: Expr b -> [Var] -> Expr b infixl 4 Source #
Apply a list of type or value variables to a function expression in a nested fashion
mkIntLit :: DynFlags -> Integer -> Expr b Source #
Create a machine integer literal expression of type Int#
from an Integer
.
If you want an expression of type Int
use mkIntExpr
mkIntLitInt :: DynFlags -> Int -> Expr b Source #
Create a machine integer literal expression of type Int#
from an Int
.
If you want an expression of type Int
use mkIntExpr
mkWordLit :: DynFlags -> Integer -> Expr b Source #
Create a machine word literal expression of type Word#
from an Integer
.
If you want an expression of type Word
use mkWordExpr
mkWordLitWord :: DynFlags -> Word -> Expr b Source #
Create a machine word literal expression of type Word#
from a Word
.
If you want an expression of type Word
use mkWordExpr
mkWord64LitWord64 :: Word64 -> Expr b Source #
mkInt64LitInt64 :: Int64 -> Expr b Source #
mkCharLit :: Char -> Expr b Source #
Create a machine character literal expression of type Char#
.
If you want an expression of type Char
use mkCharExpr
mkStringLit :: String -> Expr b Source #
Create a machine string literal expression of type Addr#
.
If you want an expression of type String
use mkStringExpr
mkFloatLit :: Rational -> Expr b Source #
Create a machine single precision literal expression of type Float#
from a Rational
.
If you want an expression of type Float
use mkFloatExpr
mkFloatLitFloat :: Float -> Expr b Source #
Create a machine single precision literal expression of type Float#
from a Float
.
If you want an expression of type Float
use mkFloatExpr
mkDoubleLit :: Rational -> Expr b Source #
Create a machine double precision literal expression of type Double#
from a Rational
.
If you want an expression of type Double
use mkDoubleExpr
mkDoubleLitDouble :: Double -> Expr b Source #
Create a machine double precision literal expression of type Double#
from a Double
.
If you want an expression of type Double
use mkDoubleExpr
mkConApp :: DataCon -> [Arg b] -> Expr b Source #
Apply a list of argument expressions to a data constructor in a nested fashion. Prefer to
use mkCoreConApps
if possible
mkTyBind :: TyVar -> Type -> CoreBind Source #
Create a binding group where a type variable is bound to a type. Per CoreSyn,
this can only be used to bind something in a non-recursive let
expression
mkCoBind :: CoVar -> Coercion -> CoreBind Source #
Create a binding group where a type variable is bound to a type. Per CoreSyn,
this can only be used to bind something in a non-recursive let
expression
varToCoreExpr :: CoreBndr -> Expr b Source #
varsToCoreExprs :: [CoreBndr] -> [Expr b] Source #
cmpAltCon :: AltCon -> AltCon -> Ordering Source #
Compares AltCon
s within a single list of alternatives
Simple Expr
access functions and predicates
bindersOfBinds :: [Bind b] -> [b] Source #
bindersOf
applied to a list of binding groups
rhssOfBind :: Bind b -> [Expr b] Source #
rhssOfAlts :: [Alt b] -> [Expr b] Source #
collectBinders :: Expr b -> ([b], Expr b) Source #
We often want to strip off leading lambdas before getting down to
business. Variants are collectTyBinders
, collectValBinders
,
and collectTyAndValBinders
collectNBinders :: Int -> Expr b -> ([b], Expr b) Source #
Strip off exactly N leading lambdas (type or value). Good for use with join points.
collectArgs :: Expr b -> (Expr b, [Arg b]) Source #
Takes a nested application expression and returns the the function being applied and the arguments to which it is applied
collectArgsTicks :: (Tickish Id -> Bool) -> Expr b -> (Expr b, [Arg b], [Tickish Id]) Source #
Like collectArgs
, but also collects looks through floatable
ticks if it means that we can find more arguments.
flattenBinds :: [Bind b] -> [(b, Expr b)] Source #
Collapse all the bindings in the supplied groups into a single
list of lhs/rhs pairs suitable for binding in a Rec
binding group
exprToType :: CoreExpr -> Type Source #
exprToCoercion_maybe :: CoreExpr -> Maybe Coercion Source #
If the expression is a Coercion
, converts.
applyTypeToArg :: Type -> CoreExpr -> Type Source #
Determines the type resulting from applying an expression with given type to a given argument expression
isValArg :: Expr b -> Bool Source #
Returns True
for value arguments, false for type args
NB: coercions are value arguments (zero width, to be sure,
like State#, but still value args).
valArgCount :: [Arg b] -> Int Source #
The number of argument expressions that are values rather than types at their top level
valBndrCount :: [CoreBndr] -> Int Source #
The number of binders that bind values rather than types
isRuntimeArg :: CoreExpr -> Bool Source #
Will this argument expression exist at runtime?
isRuntimeVar :: Var -> Bool Source #
Will this variable exist at runtime?
Tick-related functions
tickishCounts :: Tickish id -> Bool Source #
A "counting tick" (where tickishCounts is True) is one that counts evaluations in some way. We cannot discard a counting tick, and the compiler should preserve the number of counting ticks as far as possible.
However, we still allow the simplifier to increase or decrease sharing, so in practice the actual number of ticks may vary, except that we never change the value from zero to non-zero or vice versa.
tickishScoped :: Tickish id -> TickishScoping Source #
Returns the intended scoping rule for a Tickish
tickishScopesLike :: Tickish id -> TickishScoping -> Bool Source #
Returns whether the tick scoping rule is at least as permissive as the given scoping rule.
tickishFloatable :: Tickish id -> Bool Source #
Returns True
for ticks that can be floated upwards easily even
where it might change execution counts, such as:
Just (tick... foo) ==> tick... (Just foo)
This is a combination of tickishSoftScope
and
tickishCounts
. Note that in principle splittable ticks can become
floatable using mkNoTick
-- even though there's currently no
tickish for which that is the case.
tickishCanSplit :: Tickish id -> Bool Source #
Returns True
for a tick that is both counting and scoping and
can be split into its (tick, scope) parts using mkNoScope
and
mkNoTick
respectively.
tickishIsCode :: Tickish id -> Bool Source #
Return True
if this source annotation compiles to some backend
code. Without this flag, the tickish is seen as a simple annotation
that does not have any associated evaluation code.
What this means that we are allowed to disregard the tick if doing so means that we can skip generating any code in the first place. A typical example is top-level bindings:
foo = tick... y -> ... ==> foo = y -> tick... ...
Here there is just no operational difference between the first and the second version. Therefore code generation should simply translate the code as if it found the latter.
tickishPlace :: Tickish id -> TickishPlacement Source #
Placement behaviour we want for the ticks
tickishContains :: Eq b => Tickish b -> Tickish b -> Bool Source #
Returns whether one tick "contains" the other one, therefore making the second tick redundant.
Unfolding data types
Records the unfolding of an identifier, which is approximately the form the identifier would have if we substituted its definition in for the identifier. This type should be treated as abstract everywhere except in CoreUnfold
NoUnfolding | We have no information about the unfolding. |
BootUnfolding | We have no information about the unfolding, because
this |
OtherCon [AltCon] | It ain't one of these constructors.
data C = C !(Int -> Int) case x of { C f -> ... } Here, |
DFunUnfolding | |
CoreUnfolding | An unfolding with redundant cached information. Parameters: uf_tmpl: Template used to perform unfolding; NB: Occurrence info is guaranteed correct: see Note [OccInfo in unfoldings and rules] uf_is_top: Is this a top level binding? uf_is_value: uf_is_work_free: Does this waste only a little work if we expand it inside an inlining?
Basically this is a cached version of uf_guidance: Tells us about the size of the unfolding template |
|
data UnfoldingGuidance Source #
UnfoldingGuidance
says when unfolding should take place
Constructing Unfolding
s
noUnfolding :: Unfolding Source #
There is no known Unfolding
bootUnfolding :: Unfolding Source #
There is no known Unfolding
, because this came from an
hi-boot file.
evaldUnfolding :: Unfolding Source #
This unfolding marks the associated thing as being evaluated
mkOtherCon :: [AltCon] -> Unfolding Source #
unSaturatedOk :: Bool Source #
needSaturated :: Bool Source #
boringCxtOk :: Bool Source #
Predicates and deconstruction on Unfolding
unfoldingTemplate :: Unfolding -> CoreExpr Source #
Retrieves the template of an unfolding: panics if none is known
maybeUnfoldingTemplate :: Unfolding -> Maybe CoreExpr Source #
Retrieves the template of an unfolding if possible maybeUnfoldingTemplate is used mainly wnen specialising, and we do want to specialise DFuns, so it's important to return a template for DFunUnfoldings
otherCons :: Unfolding -> [AltCon] Source #
The constructors that the unfolding could never be:
returns []
if no information is available
isValueUnfolding :: Unfolding -> Bool Source #
Determines if it is certainly the case that the unfolding will
yield a value (something in HNF): returns False
if unsure
isEvaldUnfolding :: Unfolding -> Bool Source #
Determines if it possibly the case that the unfolding will
yield a value. Unlike isValueUnfolding
it returns True
for OtherCon
isCheapUnfolding :: Unfolding -> Bool Source #
Is the thing we will unfold into certainly cheap?
isConLikeUnfolding :: Unfolding -> Bool Source #
True
if the unfolding is a constructor application, the application
of a CONLIKE function or OtherCon
isStableUnfolding :: Unfolding -> Bool Source #
isFragileUnfolding :: Unfolding -> Bool Source #
hasSomeUnfolding :: Unfolding -> Bool Source #
Only returns False if there is no unfolding information available at all
isBootUnfolding :: Unfolding -> Bool Source #
isStableSource :: UnfoldingSource -> Bool Source #
Annotated expression data types
type AnnExpr bndr annot = (annot, AnnExpr' bndr annot) Source #
Annotated core: allows annotation at every node in the tree
data AnnExpr' bndr annot Source #
A clone of the Expr
type but allowing annotation at every tree node
AnnVar Id | |
AnnLit Literal | |
AnnLam bndr (AnnExpr bndr annot) | |
AnnApp (AnnExpr bndr annot) (AnnExpr bndr annot) | |
AnnCase (AnnExpr bndr annot) bndr Type [AnnAlt bndr annot] | |
AnnLet (AnnBind bndr annot) (AnnExpr bndr annot) | |
AnnCast (AnnExpr bndr annot) (annot, Coercion) | |
AnnTick (Tickish Id) (AnnExpr bndr annot) | |
AnnType Type | |
AnnCoercion Coercion |
type AnnAlt bndr annot = (AltCon, [bndr], AnnExpr bndr annot) Source #
A clone of the Alt
type but allowing annotation at every tree node
Operations on annotated expressions
collectAnnArgs :: AnnExpr b a -> (AnnExpr b a, [AnnExpr b a]) Source #
Takes a nested application expression and returns the the function being applied and the arguments to which it is applied
collectAnnArgsTicks :: (Tickish Var -> Bool) -> AnnExpr b a -> (AnnExpr b a, [AnnExpr b a], [Tickish Var]) Source #
Operations on annotations
deAnnotate :: AnnExpr bndr annot -> Expr bndr Source #
deAnnotate' :: AnnExpr' bndr annot -> Expr bndr Source #
collectAnnBndrs :: AnnExpr bndr annot -> ([bndr], AnnExpr bndr annot) Source #
As collectBinders
but for AnnExpr
rather than Expr
collectNAnnBndrs :: Int -> AnnExpr bndr annot -> ([bndr], AnnExpr bndr annot) Source #
As collectNBinders
but for AnnExpr
rather than Expr
Orphanhood
Is this instance an orphan? If it is not an orphan, contains an OccName
witnessing the instance's non-orphanhood.
See Note [Orphans]
chooseOrphanAnchor :: NameSet -> IsOrphan Source #
Core rule data types
A CoreRule
is:
- "Local" if the function it is a rule for is defined in the same module as the rule itself.
- "Orphan" if nothing on the LHS is defined in the same module as the rule itself
Rule | |
| |
BuiltinRule | Built-in rules are used for constant folding and suchlike. They have no free variables. A built-in rule is always visible (there is no such thing as an orphan built-in rule.) |
type RuleName = FastString Source #
type IdUnfoldingFun = Id -> Unfolding Source #
type InScopeEnv = (InScopeSet, IdUnfoldingFun) Source #
A full rule environment which we can apply rules from. Like a RuleBase
,
but it also includes the set of visible orphans we use to filter out orphan
rules which are not visible (even though we can see them...)
Operations on CoreRule
s
ruleArity :: CoreRule -> Int Source #
The number of arguments the ru_fn
must be applied
to before the rule can match on it
ruleActivation :: CoreRule -> Activation Source #
isBuiltinRule :: CoreRule -> Bool Source #
isLocalRule :: CoreRule -> Bool Source #
isAutoRule :: CoreRule -> Bool Source #