高云FPGA系列教程(10):letter-shell移植

2023-09-21 22:44:09

本文是高云FPGA系列教程第10篇文章

shell,中文是外壳的意思,就是操作系统的外壳。通过shell命令可以操作和控制操作系统,比如Linux中的Shell命令就包括ls、cd、pwd等等。总结来说,Shell是一个命令解释器,它通过接受用户输入的Shell命令来启动、暂停、停止程序的运行或对计算机进行控制。

嵌入式平台可以基于串口实现shell功能,通过对串口命令的解析,可以执行相应的函数,查询变量的值等等。

本文介绍letter-shell开源shell库在高云GW1NSR-4C ARM处理器上的移植和应用。

letter-shell简介

letter-shell,一个功能强大的嵌入式shell,由标准C语言开发,可以在各种嵌入式平台上使用,可以通过命令行来执行函数,查询变量的值等等,支持裸机运行或RTOS运行,最新的发布版本是v3.1.2,letter-shell有以下功能:

  • 命令自动补全
  • 快捷键功能定义
  • 命令权限管理
  • 用户管理
  • 变量支持
  • 代理函数和参数代理解析

代码完全开源,并遵循MIT开源协议,Github收获近1K Star。

开源地址:

https://github.com/NevermindZZT/letter-shell

作者的主页地址:

https://nevermindzzt.github.io/

目前还保持更新状态,最近的一次提交是2023.07.25。

letter-shell源码获取

打开上文中letter-shell的开源地址,直接下载最新版本的Release代码,

或者,通过Git命令获取目前最新的代码,

$ git clone https://github.com/NevermindZZT/letter-shell.git --depth=1

Cloning into 'letter-shell'...
remote: Enumerating objects: 72, done.
remote: Counting objects: 100% (72/72), done.
remote: Compressing objects: 100% (66/66), done.
remote: Total 72 (delta 3), reused 35 (delta 2), pack-reused 0
Receiving objects: 100% (72/72), 783.28 KiB | 344.00 KiB/s, done.
Resolving deltas: 100% (3/3), done.

src目录中就是letter-shell的源文件,demo文件夹下是基于ESP32和STM32的移植示例代码。

letter-shell移植

首先把src文件夹的所有文件复制到GW1NSR-4C Keil工程的用户目录下,并新建两个接口文件:shell_port.cshell_port.h,用来对接shell库。

把这些文件都导入到我们的工程中,并包含头文件路径。

shell_cfg.h文件通过宏定义,可以实现功能的配置,非常灵活。

在shell_port.c文件中实现shell_write函数(串口发送字符串),并进行shell初始化:

#include "shell.h"
#include "drv_uart.h"

Shell shell;
char shellBuffer[512];

short userShellWrite(char *data, unsigned short len)
{
    UART_SendString(UART0, data);
    return len;
}

void userShellInit(void)
{
    shell.write = userShellWrite;

    shellInit(&shell, shellBuffer, 512);
}

对于裸机移植,不用实现shell read函数,只需要实现write函数。

并在shellport.h文件中进行声明:

#ifndef __SHELL_PORT_H__
#define __SHELL_PORT_H__

#include "shell.h"

extern Shell shell;

void userShellInit(void);

#endif

然后在串口接收中断服务函数,每接收到一字节数据调用shellHandler函数。

void UART0_Handler(void)
{
	char rx = 0;
	
	if(UART_GetRxIRQStatus(UART0) == SET)
	{
		rx = UART_ReceiveChar(UART0);
        shellHandler(&shell, rx);
	}
	
	UART_ClearRxIRQ(UART0);
}

主程序初始化时调用userShellInit函数,

#include "main.h"

int main(void)
{
	delay_init();
	uart0_init(115200); //enable rx interrupt
    
    userShellInit();
    
    while(1)
    {

	}
}

重新编译生成bin文件,并下载到开发板,打开串口终端,如SercureCRT,可以看到串口输出如下信息,说明移植成功,按下tab键,会提示当前支持的一些命令:

函数和变量应用示例

移植成功之后,我们来演示函数和变量的调用,即通过在终端输入函数名和参数可以直接执行函数,输入变量名可以直接打印变量的实时值。

定义一些变量和函数,并注册到shell命令解析列表中。

#include "main.h"

char str[100] = "Hello GoWin GW1NSR-4C (TangNano 4K)";
int cnt = 0;

int func(int i, char ch, char *str)
{
    printf("input int: %d, char: %c, string: %s\r\n", i, ch, str);
    return 1;
}
//获取系统频率
int get_sysclk(void)
{
	printf("SystemCoreClock = %d\r\n", SystemCoreClock);
	printf("APB1 CLK = %d\r\n", PCLK1);
	printf("APB2 CLK = %d\r\n", PCLK2);
	printf("AHB CLK  = %d\r\n", HCLK);
    return 2;
}

SHELL_EXPORT_CMD(SHELL_CMD_PERMISSION(0)|SHELL_CMD_TYPE(SHELL_TYPE_CMD_FUNC), get_sysclk, get_sysclk, test);
SHELL_EXPORT_CMD(SHELL_CMD_PERMISSION(0)|SHELL_CMD_TYPE(SHELL_TYPE_CMD_FUNC), func, func, test);
SHELL_EXPORT_VAR(SHELL_CMD_PERMISSION(0)|SHELL_CMD_TYPE(SHELL_TYPE_VAR_INT), cnt, &cnt, test);
SHELL_EXPORT_VAR(SHELL_CMD_PERMISSION(0)|SHELL_CMD_TYPE(SHELL_TYPE_VAR_STRING), str, str, test);

int main(void)
{
	delay_init();
	uart0_init(115200); //enable rx interrupt
    
	printf("SystemCoreClock = %d\r\n", SystemCoreClock);
	printf("APB1 CLK = %d\r\n", PCLK1);
	printf("APB2 CLK = %d\r\n", PCLK2);
	printf("AHB CLK  = %d\r\n", HCLK);
	printf("Hello GW1NSR-4C SoC(ARM Cortex-M3)\r\n");
    printf("letter-shell Example\r\n");
    userShellInit();
    
    while(1)
    {
        delay_ms(1000);
        cnt++;
	}
}

下载,运行。

在终端中直接输入对应的函数名即可直接运行函数,如果函数带参数,还可以在后面输入参数,参数类型支持整形、字符、字符串等多种类型,而且运行结束可以看到函数的返回值,输入变量的名字,可以直接获取到当前实时值,变量类型值整形和字符、字符串多种类型,非常强大、方便。

本文是高云FPGA系列教程第10篇文章

更多推荐

Spring 6.0 新特性

文章目录Spring的发展历史AOTGraalVMSpringBoot实战AOTRuntimeHints案例分析RuntimeHintsRegistrarSpringBoot中AOT核心代码Spring的发展历史AOTSpring6.0的新特性AheadofTime(AOT)编译是一种技术,可以提前将Spring应用程

Jmeter常用线程组设置策略

一、前言​在JMeter压力测试中,我们时常见到的几个场景有:单场景基准测试、单场景并发测试、单场景容量测试、混合场景容量测试、混合场景并发测试以及混合场景稳定性测试在本篇文章中,我们会用到一些插件,在这边先给大家列出:​CustomThreadGroups插件PS:在我们正式测试中,统一使用非GUI界面运行,只有在调

对比接口测试工具在自动化测试优缺点:Jmeter、Python、Postman

一、JMeter总结:适合对代码不敏感的使用人员,不会代码也可以完成接口自动化,设计框架。适合紧急迭代的项目。JMeter接口测试的优势小巧轻量级,并且开源免费,社区接受度高,比较容易入门支持多协议,并提供了比较高级的扩展能力,允许自己定义和扩展新的协议支持,比如扩展支持阿里提供的Dubbo协议的JMeter插件等学习

JMeter 常见函数讲解

当使用JMeter进行性能测试或负载测试时,函数是一个非常有用的工具,可以帮助生成动态的测试数据或处理测试结果。下面是一些常用的JMeter函数的详细讲解和并列示例:1、__threadNum:返回当前线程的编号。可以在测试过程中用于生成唯一的标识符或动态数据。生成唯一的用户名:${__threadNum}-user动

MySQL

数据库分两大类:关系型数据SQL非关系型数据库NoSQL关系型数据库典型代表:MySQLMariaDBPostgreSQL(pgsql)OracleSQLServerDB2国产数据库:阿里云RDB华为高斯阿里Oceanbase腾讯TDBA1.SQLSQL(StructuredQueryLanguage)是具有数据操纵和

Spring-动态代理深入了解

😀前言本篇的Spring-AOP系类文章第二篇扩展了Spring-动态代理然后开发了简易的AOP类🏠个人主页:尘觉主页🧑个人简介:大家好,我是尘觉,希望我的文章可以帮助到大家,您的满意是我的动力😉😉在csdn获奖荣誉:🏆csdn城市之星2名⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣⁣💓

Nginx服务优化措施、网页安全与配置防盗链

目录一.优化Nginx二.隐藏/查看版本号隐藏版本号方法一:修改配置文件,关闭版本号隐藏版本号方法二:修改源码文件中的版本号,重新编译安装三.修改用户与组四.设置缓存时间五.日志切割脚本六.设置连接超时控制连接访问时间七.开启多进程八.配置网页压缩九.配置防盗链9.1.配置web源主机(192.168.47.103)9

【Python】PySpark 数据计算 ① ( RDD#map 方法 | RDD#map 语法 | 传入普通函数 | 传入 lambda 匿名函数 | 链式调用 )

文章目录一、RDD#map方法1、RDD#map方法引入2、RDD#map语法3、RDD#map用法4、代码示例-RDD#map数值计算(传入普通函数)5、代码示例-RDD#map数值计算(传入lambda匿名函数)6、代码示例-RDD#map数值计算(链式调用)一、RDD#map方法1、RDD#map方法引入在PyS

API网关是如何提升API接口安全管控能力的

API安全的重要性近几年,越来越多的企业开始数字化转型之路。数字化转型的核心是将企业的服务、资产和能力打包成服务(服务的形式通常为API,API又称接口,下文中提到的API和接口意思相同),从而让资源之间形成更强的连接和互动关系,释放原有资产的价值,提升企业的服务能力。企业数字化转型使得基于API的业务系统剧增,随之而

RK3568笔记分享之“如何挂载SPI FRAM铁电存储芯片”——飞凌嵌入式

对于做快速存储采集数据类产品的用户来说,在处理突发掉电情况时需要保存现有数据并避免数据丢失,这种情况下有很多种解决方案,铁电存储器(FRAM)就是个很好的选择。FRAM是一种具有快速写入速度的非易失性存储器,既可以进行非易失性数据存储,又可以像RAM一样操作。本文将借助飞凌嵌入式OK3568-C开发板来为大家介绍一种采

周界警戒AI算法+视频智能分析在安全生产场景中的应用

长期以来,周界防范安防系统在大型园区、工厂、社区、机场、火车站站台、重点单位等领域应用较为广泛和常见。随着AI人工智能等新兴技术的快速发展与落地应用,通过AI智能检测与视频智能分析技术,现代化的周界安防系统可以做到全天候快速、准确地发现入侵等异常事件,并及时报警遏制。今天我们来介绍下旭帆科技安全生产周界警戒AI算法的具

热文推荐