之前使用nodemcu时萌发了一直Lua至STM32的想法。后来项目需要,便花了2天移植。
    Lua移植大约需要67K的rom,所以STM32F103C8无法使用,至少得103CB才行。
1. 官网下载源文件
http://www.lua.org/download.html 我下载的 是当前最新版本--5.3.5。
2. 解压,源代码位于src文件夹,去掉lua.c 和 luac.c 。
3. lauxlib.c:把l_alloc 函数调用的 free 和 realloc替换为自定义的内存管理函数。也可以不修改,但后面heap size需要改大。
4. linit.c:可以利用数组loadedlibs,把一些不要的功能去掉(注释),比如luaopen_os,luaopen_io,luaopen_debug。也可以添加自己的功能库(搜索:在LUA库中添加自己的函数库)。
5. luaconf.h:看文件开头的说明,使能 #define LUA_32BITS ,文件中有文件操作的目录定义LUA_ROOT,LUA_PATH_DEFAULT等,可以不管。
6. 修改单片机启动文件里的堆栈大小。 stack size设为4K; heap size暂时不修改。
7. 在需要使用Lua的的文件里加入头文件

#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"

编译,如出现未定义函数,自己添加定义,如time();如果出现如stdio_XXX错误之类,应该是串口映射相关问题。这时,选择使用microLib。并加入以下代码。
//USART1为映射到串口1.可以自己修改其他串口

int fputc(int ch, FILE *f)
{
    USART_SendData(USART1, (uint8_t) ch);

    while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET) {}    
   
    return ch;
}

int GetKey (void)  { 

    while (!(USART1->SR & USART_FLAG_RXNE));

    return ((int)(USART1->DR & 0x1FF));
}
//end

尝试不使用microLib,结果各种问题,所以还是使用microLib。 8.开始使用: 

lua_State *L = NULL;

L =luaL_newstate();  //创建Lua状态机   
if(L == NULL)
{
    printf("LUA ERR!");
    while(1){};//死循环
}
printf("LUA INIT OK!");
luaL_openlibs(L);//注册各种库函数
//end

其中luaL_openlibs()是逐个调用linit.c中数组loadedlibs记录的库。
linit.c文件如下:

/*
** these libs are loaded by lua.c and are readily available to any Lua
** program
*/
static const luaL_Reg loadedlibs[] = 
{
  {"_G", luaopen_base},
  {LUA_LOADLIBNAME, luaopen_package},
  {LUA_COLIBNAME, luaopen_coroutine},
  {LUA_TABLIBNAME, luaopen_table},
  //{LUA_IOLIBNAME, luaopen_io},
  //{LUA_OSLIBNAME, luaopen_os},
  {LUA_STRLIBNAME, luaopen_string},
  {LUA_MATHLIBNAME, luaopen_math},
  {LUA_UTF8LIBNAME, luaopen_utf8},
  {LUA_DBLIBNAME, luaopen_debug},
#if defined(LUA_COMPAT_BITLIB)
  {LUA_BITLIBNAME, luaopen_bit32},
#endif
  {NULL, NULL}
};

LUALIB_API void luaL_openlibs (lua_State *L) 
{
  const luaL_Reg *lib;
  /* "require" functions from 'loadedlibs' and set results to global table */
  for (lib = loadedlibs; lib->func; lib++) 
  {
    luaL_requiref(L, lib->name, lib->func, 1);
    lua_pop(L, 1);  /* remove lib */
  }
}

9.运行lua脚本
/*文件形式加载运行*/
luaL_dofile(L,"文件路径");
如果移植的时候没有将使用文件IO的库加入,也可以使用自己的方式,读取脚本数据,然后写入数组buf中,然后:
/*字符变量形式*/
luaL_dostring(L,buf);
/*关闭lua*/
lua_close(L);

这时已经成功把Lua解析器移植进STM32,可以运行Lua脚本了。不过单纯的Lua解析器在项目中没什么意义。

下面讲讲Lua与C函数的交互使用。一般给项目做功能扩展,都是Lua调用C函数。譬如想通过Lua脚本去控制单片机的某个外设,可将专门编写相关控制函数,并注册到Lua,需要的时候便可以在脚本中调用。

10.lua脚本调用自定义C函数
每个版本的使用方法好像都有些不一样。我使用的是当前最新的版本(5.3.5)
可以使用函数(其实是宏)lua_register(L,name,func)注册自定义的C函数,在lua在调用name(参数列表);func为实际的C函数。
并且,func函数形式必须类似为

static int func1(lua_State *L)
{
    //luaL_checknumber检测参数是否为数字
    float  val=sin(luaL_checknumber(L, 1));
    //压入结果
    lua_pushnumber(L,val ); 

    return 1; //结果的个数,这里为1
}
//如无需返回值,则更加简单
static int func2(lua_State *L)
{
    //luaL_checkstring检测参数是否为字符串并返回地址
      printf(luaL_checkstring(L,1));
      LCD_ShowString(30,60+20,200,16,16,"LUA:show");

    return 1; //结果的个数,这里为1
}

这种逐个函数注册的方式,虽然方便,但是比较乱。可以仿照库的方式,实现注册。
(1).新建一个C文件,以库名命名文件,这是必须的。
(2).添加需要自定义的功能C函数。
(3).定义一个luaL_Reg结构数组,把上面的C函数列入,并以字符串的方式命名,比如下面:

static const luaL_Reg mylib[] = 
{
    {"my_cos",   mymath_cos},
    {"my_sin",   mymath_sin},
    {NULL, NULL}
};

(4).添加open函数luaopen_XXX,XXX    为库名。

LUALIB_API int luaopen_mylib (lua_State *L) 
{
  luaL_newlib(L, mylib);
  return 1;
}

(5).这时,有两种选择。可以按以下注册函数

luaL_requiref(L,"mylib",luaopen_mylib,1);
lua_pop(L,1);

也可以在linit.c的loadedlibs数组中添加{"mylib",luaopen_mylib}。记得要保持数组后面的{NULL, NULL};如编译不过,可在lualib.h中添加函数声明。
至此,完成移植与基本使用。