// -------------------------------------------------- module CoRecursion { codatatype Stream = More(head: T, rest: Stream); function AscendingChain(n: int): Stream { More(n, AscendingChain(n+1)) } function AscendingChainAndRead(n: nat): Stream reads this; // with a reads clause, this function is not a co-recusvie function { More(n, AscendingChainAndRead(n+1)) // error: cannot prove termination } datatype List = Nil | Cons(T, List); function Prefix(n: nat, s: Stream): List { if n == 0 then Nil else Cons(s.head, Prefix(n-1, s.rest)) } } // -------------------------------------------------- module CoRecursionNotUsed { codatatype Stream = More(T, Stream); function F(s: Stream, n: nat): Stream decreases n, true; { G(s, n) } function G(s: Stream, n: nat): Stream decreases n, false; { if n == 0 then s else Tail(F(s, n-1)) } function Tail(s: Stream): Stream { match s case More(hd, tl) => tl } function Diverge(n: nat): nat { Diverge(n) // error: cannot prove termination } }