Skip to content

Warning

This page is under construction. The content may be incomplete or incorrect. Submit an issue on GitHub if you need help or want to contribute.

The Immutable List Dialect

The immutable list dialect models a Python-like list but is immutable. There are many good reasons to have an immutable list for compilation, especially when a list is immutable, we can assume a lot of statements to be pure and foldable (and thus can be inlined or simplified without extra analysis/rewrites).

JAX

This is also the reason why JAX picks an immutable array semantic.

Why not use tuple?

Tuple can take multiple items of different types, but list can only take items of the same type. Thus they have different trade-offs when doing analysis such as type inference of iterating through a tuple/list.

Runtime

This dialect provides a runtime object IList which is a simple Python class wraps a Python list. This object can be used as a compile-time value by providing an implementation of __hash__ that returns the object id. This means common simplifications like Common Subexpression Elimination (CSE) will not detect duplicated IList unless the ir.SSAValue points to identical IList object.

Implementation Details

this is an implementation detail of IList, we can switch to a more efficient runtime in the future where the memory layout is optimized based on the assumption of items in same type and immutabiility.

The IList runtime object implements most of the Python Sequence interface, such as __getitem__, __iter__, __len__ etc.

New

This statements take a tuple of ir.SSAValue and creates an IList as result.

Syntax Sugar in Lowering

The syntax [a, b, c] will be lowered into New statement as a syntax sugar when ilist dialect is used (thus in conflict with mutable Python list). This may change in the future to give developers more freedom to choose what to lower from.

Map

This statements take a high-order function (a function object) of signature [[ElemT], OutT] and apply it on a given IList[ElemT, Len] object then return a new IList[OutT, Len].

For example:

@basic_no_opt
def main(x: ilist.IList[float, Literal[5]]):
    def closure(a):
        return a + 1
    return ilist.map(closure, x)

will be lowerd into the following

func.func main(!py.IList[!py.float, 5]) -> !Any {
  ^0(%main_self, %x):
  │ %closure = func.lambda closure() -> !Any {
  │            │ ^1(%closure_self, %a):
  │            │ │ %1 = py.constant.constant 1 : !py.int
  │            │ │ %2 = py.binop.add(%a, %1) : ~T
  │            │ │      func.return %2
  │            } // func.lambda closure
  │       %0 = py.ilist.map(fn=%closure, collection=%x : !py.IList[!py.float, 5]) : !py.IList[~OutElemT, ~ListLen]
  │            func.return %0
} // func.func main

Foldl and Foldr

These two statements represents applying a binary operator + (any binary operator) on an IList with a different reduction order, e.g given [a, b, c], Foldl represents ((a + b) + c) and Foldr represents (a + (b + c)).

Scan

While the actual implementation is not the same, this statement represents the same semantics as the following function:

def scan(fn, xs, init):
    carry = init
    ys = []
    for elem in xs:
        carry, y = fn(carry, elem)
        ys.append(y)
    return carry, ys

ForEach

this represents a for-loop without any loop variables (variables pass through each loop iteration).

API Reference

CarryT module-attribute

CarryT = TypeVar('CarryT')

ElemT module-attribute

ElemT = TypeVar('ElemT')

IListType module-attribute

IListType = Generic(IList, ElemT, ListLen)

ListLen module-attribute

ListLen = TypeVar('ListLen')

OutElemT module-attribute

OutElemT = TypeVar('OutElemT')

ResultT module-attribute

ResultT = TypeVar('ResultT')

All kirin-statement

All(collection: SSAValue)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.All[All]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.All
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.All href "" "kirin.dialects.ilist.stmts.All"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[Bool, ListLen])

result kirin-result

result: ResultValue = result(Bool)

traits class-attribute instance-attribute

traits = frozenset({Pure(), FromPythonCall()})

Any kirin-statement

Any(collection: SSAValue)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Any[Any]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Any
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Any href "" "kirin.dialects.ilist.stmts.Any"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[Bool, ListLen])

result kirin-result

result: ResultValue = result(Bool)

traits class-attribute instance-attribute

traits = frozenset({Pure(), FromPythonCall()})

Foldl kirin-statement

Foldl(
    fn: SSAValue,
    collection: SSAValue,
    init: SSAValue,
    *,
    purity: bool
)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Foldl[Foldl]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Foldl
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Foldl href "" "kirin.dialects.ilist.stmts.Foldl"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[ElemT])

fn kirin-argument

fn: SSAValue = argument(
    Generic(Method, [OutElemT, ElemT], OutElemT)
)

init kirin-argument

init: SSAValue = argument(OutElemT)

purity kirin-attribute kw-only

purity: bool = attribute(default=False)

result kirin-result

result: ResultValue = result(OutElemT)

traits class-attribute instance-attribute

traits = frozenset({MaybePure(), FromPythonCall()})

Foldr kirin-statement

Foldr(
    fn: SSAValue,
    collection: SSAValue,
    init: SSAValue,
    *,
    purity: bool
)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Foldr[Foldr]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Foldr
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Foldr href "" "kirin.dialects.ilist.stmts.Foldr"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[ElemT])

fn kirin-argument

fn: SSAValue = argument(
    Generic(Method, [ElemT, OutElemT], OutElemT)
)

init kirin-argument

init: SSAValue = argument(OutElemT)

purity kirin-attribute kw-only

purity: bool = attribute(default=False)

result kirin-result

result: ResultValue = result(OutElemT)

traits class-attribute instance-attribute

traits = frozenset({MaybePure(), FromPythonCall()})

ForEach kirin-statement

ForEach(
    fn: SSAValue, collection: SSAValue, *, purity: bool
)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.ForEach[ForEach]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.ForEach
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.ForEach href "" "kirin.dialects.ilist.stmts.ForEach"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[ElemT])

fn kirin-argument

fn: SSAValue = argument(Generic(Method, [ElemT], NoneType))

purity kirin-attribute kw-only

purity: bool = attribute(default=False)

traits class-attribute instance-attribute

traits = frozenset({MaybePure(), FromPythonCall()})

Map kirin-statement

Map(fn: SSAValue, collection: SSAValue, *, purity: bool)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Map[Map]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Map
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Map href "" "kirin.dialects.ilist.stmts.Map"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[ElemT, ListLen])

fn kirin-argument

fn: SSAValue = argument(TypeofMethodType[[ElemT], OutElemT])

purity kirin-attribute kw-only

purity: bool = attribute(default=False)

result kirin-result

result: ResultValue = result(IListType[OutElemT, ListLen])

traits class-attribute instance-attribute

traits = frozenset({MaybePure(), FromPythonCall()})

New kirin-statement

New(
    values: Sequence[SSAValue],
    elem_type: TypeAttribute | None = None,
)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.New[New]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.New
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.New href "" "kirin.dialects.ilist.stmts.New"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            
Source code in src/kirin/dialects/ilist/stmts.py
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
def __init__(
    self,
    values: Sequence[ir.SSAValue],
    elem_type: types.TypeAttribute | None = None,
) -> None:
    if not elem_type:
        if not values:
            elem_type = types.Any
        else:
            elem_type = values[0].type
            for v in values[1:]:
                elem_type = elem_type.join(v.type)

    result_type = IListType[elem_type, types.Literal(len(values))]
    super().__init__(
        args=values,
        result_types=(result_type,),
        args_slice={"values": slice(0, len(values))},
        attributes={"elem_type": elem_type},
    )

elem_type kirin-attribute kw-only

elem_type: TypeAttribute = attribute()

result kirin-result

result: ResultValue = result(IListType[ElemT])

traits class-attribute instance-attribute

traits = frozenset({Pure(), FromPythonCall()})

values kirin-argument

values: tuple[SSAValue, ...] = argument(ElemT)

Push kirin-statement

Push(lst: SSAValue, value: SSAValue)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Push[Push]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Push
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Push href "" "kirin.dialects.ilist.stmts.Push"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

lst kirin-argument

lst: SSAValue = argument(IListType[ElemT])

result kirin-result

result: ResultValue = result(IListType[ElemT])

traits class-attribute instance-attribute

traits = frozenset({Pure(), FromPythonCall()})

value kirin-argument

value: SSAValue = argument(IListType[ElemT])

Range kirin-statement

Range(start: SSAValue, stop: SSAValue, step: SSAValue)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Range[Range]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Range
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Range href "" "kirin.dialects.ilist.stmts.Range"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

name class-attribute instance-attribute

name = 'range'

result kirin-result

result: ResultValue = result(IListType[Int, Any])

start kirin-argument

start: SSAValue = argument(Int)

step kirin-argument

step: SSAValue = argument(Int)

stop kirin-argument

stop: SSAValue = argument(Int)

traits class-attribute instance-attribute

traits = frozenset({Pure(), FromPythonRangeLike()})

Scan kirin-statement

Scan(
    fn: SSAValue,
    collection: SSAValue,
    init: SSAValue,
    *,
    purity: bool
)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Scan[Scan]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Scan
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Scan href "" "kirin.dialects.ilist.stmts.Scan"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[ElemT, ListLen])

fn kirin-argument

fn: SSAValue = argument(
    Generic(
        Method, [OutElemT, ElemT], Tuple[OutElemT, ResultT]
    )
)

init kirin-argument

init: SSAValue = argument(OutElemT)

purity kirin-attribute kw-only

purity: bool = attribute(default=False)

result kirin-result

result: ResultValue = result(
    Tuple[OutElemT, IListType[ResultT, ListLen]]
)

traits class-attribute instance-attribute

traits = frozenset({MaybePure(), FromPythonCall()})

Sorted kirin-statement

Sorted(
    collection: SSAValue,
    key: SSAValue,
    reverse: SSAValue,
    *,
    purity: bool
)

Bases: Statement


              flowchart TD
              kirin.dialects.ilist.stmts.Sorted[Sorted]
              kirin.ir.nodes.stmt.Statement[Statement]
              kirin.ir.nodes.base.IRNode[IRNode]
              kirin.print.printable.Printable[Printable]

                              kirin.ir.nodes.stmt.Statement --> kirin.dialects.ilist.stmts.Sorted
                                kirin.ir.nodes.base.IRNode --> kirin.ir.nodes.stmt.Statement
                                kirin.print.printable.Printable --> kirin.ir.nodes.base.IRNode
                




              click kirin.dialects.ilist.stmts.Sorted href "" "kirin.dialects.ilist.stmts.Sorted"
              click kirin.ir.nodes.stmt.Statement href "" "kirin.ir.nodes.stmt.Statement"
              click kirin.ir.nodes.base.IRNode href "" "kirin.ir.nodes.base.IRNode"
              click kirin.print.printable.Printable href "" "kirin.print.printable.Printable"
            

collection kirin-argument

collection: SSAValue = argument(IListType[ElemT, ListLen])

key kirin-argument

key: SSAValue = argument(
    Union((Generic(Method, [ElemT], ElemT), NoneType))
)

purity kirin-attribute kw-only

purity: bool = attribute(default=False)

result kirin-result

result: ResultValue = result(IListType[ElemT, ListLen])

reverse kirin-argument

reverse: SSAValue = argument(Bool)

traits class-attribute instance-attribute

traits = frozenset({MaybePure(), SortedLowering()})