mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
224 lines
7.8 KiB
C
224 lines
7.8 KiB
C
/*
|
||
* Copyright (C) 2018 Freie Universität Berlin
|
||
*
|
||
* This file is subject to the terms and conditions of the GNU Lesser
|
||
* General Public License v2.1. See the file LICENSE in the top level
|
||
* directory for more details.
|
||
*/
|
||
/**
|
||
* @ingroup pkg_lua
|
||
* @file
|
||
* @{
|
||
*
|
||
* @brief Convenience functions for running Lua code.
|
||
* @author Juan Carrano <j.carrano@fu-berlin.de>
|
||
*
|
||
* This functions make it easy to create and use new Lua context:
|
||
* It provides:
|
||
*
|
||
* - Easy to use routines for executing modules as scrips.
|
||
* - Control over which modules get loaded.
|
||
* - Support for using a local heap allocator.
|
||
* - Out of memory handling via setjmp/longjmp.
|
||
*
|
||
* This library is not strictly required, as all of the functionality could be
|
||
* implemented in terms of the public lua api, but it covers most of the use
|
||
* cases, and thus avoids code repetition in applications.
|
||
*
|
||
*/
|
||
|
||
#ifndef LUA_RUN_H
|
||
#define LUA_RUN_H
|
||
|
||
#include <stdint.h>
|
||
|
||
#include "lua.h"
|
||
|
||
#ifdef __cplusplus
|
||
extern "C" {
|
||
#endif
|
||
|
||
/**
|
||
* Convert a library index into a bit mask.
|
||
*/
|
||
#define LUAR_LOAD_FLAG(n) (((uint16_t)1) << (n))
|
||
|
||
/**
|
||
* Order in which the builtin libraries are loaded.
|
||
*/
|
||
enum LUAR_LOAD_ORDER {
|
||
LUAR_LOAD_O_BASE,
|
||
LUAR_LOAD_O_PACKAGE,
|
||
LUAR_LOAD_O_CORO,
|
||
LUAR_LOAD_O_TABLE,
|
||
LUAR_LOAD_O_IO,
|
||
LUAR_LOAD_O_OS,
|
||
LUAR_LOAD_O_STRING,
|
||
LUAR_LOAD_O_MATH,
|
||
LUAR_LOAD_O_UTF8,
|
||
LUAR_LOAD_O_DEBUG,
|
||
LUAR_LOAD_O_ALL,
|
||
};
|
||
|
||
/** Load the base globals (_G) */
|
||
#define LUAR_LOAD_BASE LUAR_LOAD_FLAG(LUAR_LOAD_O_BASE)
|
||
/** Load ´package´ */
|
||
#define LUAR_LOAD_PACKAGE LUAR_LOAD_FLAG(LUAR_LOAD_O_PACKAGE)
|
||
/** Load ´coroutine´ */
|
||
#define LUAR_LOAD_CORO LUAR_LOAD_FLAG(LUAR_LOAD_O_CORO)
|
||
/** Load ´table´ */
|
||
#define LUAR_LOAD_TABLE LUAR_LOAD_FLAG(LUAR_LOAD_O_TABLE)
|
||
/** Load ´io´ */
|
||
#define LUAR_LOAD_IO LUAR_LOAD_FLAG(LUAR_LOAD_O_IO)
|
||
/** Load ´os´ */
|
||
#define LUAR_LOAD_OS LUAR_LOAD_FLAG(LUAR_LOAD_O_OS)
|
||
/** Load ´string´ */
|
||
#define LUAR_LOAD_STRING LUAR_LOAD_FLAG(LUAR_LOAD_O_STRING)
|
||
/** Load ´math´ */
|
||
#define LUAR_LOAD_MATH LUAR_LOAD_FLAG(LUAR_LOAD_O_MATH)
|
||
/** Load ´utf8´ */
|
||
#define LUAR_LOAD_UTF8 LUAR_LOAD_FLAG(LUAR_LOAD_O_UTF8)
|
||
/** Load ´debug´ */
|
||
#define LUAR_LOAD_DEBUG LUAR_LOAD_FLAG(LUAR_LOAD_O_DEBUG)
|
||
|
||
/* TODO: maybe we can implement a "restricted base" package containing a subset
|
||
* of base that is safe. */
|
||
|
||
#define LUAR_LOAD_ALL (0xFFFF) /** Load all standard modules */
|
||
#define LUAR_LOAD_NONE (0x0000) /** Do not load any modules */
|
||
|
||
/** Errors that can be raised when running lua code. */
|
||
enum LUAR_ERRORS {
|
||
LUAR_EXIT, /** The program exited without error. */
|
||
LUAR_STARTUP_ERR, /** Error setting up the interpreter. */
|
||
LUAR_LOAD_ERR, /** Error while loading libraries. */
|
||
LUAR_NOMODULE, /** The specified module could not be found. */
|
||
LUAR_COMPILE_ERR, /** The Lua code failed to compile. */
|
||
LUAR_RUNTIME_ERR, /** Error in code execution. */
|
||
LUAR_MEMORY_ERR, /** Lua could not allocate enough memory */
|
||
LUAR_INTERNAL_ERR /** Error inside the Lua VM.
|
||
* Right now, if this happens, you may leak memory from
|
||
* the heap. If your program is the only one using the
|
||
* dynamic allocation, just clean the heap.
|
||
*/
|
||
};
|
||
|
||
/**
|
||
* Human-readable description of the errors
|
||
*/
|
||
extern const char *lua_riot_str_errors[];
|
||
|
||
/**
|
||
* Return a string describing an error from LUAR_ERRORS.
|
||
*
|
||
* @param errn Error number as returned by lua_riot_do_buffer() or
|
||
* lua_riot_do_buffer()
|
||
*
|
||
* @return A string describing the error, or "Unknown error".
|
||
*/
|
||
LUALIB_API const char *lua_riot_strerror(int errn);
|
||
|
||
/**
|
||
* Initialize a lua state and set the panic handler.
|
||
*
|
||
* @todo Use a per-state allocator
|
||
*
|
||
* @param memory Pointer to memory region that will be used as heap for
|
||
* the allocator. Currently this functionality is not
|
||
* supported and this must be set to NULL.
|
||
* @param mem_size Size of the memory region that will be used as heap.
|
||
* Currently this functionality is not supported and this
|
||
* must be set to 0.
|
||
* @param panicf Function to be passed to lua_atpanic. If set to NULL,
|
||
* a generic function that does nothing will be used.
|
||
*
|
||
* @return the new state, or NULL if there is a memory allocation error.
|
||
*/
|
||
LUALIB_API lua_State *lua_riot_newstate(void *memory, size_t mem_size,
|
||
lua_CFunction panicf);
|
||
|
||
/**
|
||
* Terminate the lua state.
|
||
*
|
||
* You must call this function if you want the finalizers (the __gc metamethods)
|
||
* to be called.
|
||
*/
|
||
#ifndef LUA_DEBUG
|
||
#define lua_riot_close lua_close
|
||
#else
|
||
#define lua_riot_close luaB_close
|
||
#endif /* LUA_DEBUG */
|
||
|
||
/**
|
||
* Open builtin libraries.
|
||
*
|
||
* This is like luaL_openlibs but it allows selecting which libraries will
|
||
* be loaded.
|
||
*
|
||
* Libraries are loaded in the order specified by the LUAR_LOAD_ORDER enum. If
|
||
* there is an error the load sequence is aborted and the index of the library
|
||
* that failed is reported.
|
||
*
|
||
* If debugging is enabled (compile with the LUA_DEBUG macro), then the test
|
||
* library will be unconditionally loaded.
|
||
*
|
||
* @param L Lua state
|
||
* @param modmask Binary mask that indicates which modules should be
|
||
* loaded. The mask is made from a combination of the
|
||
* LUAR_LOAD_* macros.
|
||
*
|
||
* @return The index of the library that failed to load, or LUAR_LOAD_O_ALL
|
||
* if all libraries were loaded.
|
||
*/
|
||
LUALIB_API int lua_riot_openlibs(lua_State *L, uint16_t modmask);
|
||
|
||
/**
|
||
* Initialize the interpreter and run a built-in module in protected mode.
|
||
*
|
||
* In addition to running code in protected mode, this also sets a panic
|
||
* function that long-jumps back to this function, in case there is an internal
|
||
* interpreter error (LUAR_INTERNAL_ERR).
|
||
* Right now the only things that the application can are either to abort(),
|
||
* or to manually reset the heap (only if there's no other thread using it).
|
||
*
|
||
* @param modname name of the module.
|
||
* @param memory @see lua_riot_newstate()
|
||
* @param mem_size @see lua_riot_newstate()
|
||
* @param modmask @see lua_riot_newstate()
|
||
* @param[out] retval Value returned by the lua code, if it is a number,
|
||
* or zero if no value is returned or the value is not
|
||
* a number. If retval is null, the value is not stored.
|
||
*
|
||
* @return An error code ( @see LUAR_ERRORS). LUAR_EXIT indicates no error.
|
||
*/
|
||
LUALIB_API int lua_riot_do_module(const char *modname, void *memory, size_t mem_size,
|
||
uint16_t modmask, int *retval);
|
||
|
||
/**
|
||
* Initialize the interpreter and run a user supplied buffer in protected mode.
|
||
*
|
||
* Only text data (i.e. lua source code) can be loaded by this function. The
|
||
* lua interpreter is not robust against corrupt binary code.
|
||
*
|
||
* @see lua_riot_do_module() for more information on internal errors.
|
||
*
|
||
* @param buf Text data (lua source code).
|
||
* @param buflen Size of the text data in bytes. If buf is
|
||
* a zero-terminated string, the zero must not be counted.
|
||
* @param memory @see lua_riot_newstate()
|
||
* @param mem_size @see lua_riot_newstate()
|
||
* @param modmask @see lua_riot_newstate()
|
||
* @param[out] retval @see lua_riot_do_module()
|
||
* @return @see lua_riot_do_module().
|
||
*/
|
||
LUALIB_API int lua_riot_do_buffer(const uint8_t *buf, size_t buflen, void *memory,
|
||
size_t mem_size, uint16_t modmask, int *retval);
|
||
|
||
#ifdef __cplusplus
|
||
extern "C" }
|
||
#endif
|
||
|
||
/** @} */
|
||
|
||
#endif /* LUA_RUN_H */
|