11 Type Checking and Inference
Type checking and inference is just as in ML (Hindley-Milner), with a few small exceptions:
- Functions can take multiple arguments, instead of requring a tuple of arguments. Thus, (Int, Int) -> Int is a different type than either (Int * Int) -> Int, which is the tuple variant, or Int -> (Int -> Int), which is the curried variant. 
- Since all top-level definitions are in the same mutually-recursive scope, the type of a definition’s right-hand side is not directly unified with references to the defined identifier on the right-hand side. Instead, every reference to an identifier— - even a reference in the identifier’s definition— - is unified with a instantiation of a polymorphic type inferred for the definition. - Compare OCaml: - # let rec f = fun x -> x - and h = fun y -> f 0 - and g = fun z -> f "x";; - This expression has type string but is here used with type int - with - > f - - ?a -> ?a - #<function:f> - > h - - ?a -> Int - #<function:h> - > g - - ?a -> String - #<function:g> - A minor consequence is that polymorphic recursion (i.e., a self call with an argument whose type is different than that for the current call) is allowed. Recursive types, however, are prohibited. Polymorphic recursion is not decidable, so see ~fuel in Inference Fuel. - The usual value restriction applies for inferring polymorphic types, where expression matching the following grammar (before macro expansion, unfortunately) are considered values: - = - (value) - | - fun (typed_id, ...) maybe_type: expr - | - | - [value, ...] - | - variant_id(value, ...) - | - literal - | - where variand_id refers to any or a constructor bound by type. 
- Since all definitions are recursively bound, and since the right-hand side of a definition does not have to be a function, its possible to refer to a variable before it is defined. The type system does not prevent “reference to identifier before definition” errors. 
- Interactive evaluation (e.g., in DrRacket’s interactions window) can redefine identifiers that were previously defined interactively or that were defined in a module as mutable. Redefinition cannot change the identifier’s type. Due to a limitation of the type checker, identifiers of polymorphic type cannot be redefined or redeclared. Type declarations are allowed in interactive evaluation, but a declared type is never treated as a polymorphic type. 
When typechecking fails, the error messages reports and highlights (in pink) all of the expressions whose type contributed to the failure. That’s often too much information. As usual, explicit type annotations can help focus the error message.
Type checking can be disabled for a Shplait module using ~untyped language option. See Untyped Mode for more information.