luaur-vm 0.1.0

The Luau register virtual machine and standard library (Rust).
Documentation
//! Node: `cxx:Function:Luau.VM:VM/src/lstrlib.cpp:966:str_format`
//!
//! `string.format` — walk the format string, copying literals and dispatching
//! each `%` spec to the matching argument: `%c/d/i/o/u/x/X/e/E/f/g/G` go through
//! C `snprintf` with the scanned format (integer specs get an int64 length
//! modifier), `%q` quotes, `%s` formats or fast-paths long strings, `%*` appends
//! any value, and `%%` is a literal percent.

use crate::functions::add_int_64_format::add_int_64_format;
use crate::functions::addquoted::addquoted;
use crate::functions::lua_gettop::lua_gettop;
use crate::functions::lua_l_addchar::lua_l_addchar;
use crate::functions::lua_l_addlstring::lua_l_addlstring;
use crate::functions::lua_l_addvalueany::lua_l_addvalueany;
use crate::functions::lua_l_buffinit::lua_l_buffinit;
use crate::functions::lua_l_checkinteger_64::lua_l_checkinteger_64;
use crate::functions::lua_l_checklstring::lua_l_checklstring;
use crate::functions::lua_l_checknumber::lua_l_checknumber;
use crate::functions::lua_l_pushresult::lua_l_pushresult;
use crate::functions::scanformat::scanformat;
use crate::macros::l_esc::L_ESC;
use crate::macros::lua_isinteger_64::lua_isinteger_64;
use crate::macros::lua_l_error::luaL_error;
use crate::macros::max_format::MAX_FORMAT;
use crate::macros::max_item::MAX_ITEM;
use crate::records::lua_l_strbuf::LuaLStrbuf;
use crate::type_aliases::lua_state::lua_State;
use core::ffi::{c_char, c_int};

extern "C" {
    fn snprintf(s: *mut c_char, n: usize, format: *const c_char, ...) -> c_int;
    fn strchr(s: *const c_char, c: c_int) -> *mut c_char;
}

pub unsafe fn str_format(L: *mut lua_State) -> c_int {
    let top = lua_gettop(L);
    let mut arg: c_int = 1;
    let mut sfl: usize = 0;
    let mut strfrmt = lua_l_checklstring(L, arg, &mut sfl);
    let strfrmt_end = strfrmt.add(sfl);

    let mut b: LuaLStrbuf = LuaLStrbuf {
        p: core::ptr::null_mut(),
        end: core::ptr::null_mut(),
        L: core::ptr::null_mut(),
        storage: core::ptr::null_mut(),
        buffer: [0; 512],
    };
    lua_l_buffinit(L, &mut b);

    while strfrmt < strfrmt_end {
        if *strfrmt != L_ESC {
            lua_l_addchar(&mut b, *strfrmt);
            strfrmt = strfrmt.add(1);
        } else {
            strfrmt = strfrmt.add(1); // *++strfrmt
            if *strfrmt == L_ESC {
                lua_l_addchar(&mut b, *strfrmt);
                strfrmt = strfrmt.add(1); // %%
            } else if *strfrmt == b'*' as c_char {
                strfrmt = strfrmt.add(1);
                arg += 1;
                if arg > top {
                    luaL_error!(L, "missing argument #{}", arg);
                }
                lua_l_addvalueany(&mut b, arg);
            } else {
                // format item
                let mut form: [c_char; MAX_FORMAT as usize] = [0; MAX_FORMAT as usize];
                let mut buff: [c_char; MAX_ITEM] = [0; MAX_ITEM];
                arg += 1;
                if arg > top {
                    luaL_error!(L, "missing argument #{}", arg);
                }
                let mut format_item_size: usize = 0;
                strfrmt = scanformat(L, strfrmt, form.as_mut_ptr(), &mut format_item_size);
                let format_indicator = *strfrmt;
                strfrmt = strfrmt.add(1);
                match format_indicator as u8 {
                    b'c' => {
                        let count = snprintf(
                            buff.as_mut_ptr(),
                            buff.len(),
                            form.as_ptr(),
                            lua_l_checknumber(L, arg) as c_int,
                        );
                        lua_l_addlstring(&mut b, buff.as_ptr(), count as usize);
                        continue; // skip the 'luaL_addlstring' at the end
                    }
                    b'd' | b'i' => {
                        let value: i64 = if lua_isinteger_64!(L, arg) {
                            lua_l_checkinteger_64(L, arg)
                        } else {
                            lua_l_checknumber(L, arg) as i64
                        };
                        add_int_64_format(&mut form, format_indicator, format_item_size);
                        snprintf(buff.as_mut_ptr(), buff.len(), form.as_ptr(), value);
                    }
                    b'o' | b'u' | b'x' | b'X' => {
                        let v: u64 = if lua_isinteger_64!(L, arg) {
                            lua_l_checkinteger_64(L, arg) as u64
                        } else {
                            let arg_value = lua_l_checknumber(L, arg);
                            if arg_value < 0.0 {
                                (arg_value as i64) as u64
                            } else {
                                arg_value as u64
                            }
                        };
                        add_int_64_format(&mut form, format_indicator, format_item_size);
                        snprintf(buff.as_mut_ptr(), buff.len(), form.as_ptr(), v);
                    }
                    b'e' | b'E' | b'f' | b'g' | b'G' => {
                        snprintf(
                            buff.as_mut_ptr(),
                            buff.len(),
                            form.as_ptr(),
                            lua_l_checknumber(L, arg),
                        );
                    }
                    b'q' => {
                        addquoted(L, &mut b, arg);
                        continue; // skip the 'luaL_addlstring' at the end
                    }
                    b's' => {
                        let mut l: usize = 0;
                        let s = lua_l_checklstring(L, arg, &mut l);
                        // no precision and string too long to format, or no format necessary
                        if form[2] == 0
                            || (strchr(form.as_ptr(), b'.' as c_int).is_null() && l >= 100)
                        {
                            lua_l_addlstring(&mut b, s, l);
                            continue; // skip the 'luaL_addlstring' at the end
                        } else {
                            snprintf(buff.as_mut_ptr(), buff.len(), form.as_ptr(), s);
                        }
                    }
                    b'*' => {
                        // %* is parsed above, so if we got here we must have %...*
                        luaL_error!(L, "'%*' does not take a form");
                    }
                    _ => {
                        // also treat cases 'pnLlh'
                        luaL_error!(
                            L,
                            "invalid option '%{}' to 'format'",
                            *(strfrmt.offset(-1)) as u8 as char
                        );
                    }
                }
                let blen = core::ffi::CStr::from_ptr(buff.as_ptr()).to_bytes().len();
                lua_l_addlstring(&mut b, buff.as_ptr(), blen);
            }
        }
    }

    lua_l_pushresult(&mut b);
    1
}