当前位置: 首页 > news >正文

LUA移植到STM32F4,移植REPL,通过RTT Viewer交互

概述

站内移植LUA多数是使用C函数调用LUA,并没有移植REPL交互端口
本文将REPL也移植进去,做了简单的适配

LUA源码使用标准C库函数,如fgets,fwrite等,在嵌入式环境中要使用fgets,fwrite等C库函数,需要做的工作就是重定向。

本文重定向了STDIN和STDOUT数据流到J-Link RTT Viewer,可以通过 RTT Viewer向LUA虚拟机进行交互。
在这里插入图片描述

环境

MCU:STM32F407, 192KB RAM, 1MFLASH。建议运行平台至少有256KBRAM,256KB的FLASH,否则加载lib的时候会爆内存或者FLASH。
KEIL:527
编译器:AC6

准备工程

从https://www.lua.org/download.html网站下载LUA源码,在KEIL中新建一个LUA文件夹,将所有文件添加到里面。luac.c不要添加进去,这个文件是用来编译lua脚本的,我们不需要。

准备SEGGER RTT打印相关文件

新建一个空文件syscall.c,后面的C库系统调用函数我们会写入到此文件中。

处理好所有文件的头文件,包含路径问题。
在这里插入图片描述

对接C库系统调用函数

以下函数的编写参考了如下资源

  • <rt_sys.h>文件定义了函数头文件,里面还有函数功能和返回值的描述
  • Arm® C and C++ Libraries and Floating-Point Support User Guide:一些函数描述

主要实现了

  • 关闭半主机模式
  • _sys_open:打开文件,返回STDIN,STDOUT,STDERR的文件描述符,,普通文件流不处理
  • _sys_write:写文件,向STDOUT写入的数据流流向SEGGER RTT,普通文件流不处理
  • _sys_read:读文件,SEGGER RTT读取的数据流向STDIN,普通文件流不处理、
  • _sys_istty:判断文件描述符是否为终端
  • time:事件相关的函数,对接的是hal_gettick,返回系统上电运行了多少ms
#include <rt_sys.h>
#include <stdio.h>
#include <string.h>
#include <time.h>#include "SEGGER_RTT.h"
#include "main.h"//关闭半主机模式
/********************************************************************************/
#if defined(__clang__)__asm(".global __use_no_semihosting\n\t");
#elif defined(__CC_ARM)#pragma import(__use_no_semihosting)
#endif#define STDOUT      0x00000001
#define STDIN       0x00000002
#define STDERR      0x00000003 const char __stdin_name[]  = "STDIN";
const char __stdout_name[] = "STDOUT";
const char __stderr_name[] = "STDERR";FILEHANDLE _sys_open(const char *pcFile, int openmode)
{if(0 == strncmp(pcFile, __stdin_name,  strlen(__stdin_name)))  return STDIN;if(0 == strncmp(pcFile, __stdout_name, strlen(__stdout_name))) return STDOUT;if(0 == strncmp(pcFile, __stderr_name, strlen(__stderr_name))) return STDERR;//pcFile    :文件路径//openmode  :文件打开模式//返回值    :文件描述符return 0;
}int _sys_close(FILEHANDLE fh)
{return 0;
}int _sys_write(FILEHANDLE fh, const unsigned char * buf, unsigned len, int mode)
{if (fh == STDOUT){SEGGER_RTT_Write(0, (const char*)buf, len);		return 0;}return 0;
}int _sys_read(FILEHANDLE fh, unsigned char * buf, unsigned len, int mode)
{
//读取一行数据,回车结束。读取完毕之后在字符串末尾添加结束符static int count_p = 0;if (fh == STDIN){count_p = 0;buf[count_p] = SEGGER_RTT_WaitKey();while(buf[count_p] != '\n'){count_p++;buf[count_p] = SEGGER_RTT_WaitKey();}buf[count_p + 1] = '\0';return 0;}return 0;  //EOF
}void _ttywrch(int ch)
{fputc(ch, stdout); // stdoutfflush(stdout);
}int _sys_istty(FILEHANDLE fh)
{return (fh==STDIN || fh==STDOUT || fh==STDERR);  
}int _sys_seek(FILEHANDLE fh, long pos)
{return 0;    
}int _sys_ensure(FILEHANDLE fh)
{return 0;    
}long _sys_flen(FILEHANDLE fh)
{return 0;    
}int _sys_tmpnam(char * name, int sig, unsigned maxlen)
{return 0;
}void _sys_exit(int returncode)   /* never returns */
{
}char *_sys_command_string(char * cmd, int len)
{return 0;    
}
int remove(const char *filename)
{return 0;    
}
int system(const char *string)
{return 0;     
}
int rename(const char *old, const char *new)
{return 0;
}
time_t time(time_t *timer)
{return HAL_GetTick();
}
clock_t clock(void)
{return 0;  
}

修改LUA源码

LUA源码中操作行数据使用fgets和puts,这个函数我的对接始终有问题,这里更改为fread和fwrite函数
在luaconf.h末尾添加如下代码

/* =================================================================== *//*
** Local configuration. You can use this space to add your redefinitions
** without modifying the main part of the file.
*/
#define LUA_MAXINPUT                128
#define lua_readline(L,b,p)         ( fread(b, 1, LUA_MAXINPUT, stdin) != 0)
#define lua_initreadline(L)         ( (void)L              )
#define lua_saveline(L,line)        { (void)L; (void)line; }
#define lua_freeline(L,b)           { (void)L; (void)b;    }#define lua_writestring(s,l)        fwrite(s, 1, l, stdout)
#define lua_writeline()             fwrite("\n", 1, 1, stdout)
#define lua_writestringerror(...)   printf(__VA_ARGS__)

lua.c中已经有一个main函数,我们需要将这个main函数改名为lua_main,在keil中的main函数调用lua_main来启动LUA

int lua_main (int argc, char **argv) {	//修改函数名int status, result;lua_State *L = luaL_newstate();  /* create state */if (L == NULL) {l_message(argv[0], "cannot create state: not enough memory");return EXIT_FAILURE;}lua_gc(L, LUA_GCSTOP);  /* stop GC while building state */lua_pushcfunction(L, &pmain);  /* to call 'pmain' in protected mode */lua_pushinteger(L, argc);  /* 1st argument */lua_pushlightuserdata(L, argv); /* 2nd argument */status = lua_pcall(L, 2, 1, 0);  /* do the call */result = lua_toboolean(L, -1);  /* get result */report(L, status);lua_close(L);return (result && status == LUA_OK) ? EXIT_SUCCESS : EXIT_FAILURE;
}

lua .h中增加lua_main的函数声明

int lua_main (int argc, char **argv);

启动LUA虚拟机

main函数中,增加如下代码
这里我们要给lua_main 传递两个假参数,如下

  int   fake_argc = 1;char *fake_argv = NULL;lua_main (fake_argc, &fake_argv);

启动

启动前要先配置好RTT VIEWER,复位启动即可.
测试指令如下

  < _VERSION< print("hello world")< print("abc".."666")< print("system run "..os.time().." msec")

在这里插入图片描述
错误指令和提示如下

  < print("system run "..XXX.time().." msec")

在这里插入图片描述

TODO&其他

工程参考:https://gitee.com/nwwhhh/stm32f407
TODO:对接文件函数,调用本地文件

在这里插入图片描述

相关文章:

  • GraogGNSSLib学习
  • 医学人工智能在“免疫组化”领域的最新研究进展|顶刊速递·24-06-19
  • 美业人专用宝藏系统、Java收银系统源码分享-美业SAAS系统的应用价值分析
  • 关于INCA的几个实用功能
  • Top10在线音频剪辑软件,你了解几款?(免费分享)
  • 前端技术回顾系列 11|TS 中一些实用概念
  • Android C++系列:函数知识知多少
  • Linux时间子系统7:sleep timer接口定时实现
  • Anti-human IL-10 mAb (12G8), biotin:Mabtech热销品
  • vue大作业-实现学校官网
  • 【杂记-浅谈Sequence Number/序列号】
  • 第三方美颜SDK开发详解:直播美颜工具的功能与技术实现
  • upload-labs第十二关教程
  • 【Redis】基于Redission实现分布式锁(代码实现)
  • macOS聚集搜索功能开启与关闭
  • “寒冬”下的金三银四跳槽季来了,帮你客观分析一下局面
  • 《Java编程思想》读书笔记-对象导论
  • 【EOS】Cleos基础
  • Effective Java 笔记(一)
  • JAVA 学习IO流
  • JAVA并发编程--1.基础概念
  • jQuery(一)
  • React的组件模式
  • React系列之 Redux 架构模式
  • Redis 中的布隆过滤器
  • uni-app项目数字滚动
  • Vue 动态创建 component
  • windows下使用nginx调试简介
  • Xmanager 远程桌面 CentOS 7
  • 闭包--闭包作用之保存(一)
  • 初探 Vue 生命周期和钩子函数
  • 基于HAProxy的高性能缓存服务器nuster
  • 简单实现一个textarea自适应高度
  • 理解 C# 泛型接口中的协变与逆变(抗变)
  • 配置 PM2 实现代码自动发布
  • 前端面试总结(at, md)
  • 前嗅ForeSpider教程:创建模板
  • 王永庆:技术创新改变教育未来
  • 问:在指定的JSON数据中(最外层是数组)根据指定条件拿到匹配到的结果
  • 在Unity中实现一个简单的消息管理器
  • ​决定德拉瓦州地区版图的关键历史事件
  • #if #elif #endif
  • (20050108)又读《平凡的世界》
  • (2009.11版)《网络管理员考试 考前冲刺预测卷及考点解析》复习重点
  • (26)4.7 字符函数和字符串函数
  • (c语言版)滑动窗口 给定一个字符串,只包含字母和数字,按要求找出字符串中的最长(连续)子串的长度
  • (Windows环境)FFMPEG编译,包含编译x264以及x265
  • (不用互三)AI绘画:科技赋能艺术的崭新时代
  • (附源码)springboot电竞专题网站 毕业设计 641314
  • (附源码)springboot工单管理系统 毕业设计 964158
  • (附源码)ssm高校志愿者服务系统 毕业设计 011648
  • (含笔试题)深度解析数据在内存中的存储
  • (四)stm32之通信协议
  • (原)Matlab的svmtrain和svmclassify
  • (转)Android学习系列(31)--App自动化之使用Ant编译项目多渠道打包