Crate rusttyc[][src]

Expand description

This crate provides an interface to perform lattice-based type checking on arbitrary structures.

The TypeChecker struct constitutes the main struct. It provides functions to create new TcKeys.
These keys represent typed entities such as terms or variables in programming languages. TcKey provides a set of functions generating constraints such as ‘the type behind key a is more concrete than the type behind key b’ or ‘type X is an upper bound of the type behind key a’.

The user needs to supply a type lattice by implementing the Variant trait.

Then, iterate over your data structure, e.g. your abstract syntax tree, generate keys for terms and variables, and impose constraints on the keys. Lastly, generate a type table mapping keys to their resolved types. These can either be Variants with the keys representing their children, or constructed types if Variant implements Constructable.


Consider a type lattice consisting of a boolean type and an integer type, where the integer type is a variable bit length.

#[derive(PartialEq, Eq, Clone, Debug)]
enum MyVariant {

Implement the Variant type for the enum. This requires an additional error type, access to a Variant::top() element, and information whether or not the variant has a fixed or variable arity, i.e., the number of subtypes relevant for this type. As an example consider an Option-like type. As a monad, it has arity 1. Scalar types have arity 0, tuples of undetermined length have variable arity.

type MyTypeErr = String;
impl Variant for MyVariant {
    type Err = MyTypeErr;
    fn arity(&self) -> Arity { Arity::Fixed(0) }
    fn top() -> Self { MyVariant::Top }
    fn meet(lhs: Partial<Self>, rhs: Partial<Self>) -> Result<Partial<Self>, Self::Err> {
        assert_eq!(lhs.least_arity, 0);
        assert_eq!(lhs.least_arity, 0);
        let variant = match (lhs.variant, rhs.variant) {
            (MyVariant::Top, x) | (x, MyVariant::Top) => Ok(x),
            (MyVariant::Boolean, MyVariant::Boolean) => Ok(MyVariant::Boolean),
            (MyVariant::Integer(a), MyVariant::Integer(b)) => Ok(MyVariant::Integer(usize::max(a, b))),
            (MyVariant::Boolean, MyVariant::Integer(_)) | (MyVariant::Integer(_), MyVariant::Boolean) => Err(String::from("Cannot combine Boolean and Integer")),
        Ok(Partial { variant, least_arity: 0 })
#[derive(PartialEq, Eq, Hash, Clone, Copy, Debug)]
struct MyVariable(u8);
impl TcVar for MyVariable {}

let mut tc: TypeChecker<MyVariant, MyVariable> = TypeChecker::new();
// We type check `x = 0b111 ^ 0b11010`, so x needs 5 bits.
let t1 = tc.new_term_key();
// The first term is an int with at least a width of 3 bits.
let t2 = tc.new_term_key();
// The second term is an int with at least a width of 5 bits.
let tx = tc.new_term_key(); // tx is the combination of t1 and t2, e.g. xor or addition.
tc.impose(tx.is_meet_of(t1, t2))?; // The result is the meet of both types.
let type_table = tc.type_check_preliminary()?;
assert_eq!(type_table[&tx].variant, MyVariant::Integer(5));

Additional Examples

Check the documentation of TcKey for all possible constraints imposable on keys and their effects. Check the RustTyC examples on github for more elaborate examples.


pub use types::Arity;
pub use types::Constructable;
pub use types::Partial;
pub use types::Preliminary;
pub use types::PreliminaryTypeTable;
pub use types::TypeTable;
pub use types::Variant;



This mod contains everything related to types and collections of types (type tables).



An inexpensive and simple indexing mechanism using during the type checking procedure.


The TypeChecker is the main interaction point for the type checking procedure.



Represents an error during the type check procedure.



Represents a re-usable variable in the type checking procedure.

Type Definitions


A TypeChecker instance in case no variables are required.