beamr 0.1.0

A Rust runtime with the BEAM's execution model, targeting Gleam
Documentation
//! Gleam stdlib native stubs used by generated Gleam modules.

use crate::atom::Atom;
use crate::native::ProcessContext;
use crate::term::Term;
use crate::term::binary::Binary;
use crate::term::boxed::{Cons, write_tuple};

pub fn bif_string_replace(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, pattern, replacement] = args else {
        return Err(badarg());
    };
    let input = binary_bytes(*input)?;
    let pattern = binary_bytes(*pattern)?;
    let replacement = binary_bytes(*replacement)?;
    if pattern.is_empty() {
        return Err(badarg());
    }

    let mut out = Vec::with_capacity(input.len());
    let mut index = 0;
    while let Some(relative) = find_bytes(&input[index..], pattern) {
        let match_start = index + relative;
        out.extend_from_slice(&input[index..match_start]);
        out.extend_from_slice(replacement);
        index = match_start + pattern.len();
    }
    out.extend_from_slice(&input[index..]);
    make_binary(&out)
}

pub fn bif_less_than(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [left, right] = args else {
        return Err(badarg());
    };
    Ok(bool_term(left < right))
}

pub fn bif_slice(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, offset, length] = args else {
        return Err(badarg());
    };
    let bytes = binary_bytes(*input)?;
    let offset = non_negative_usize(*offset)?;
    let length = non_negative_usize(*length)?;
    let end = offset.checked_add(length).ok_or_else(badarg)?;
    if end > bytes.len() {
        return Err(badarg());
    }
    make_binary(&bytes[offset..end])
}

pub fn bif_crop_string(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, length] = args else {
        return Err(badarg());
    };
    let bytes = binary_bytes(*input)?;
    let length = non_negative_usize(*length)?;
    let end = length.min(bytes.len());
    make_binary(&bytes[..end])
}

pub fn bif_contains_string(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [haystack, needle] = args else {
        return Err(badarg());
    };
    let haystack = binary_bytes(*haystack)?;
    let needle = binary_bytes(*needle)?;
    Ok(bool_term(find_bytes(haystack, needle).is_some()))
}

pub fn bif_string_starts_with(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, prefix] = args else {
        return Err(badarg());
    };
    Ok(bool_term(
        binary_bytes(*input)?.starts_with(binary_bytes(*prefix)?),
    ))
}

pub fn bif_string_ends_with(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, suffix] = args else {
        return Err(badarg());
    };
    Ok(bool_term(
        binary_bytes(*input)?.ends_with(binary_bytes(*suffix)?),
    ))
}

pub fn bif_string_pop_grapheme(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input] = args else {
        return Err(badarg());
    };
    let bytes = binary_bytes(*input)?;
    if bytes.is_empty() {
        return Ok(Term::atom(Atom::ERROR));
    }

    let first_len = std::str::from_utf8(bytes)
        .ok()
        .and_then(|text| text.chars().next().map(char::len_utf8))
        .unwrap_or(1);
    let head = make_binary(&bytes[..first_len])?;
    let rest = make_binary(&bytes[first_len..])?;
    ok_tuple2(head, rest)
}

pub fn bif_utf_codepoint_list_to_string(
    args: &[Term],
    context: &mut ProcessContext,
) -> Result<Term, Term> {
    let _ = context;
    let [input] = args else {
        return Err(badarg());
    };
    let mut out = String::new();
    let mut current = *input;
    loop {
        if current.is_nil() {
            return make_binary(out.as_bytes());
        }
        let cons = Cons::new(current).ok_or_else(badarg)?;
        let codepoint = cons.head().as_small_int().ok_or_else(badarg)?;
        let codepoint = u32::try_from(codepoint).map_err(|_| badarg())?;
        let ch = char::from_u32(codepoint).ok_or_else(badarg)?;
        out.push(ch);
        current = cons.tail();
    }
}

pub fn bif_inspect(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let [value] = args else {
        return Err(badarg());
    };
    if let Some(binary) = Binary::new(*value) {
        return make_binary(binary.as_bytes());
    }
    if let Some(integer) = value.as_small_int() {
        return make_binary(integer.to_string().as_bytes());
    }
    if let Some(atom) = value.as_atom() {
        if let Some(table) = context.atom_table()
            && let Some(name) = table.resolve(atom)
        {
            return make_binary(name.as_bytes());
        }
        return make_binary(format!("Atom({atom:?})").as_bytes());
    }
    make_binary(format!("{value:?}").as_bytes())
}

pub fn bif_string_remove_prefix(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, prefix] = args else {
        return Err(badarg());
    };
    let input = binary_bytes(*input)?;
    let prefix = binary_bytes(*prefix)?;
    if input.starts_with(prefix) {
        ok_tuple1(make_binary(&input[prefix.len()..])?)
    } else {
        Ok(Term::atom(Atom::ERROR))
    }
}

pub fn bif_string_remove_suffix(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [input, suffix] = args else {
        return Err(badarg());
    };
    let input = binary_bytes(*input)?;
    let suffix = binary_bytes(*suffix)?;
    if input.ends_with(suffix) {
        ok_tuple1(make_binary(&input[..input.len() - suffix.len()])?)
    } else {
        Ok(Term::atom(Atom::ERROR))
    }
}

pub fn bif_iodata_append(args: &[Term], context: &mut ProcessContext) -> Result<Term, Term> {
    let _ = context;
    let [left, right] = args else {
        return Err(badarg());
    };
    let mut out = Vec::new();
    collect_iodata(*left, &mut out)?;
    collect_iodata(*right, &mut out)?;
    make_binary(&out)
}

fn collect_iodata(term: Term, bytes: &mut Vec<u8>) -> Result<(), Term> {
    if term.is_nil() {
        return Ok(());
    }
    if let Some(byte) = term.as_small_int() {
        let byte = u8::try_from(byte).map_err(|_| badarg())?;
        bytes.push(byte);
        return Ok(());
    }
    if let Some(binary) = Binary::new(term) {
        bytes.extend_from_slice(binary.as_bytes());
        return Ok(());
    }
    if let Some(cons) = Cons::new(term) {
        collect_iodata(cons.head(), bytes)?;
        return collect_iodata(cons.tail(), bytes);
    }
    Err(badarg())
}

fn binary_bytes(term: Term) -> Result<&'static [u8], Term> {
    Binary::new(term).map(Binary::as_bytes).ok_or_else(badarg)
}

fn non_negative_usize(term: Term) -> Result<usize, Term> {
    term.as_small_int()
        .and_then(|value| usize::try_from(value).ok())
        .ok_or_else(badarg)
}

fn find_bytes(haystack: &[u8], needle: &[u8]) -> Option<usize> {
    if needle.is_empty() {
        return Some(0);
    }
    haystack
        .windows(needle.len())
        .position(|window| window == needle)
}

fn bool_term(value: bool) -> Term {
    Term::atom(if value { Atom::TRUE } else { Atom::FALSE })
}

fn ok_tuple1(value: Term) -> Result<Term, Term> {
    let heap = Box::leak(Box::new([0u64; 3]));
    write_tuple(heap, &[Term::atom(Atom::OK), value]).ok_or_else(badarg)
}

fn ok_tuple2(first: Term, second: Term) -> Result<Term, Term> {
    let heap = Box::leak(Box::new([0u64; 4]));
    write_tuple(heap, &[Term::atom(Atom::OK), first, second]).ok_or_else(badarg)
}

fn make_binary(bytes: &[u8]) -> Result<Term, Term> {
    let data_words = crate::term::binary::packed_word_count(bytes.len());
    let heap: &mut [u64] = Box::leak(vec![0u64; 2 + data_words].into_boxed_slice());
    crate::term::binary::write_binary(heap, bytes).ok_or_else(badarg)
}

fn badarg() -> Term {
    Term::atom(Atom::BADARG)
}