手把手教你写单片机shell (一、基本功能实现)

资料

这是我的单片机shell开源地址单片机shell(xcmd)

计划

  1. 基本功能实现手把手教你写单片机shell (一、基本功能实现)
  2. 实现快捷键功能
  3. 控制光标移动
  4. 实现历史记录
  5. 实现TAB自动补全
  6. 添加彩色字符串
  7. 添加常用扩展指令

简介

今天开始我们开始一步一步编写一个适合单片机运行的命令行工具,其功能类似linux上的超级终端。为了快速测试验证,我们使用的平台是arduino uno+putty,后续再尝试移植到其他平台。

实现

1. 测试putty串口行为

我们先来看一下串口终端软件,以putty为例。我们先编写第一个测试程序

void setup() {
  Serial.begin(115200);
}

void loop() {
  if(Serial.available())
  {
    char rcv = Serial.read();
    Serial.print("rcv:");
    Serial.println(rcv);
  }
}

在这里插入图片描述
可以看到再putty中当我们输入字符时会自动通过串口将是数据发送到单片机中,并且,显示的内容为单片机发过来的数据。总结

  • putty中按下一个按键就会发送给单片机一个对应的字符
  • putty并不会显示自己发送了什么字符,显示的是自己接收到的字符
2. 实现putty串口回显

修改程序,完成从putty接受字符并把字符打印到putty上

void setup() {
  Serial.begin(115200);
}

void loop() {
  if(Serial.available())
  {
    char rcv = Serial.read();
    Serial.print(rcv);
  }
}

在这里插入图片描述

3. 实现从putty中读取字符串

接下来从putty接受字符并把字符打印到putty上,并且当接收到回车换行时把接收到的字符组织成字符串,回车后打印这句字符串。

为了方便后续打印我们自定义实现printf函数

void my_print(char* fmt, ...)
{
	char buf[128] = {0};
	va_list ap;
	va_start(ap, fmt);
	vsnprintf(buf,128,fmt, ap);
	Serial.print(buf);
}

通过putty发送字符并以回车换行为结束,将获取到的字符组织成字符串。

/*用于从串口获取一条以回车换行结尾的字符串
*/
uint8_t get_line(char *line, uint8_t maxLen)
{
	char rcv_char;
	static uint8_t count = 0; /*用于记录除特殊字符外的其他有效字符的数量*/
	if (Serial.available())
	{
		rcv_char = Serial.read();
		if (count >= maxLen) /*长度超限*/
		{
			count = 0; /*清零计数器以便后续使用*/
			return 1;  /*返回有效标志*/
		}
		line[count] = rcv_char; /*记录数据*/
		switch (rcv_char)
		{
			case 0x08:
			case 0x7F: /*退格键或者删除键*/
			{
				if (count > 0)
				{
					count--; /*删除上一个接收到的字符*/
				}
			}
			break;

			case '\r':
			case '\n': /*接收到回车换行,证明已经收到一个完整的命令*/
			{
				line[count] = '\0'; /*添加字符串结束符,刚好可以去掉'\r'或者'\n'*/
				count = 0;			/*清零计数器以便后续使用*/
				return 1;			/*返回有效标志*/
			}
			break;

			default:
				count++;
		}
		Serial.print(rcv_char); /*把收到的字符输出到串口*/
	}
	return 0;
}

...详细代码在文末

void loop()
{
	if(get_line(g_line_buf, LINE_BUF_MAX_LEN))
	{
		/* 获得到指令 */
		if(strlen(g_line_buf)) /* 判断接受到的指令字符串是否为0 */
			my_print("\r\ncmd is:\"%s\"\r\n", g_line_buf);
		my_print("\r\n->");
	}
}

在这里插入图片描述

4. 分割字符串获取输入的参数

接下来我们需要将获得到的字符串按照空格分隔成多个字符串以便命令函数调用。

/* 参数解析函数 */
static int get_param(char* msg, char*delim, char* get[], int max_num)
{
	int i,ret;
	char *ptr = NULL;
	ptr = strtok(msg, delim);
	for(i=0; ptr!=NULL &&i<max_num; i++)
	{
		get[i] = ptr;
		ptr = strtok(NULL, delim);
	}
	ret = i;
	return ret;
}

...详细代码在文末

void loop()
{
	if(get_line(g_line_buf, LINE_BUF_MAX_LEN))
	{
		/* 获得到指令 */
		if(strlen(g_line_buf)) /* 判断接受到的指令字符串是否为0 */
		{
			char *argv[16];
			int argc = get_param(g_line_buf, " ", argv, 16);
			
			/*打印参数信息*/
			int i = 0;
			my_print("\r\nargc=%d ", argc);
			my_print("argv=[");
			for(i=0; i<(argc-1); i++)
			{
				my_print("%s,", argv[i]);
			}
			my_print("%s]\r\n", argv[i]);
		}
		my_print("\r\n->");
	}
}

在这里插入图片描述

5. 定义命令函数以及命令调用过程

我们仿照main函数定义int main(int argc, char** argv)我们的命令函数

typedef int(*cmd_func_t)(int argc, char**argv);

定义一个结构体用来描述命令

typedef struct /* 定义命令结构体 */
{
	char* name; /* 命令的名字 */
	cmd_func_t func; /* 执行函数 */
	char* help; /* 帮助文字 */
}cmd_t;

定义两个命令函数

int cmd_test(int argc, char**argv)
{
	int i = 0;
	my_print("argc=%d ", argc);
	my_print("argv=[");
	for(i=0; i<(argc-1); i++)
	{
		my_print("%s,", argv[i]);
	}
	my_print("%s]\r\n", argv[i]);
}

int cmd_help(int argc, char**argv)
{
	for(int i=0; i<g_num_cmd; i++)
	{
		my_print("%-10s %s\r\n", g_my_cmds[i].name, g_my_cmds[i].help);
	}
}

定义一个命令结构体数组用来存放自定义命令

cmd_t g_my_cmds[] = 
{
	{"test", cmd_test, "Show all params"},
	{"help", cmd_help, "Show this list"}
};
int g_num_cmd = sizeof(g_my_cmds)/sizeof(cmd_t);

通过步骤3中解析的字符串来循环查找与命令名匹配的命令函数,执行命令函数并传入参数。

for(int i=0; i<g_num_cmd; i++)
{
	if(strcmp(argv[0], g_my_cmds[i].name) == 0)
	{
		g_my_cmds[i].func(argc, argv);
	}
}

在这里插入图片描述

6. 总结

基本上到这里我们的单片机shell的基本功能已经实现,可以接收来自putty的输入,并解析输入的字符串运行相关的自定义命令。接下来我们将进入下一阶段,来实现一个操作更加友好的单片机shell。

最后贴上完整的代码

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>

#define LINE_BUF_MAX_LEN	(64)
char g_line_buf[LINE_BUF_MAX_LEN+1] = {0};

/***********************************************************/
/* 定义函数类型 */
typedef int(*cmd_func_t)(int argc, char**argv);
typedef struct /* 定义命令结构体 */
{
	char* name;
	cmd_func_t func;
	char* help;
}cmd_t;

int cmd_test(int argc, char**argv);
int cmd_help(int argc, char**argv);

cmd_t g_my_cmds[] = 
{
	{"test", cmd_test, "Show all params"},
	{"help", cmd_help, "Show this list"}
};
int g_num_cmd = sizeof(g_my_cmds)/sizeof(cmd_t);

int cmd_test(int argc, char**argv)
{
	int i = 0;
	my_print("argc=%d ", argc);
	my_print("argv=[");
	for(i=0; i<(argc-1); i++)
	{
		my_print("%s,", argv[i]);
	}
	my_print("%s]\r\n", argv[i]);
}

int cmd_help(int argc, char**argv)
{
	for(int i=0; i<g_num_cmd; i++)
	{
		my_print("%-10s %s\r\n", g_my_cmds[i].name, g_my_cmds[i].help);
	}
}

/*************************************************************/

void my_print(char* fmt, ...)
{
	char buf[128] = {0};
	va_list ap;
	va_start(ap, fmt);
	vsnprintf(buf,128,fmt, ap);
	Serial.print(buf);
}

static int get_param(char* msg, char*delim, char* get[], int max_num)
{
	int i,ret;
	char *ptr = NULL;
	ptr = strtok(msg, delim);
	for(i=0; ptr!=NULL &&i<max_num; i++)
	{
		get[i] = ptr;
		ptr = strtok(NULL, delim);
	}
	ret = i;
	return ret;
}

/*
*用于从串口获取一条以回车换行结尾的命令
*/
uint8_t get_line(char *line, uint8_t maxLen)
{
	char rcv_char;
	static uint8_t count = 0; /*用于记录除特殊字符外的其他有效字符的数量*/
	if (Serial.available())
	{
		rcv_char = Serial.read();
		if (count >= maxLen) /*长度超限*/
		{
			count = 0; /*清零计数器以便后续使用*/
			return 1;  /*返回有效标志*/
		}
		line[count] = rcv_char; /*记录数据*/
		switch (rcv_char)
		{
			case 0x08:
			case 0x7F: /*退格键或者删除键*/
			{
				if (count > 0)
				{
					count--; /*删除上一个接收到的字符*/
				}
			}
			break;

			case '\r':
			case '\n': /*接收到回车换行,证明已经收到一个完整的命令*/
			{
				line[count] = '\0'; /*添加字符串结束符,刚好可以去掉'\r'或者'\n'*/
				count = 0;			/*清零计数器以便后续使用*/
				return 1;			/*返回有效标志*/
			}
			break;

			default:
				count++;
		}
		Serial.print(rcv_char); /*把收到的字符输出到串口*/
	}
	return 0;
}

int match_cmd(int argc, char**argv)
{
	int i = 0;
	for(int i=0; i<g_num_cmd; i++)
	{
		if(strcmp(argv[0], g_my_cmds[i].name) == 0)
		{
			g_my_cmds[i].func(argc, argv);
		}
	}
	if(i == g_num_cmd)
	{
		my_print("cmd \"%s\" does not exist!!\r\n");
	}
}

void setup()
{
	Serial.begin(115200);
}

void loop()
{
	if(get_line(g_line_buf, LINE_BUF_MAX_LEN))
	{
		my_print("\r\n");
		/* 获得到指令 */
		if(strlen(g_line_buf)) /* 判断接受到的指令字符串是否为0 */
		{
			char *argv[16];
			int argc = get_param(g_line_buf, " ", argv, 16);

			match_cmd(argc, argv);
			
		}
		my_print("->");
	}
}
  • 3
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
学习单片机C程序设计是一项具有挑战性但也非常有趣的任务。下面我将用手把手的方式来你学习单片机C程序设计的程序。 首先,你需要准备一款可以编C程序的单片机开发板。常见的单片机开发板有Arduino、Raspberry Pi等。选择一款适合你的开发板,并确保你已经安装了相应的开发环境,如Arduino IDE或Raspberry Pi OS。 接下来,我们来学习C语言的基础知识。你可以通过阅读C语言程书籍或在线资源来学习C语言的基本语法、数据类型、运算符等。掌握好这些基础知识对于后续的单片机C程序设计至关重要。 然后,你可以从简单的实例开始编C程序。比如,点亮一个LED灯或在LCD屏幕上显示一些文字。你可以通过查阅开发板的说明书来了解如何连接电路和控制IO口,并通过C编程来实现你的想法。 在你编C程序的过程中,要注意一些常见的编程技巧和注意事项。比如,要注意变量的声明和作用域、函数的调用和参数传递、循环和条件语句的使用等。此外,还需要学会调试和排除程序中的错误,这对于程序的正确运行至关重要。 最后,为了提高自己的编程能力,你可以尝试解决一些更复杂的问题或挑战。比如,设计一个温度监测系统或一个遥控车。通过不断地学习和实践,你会逐渐掌握单片机C程序设计的技巧和方法。 总之,学习单片机C程序设计需要耐心和实践。通过手把手学,你可以逐步学习和掌握相关知识,并在实践中不断提升自己的编程能力。祝你学习愉快!
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值