seq module

The module seq provides a generic sequence container Seq for Rust.

Seq is a lightweight container of data sequences (LIFO), managing dynamic list without dynamic memory allocation involved. Sequences are stored in stack frames of function contexts. Each element of a sequence has an individual lifetime 'a managed by the Rust compiler.

Add the following dependency to your Cargo.toml file: ```toml

Cargo.toml file

[dependencies] seq = "0.3.1" ```

Definition

Seq is defined as generic enum. Seq is a sequence of data of type T and the top most element has lifetime 'a.

Either a sequence is Empty or a sequence is a construction of a new value (head or first (ft)) on-top of another sequence (the tail or rest (rt)). The lifetime of the tail must be at least as long as the one of the head. Two kind of constructions are possible, depending on location of tail in memory-heap or on stack. rust pub enum Seq<'a, T: 'a> { Empty, ConsRef(T, &'a Seq<'a, T>), ConsOwn(T, Box<Seq<'a, T>>), } Explaning the enum variants: * Empty: The empty sequence <> * ConsRef(head, tail): Constructs a new sequence with head being the first element and tail referencing another, borrowed sequence. This variant permits construction of sequences using stack-allocated data solely. * ConsOwn(head, boxedtail): Constructs a new sequence with head being the first element and boxedtail referencing another, owned, boxed sequence. Here the tail is residing in heap allocated memory. This variant permits construction of sequences using heap-allocated dynamic data.

These variants may be combined with each other, representing a mixture of borrowed and owned elements. The memory safety feature of Rust allows automated and correct management of lifetime of each element of the sequence.

The lifetime of each element of the sequence depends on the function-context it has been added to the top of the sequence; Empty is the element with longest lifetime (see image).

The following image illustrates the sequences s, t, u. The sequence s is a sub-sequence of t, and t being a sub-sequence of u; each one accessible in its function context only. Illustration of sequence elements in stack frames

For use-cases where a sub-routine/expression shall return a temporary extended sequence, it is possible to construct new sequences using elements in heap-memory. In this case these heap-elements are boxed/owned. Illustration of sequence elements in stack frames and heap

But, first and foremost, the container Seq is intended as lightweight, dynamic, stack-allocated, linked list for use cases such as managing state/context while traversing tree-like data structures - without any dynamic memory-allocation (heap) involved.

The sequence type Seq implements the trait IntoIterator, enabling the usage of Rust's iterator framework.

Example: Stack-only Allocated

A stack allocated sequnece is based on the variants Seq::Empty and Seq::ConsRef only ```rust extern crate seq; use seq::Seq;

fn myfun() { // constructing the sequence seq!(3,2,1) let s0: Seq = Seq::Empty(); let s1: Seq = Seq::ConsRef(1, &s0); let s3: Seq = Seq::ConsRef(2, &s1); let s3: Seq = Seq::ConsRef(3, &s2); println!("seq {:?}", &s3); } ```

Example: Stack-and-Heap Allocated

A sequence can be a mixture of stack-allocated and heap-allocated data elements. The following sequence ```rust extern crate seq; use seq::Seq;

fn myfun() { let s0: Seq = Seq::Empty(); // stack let s1: Seq = Seq::ConsRef(0, &s0); // stack let s2: Box> = Box::new(Seq::ConsRef(1, &s1)); // heap let s3: Box> = Box::new(Seq::ConsOwn(2, s2)); // heap let s4: Seq = Seq::ConsOwn(Data(3, s3)); // stack println!("seq {:?}", &s4); } ```

Example: Pattern Matching

Pattern-matching is used to de-construct a sequence. ```rust extern crate seq; use seq::Seq;

fn head(sequence: &Seq) -> Option { match sequence { &Seq::Empty => Option::None, &Seq::ConsRef(ref ft, ref rt) => Option::Some(ft), &Seq::ConsOwn(ref ft, ref rt) => Option::Some(ft) } } ```

Example: Dynamic sequence in nested/recursive function-calls

Sequences can be used to manage state in nested function calls. This code demonstrates how the iterator is used. ```rust extern crate seq; use seq::Seq; use std::ops;

// Recursive, nested invocation while val) { if val < max { let sequence = Seq::ConsRef(val, tail); println!("sum is: {}", sequence.into_iter().fold(0, ops::Add::add)); recurs(val + 1, max, &sequence); } }

fn main() { recurs(0, 10, seq::empty()); } ```

Example: Dynamic sequence in nested/recursive function-calls, combined with heap-alloc. data

'Seq' permits mixture of stack allocated data and heap allocated data within a single linked list. The following code is a variation of previous sample, just adding two heap-allocated elements onto top of sequence finally. This code demonstrates how the iterator is used. ```rust extern crate seq; use seq::Seq; use std::ops;

fn prependTwoBoxedValues <'a> (v: u32, w: u32, seq: &'a Seq ) -> Box> { Box::new(Seq::ConsOwn(v +1, Box::new(Seq::ConsRef(w, seq)))) }

// Recursive, nested invocation while val) { if val < max { let sequence = Seq::ConsRef(val, tail); println!("sum is: {}", sequence.intoiter().fold(0, ops::Add::add)); recurs(val + 1, max, &sequence); } else { let sequence: Box> = prependTwoBoxedValues(max+1, max, tail); println!("sum is: {}", sequence.intoiter().fold(0, ops::Add::add)); } }

fn main() { recurs(0, 10, seq::empty()); } ```

Example: Demonstrating the macro seqdef!

The seqdef! macro defines a stack-allocated sequence variable using the speficied data list, the last data item in the list will be the top most in the sequence (head). The macro can be used to create a new sequence on top of another one (tail).

Macro 1: Creating a seq variable s, the head will be 2.

Is constructing a sequence s := <>|0|1|2; rust seqdef!(s; seq::empty() => 0, 1, 2);

Macro 2: Creating a seq variable t without explicit seq::empty(). Seq t is identical to s.

Is constructing a sequence t := <>|0|1|2; rust seqdef!(t; 0, 1, 2);

Macro 3: Creating a seq variable u, using Seq s of example 1 as tail, the head will be 5.

On top of sequence s, constructing a sequence u := &s|3|4|5 the effictive sequence is u := <>|0|1|2|3|4|5; rust seqdef!(u; &s => 3, 4, 5);

Macro 4: Creating a dynamic seq with MAX elements within stack frame, reading values from iterator

The previous macros are useful, but limited as the exact number of elements must be known at compile time. The previous macros are not able to deal with dynamic number of elements between 1-MAX, where MAX might be a large value.

The following macro seqdef_try! simplifies handling of dynamic number of values when constructing a sequence. This macro reserves MAX elements on the stack every time when entering the function context. The upper limit MAX is defined at compile time. At runtime the sequence is constructed, reading the elements from the iterator and placing them in the reserved stack-memory. When the function context is left, the stack-memory is released. The macro seqdef_try! will declare the specified identifier as type Result>. Rolling out the values from iterator into the reserved stack may fail if the iterator is empty, or if the amount exceeds MAX. The value of x must be checked after construction with seqdef_try!.

Note! No matter the number of elements returned by the iterator, the macro is always reserving stack-memory for MAX elements. If you choose too large, the stack might run out of memory. The iterator provided to the macro may consume the underlying container-elements or clone each element. ```rust use std::mem; use std::ptr;

fn dynamicseqrolloutonstack() { const MAX: usize = 10; let large_list: &[i32] = &[42; MAX];

  // define x of type Result<Seq<i32>>, read all elements from array and place them on stack as sequence
  seqdef_try!(x, i32, MAX; empty() => large_list.iter());

  // handling the result, Ok or Error
  match &x {
      &Ok(ref sequence) => println!("large sum {}", (&*sequence).into_iter().fold(0i32, ops::Add::add)),
      &Err(reason) => println!("roll out failed due to {}", reason)
  }

} ```