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

OpenHarmony LiteOS-M Shell 命令开发指南

概述

本文档详细介绍如何在 OpenHarmony LiteOS-M 内核中添加自定义 shell 命令,以versionrebootpoweroff命令为例进行说明。

目录结构

kernel/liteos_m/components/shell/ ├── include/shcmd.h # 命令声明头文件 ├── src/base/shcmd.c # 命令注册实现 ├── src/cmds/ # 命令实现目录 │ ├── version_shellcmd.c # version命令实现 │ ├── reboot_shellcmd.c # reboot命令实现 │ └── poweroff_shellcmd.c # poweroff命令实现 └── BUILD.gn # 构建配置文件

开发步骤

第一步:创建命令实现文件

kernel/liteos_m/components/shell/src/cmds/目录下创建命令实现文件。

示例:version_shellcmd.c
#include"los_compiler.h"#include"los_debug.h"#include"syspara/parameter.h"INT32OsShellCmdVersion(INT32 argc,constCHAR**argv){charvalue[128]={0};// 检查参数个数if(argc!=0){PRINTK("Usage: version\n");return-1;}// 输出系统信息PRINTK("OpenHarmony System\n");PRINTK("Kernel: LiteOS-M\n");// 从参数系统获取版本信息if(GetParameter("const.ohos.fullname","",value,sizeof(value))>0){PRINTK("Version: %s\n",value);}else{PRINTK("Version: Unknown\n");}return0;}
示例:reboot_shellcmd.c
#include"los_compiler.h"#include"los_debug.h"// LS2K300 PMC寄存器地址#definePMC_BASE_ADDR0x16124000#definePMC_RESET_REG_OFFSET0x00INT32OsShellCmdReboot(INT32 argc,constCHAR**argv){if(argc!=0){PRINTK("Usage: reboot\n");return-1;}PRINTK("System rebooting...\n");// 写入复位寄存器volatileUINT32*reg=(volatileUINT32*)(PMC_BASE_ADDR+PMC_RESET_REG_OFFSET);*reg=0x1;// 第0位写1触发复位// 内存屏障,确保写入完成__asm__volatile("dbar 0":::"memory");return0;}
示例:poweroff_shellcmd.c
#include"los_compiler.h"#include"los_debug.h"// LS2K300 PMC寄存器地址#definePMC_BASE_ADDR0x16124000#definePMC_SHUTDOWN_REG_OFFSET0x14INT32OsShellCmdPoweroff(INT32 argc,constCHAR**argv){if(argc!=0){PRINTK("Usage: poweroff\n");return-1;}PRINTK("System shutting down...\n");// 写入关机寄存器volatileUINT32*reg=(volatileUINT32*)(PMC_BASE_ADDR+PMC_SHUTDOWN_REG_OFFSET);*reg=0x3c00;// 写入0x3c00触发关机// 内存屏障__asm__volatile("dbar 0":::"memory");return0;}

第二步:声明命令函数

kernel/liteos_m/components/shell/include/shcmd.h文件中添加命令函数声明。

#ifndefSHCMD_H#defineSHCMD_H// ... 其他声明 ...// 添加新命令声明externINT32OsShellCmdVersion(INT32 argc,constCHAR**argv);externINT32OsShellCmdReboot(INT32 argc,constCHAR**argv);externINT32OsShellCmdPoweroff(INT32 argc,constCHAR**argv);#endif

第三步:注册命令

kernel/liteos_m/components/shell/src/base/shcmd.c文件中注册命令。

找到g_shellcmdAll数组,添加新命令的注册信息:

STATICconstSHELL_CMD_INFO_S g_shellcmdAll[]={// ... 其他命令 ...// version 命令{.cmdKey="version",.handler=(SHELL_CMD_HANDLER)OsShellCmdVersion,.help="Display system version information",},// reboot 命令{.cmdKey="reboot",.handler=(SHELL_CMD_HANDLER)OsShellCmdReboot,.help="Reboot the system",},// poweroff 命令{.cmdKey="poweroff",.handler=(SHELL_CMD_HANDLER)OsShellCmdPoweroff,.help="Power off the system",},};

第四步:添加构建配置

kernel/liteos_m/components/shell/BUILD.gn文件中添加新源文件。

sources = [ # ... 其他源文件 ... "src/cmds/version_shellcmd.c", "src/cmds/reboot_shellcmd.c", "src/cmds/poweroff_shellcmd.c", ]

如果需要使用参数系统(如 version 命令),还需要添加依赖:

external_deps = [ # ... 其他依赖 ... "init:parameterbase", ]

第五步:编译测试

执行编译命令:

cd/home/vm/oh/oh61 hb build-pls2k300_mini_dp --gn-argsbuild_xts=true

关键概念说明

1. 命令函数原型

INT32OsShellCmdXXX(INT32 argc,constCHAR**argv)
  • argc: 参数个数(不包括命令本身)
  • argv: 参数数组
  • 返回值: 0 表示成功,非0表示失败

2. 输出函数

使用PRINTK宏进行输出:

PRINTK("Hello World\n");PRINTK("Value: %d\n",value);

3. 参数解析示例

INT32OsShellCmdExample(INT32 argc,constCHAR**argv){// 无参数命令if(argc!=0){PRINTK("Usage: example\n");return-1;}// 带参数命令if(argc!=1){PRINTK("Usage: example <param>\n");return-1;}PRINTK("Param: %s\n",argv[0]);return0;}

4. 硬件寄存器访问

// 定义寄存器地址#defineREG_BASE0x16124000#defineREG_OFFSET0x00// 访问寄存器volatileUINT32*reg=(volatileUINT32*)(REG_BASE+REG_OFFSET);UINT32 value=*reg;// 读寄存器*reg=0x1;// 写寄存器// 内存屏障(LoongArch)__asm__volatile("dbar 0":::"memory");

常见问题

1. 编译错误:函数未定义

错误信息:

undefined reference to `OsShellCmdXXX'

解决方法:

  • 检查函数是否只在 .c 文件中定义一次
  • 检查头文件中是否使用了#ifndef保护
  • 检查是否有内联函数重复定义

2. 运行时显示 “Unknown”

问题:version 命令显示 “Version: Unknown”

原因:参数系统未正确加载参数配置

解决方法:
在设备配置文件中启用参数加载:

# device/board/loongson/ls2k300_mini_dp/liteos_m/config.gni init_feature_begetctl_liteos = true

3. 编译错误:头文件找不到

错误信息:

fatal error: xxx.h: No such file or directory

解决方法:

  • 检查头文件路径是否正确
  • 检查 BUILD.gn 中是否添加了正确的 include_dirs

完整示例代码

version_shellcmd.c

#include"los_compiler.h"#include"los_debug.h"#include"syspara/parameter.h"INT32OsShellCmdVersion(INT32 argc,constCHAR**argv){charvalue[128]={0};if(argc!=0){PRINTK("Usage: version\n");return-1;}PRINTK("OpenHarmony System\n");PRINTK("Kernel: LiteOS-M\n");if(GetParameter("const.ohos.fullname","",value,sizeof(value))>0){PRINTK("Version: %s\n",value);}else{PRINTK("Version: Unknown\n");}return0;}

reboot_shellcmd.c

#include"los_compiler.h"#include"los_debug.h"#definePMC_BASE_ADDR0x16124000#definePMC_RESET_REG_OFFSET0x00INT32OsShellCmdReboot(INT32 argc,constCHAR**argv){if(argc!=0){PRINTK("Usage: reboot\n");return-1;}PRINTK("System rebooting...\n");volatileUINT32*reg=(volatileUINT32*)(PMC_BASE_ADDR+PMC_RESET_REG_OFFSET);*reg=0x1;__asm__volatile("dbar 0":::"memory");return0;}

poweroff_shellcmd.c

#include"los_compiler.h"#include"los_debug.h"#definePMC_BASE_ADDR0x16124000#definePMC_SHUTDOWN_REG_OFFSET0x14INT32OsShellCmdPoweroff(INT32 argc,constCHAR**argv){if(argc!=0){PRINTK("Usage: poweroff\n");return-1;}PRINTK("System shutting down...\n");volatileUINT32*reg=(volatileUINT32*)(PMC_BASE_ADDR+PMC_SHUTDOWN_REG_OFFSET);*reg=0x3c00;__asm__volatile("dbar 0":::"memory");return0;}

调试技巧

1. 使用 PRINTK 调试

INT32OsShellCmdTest(INT32 argc,constCHAR**argv){PRINTK("Debug: argc = %d\n",argc);for(inti=0;i<argc;i++){PRINTK("Debug: argv[%d] = %s\n",i,argv[i]);}return0;}

2. 检查参数系统状态

#include"init_param.h"voidCheckParamStatus(void){intret=SystemGetParameter("const.ohos.fullname",value,&len);PRINTK("GetParameter ret = %d\n",ret);}

参考资料

  • OpenHarmony 官方文档
  • LiteOS-M 内核开发指南
  • LoongArch 架构参考手册

http://www.jsqmd.com/news/636633/

相关文章:

  • 为什么92%的SITS2026项目在Phase 2失败?——多Agent角色编排、任务分解与状态同步的黄金三角模型,
  • STM32+OpenCV智能分拣实战:从图像采集到云端监控的嵌入式系统设计
  • Spring AI MCP客户端实战:从配置到工具调用的完整指南
  • OV2640寄存器配置黑魔法:手把手教你用ESP32-S3调出专业级画质
  • Devuino:面向Arduino的现代C++设备抽象库
  • 避坑指南:VS2022配置IMSL Fortran库时常见的路径错误与权限问题(附64位系统专属解决方案)
  • Phi-3-mini-128k-instruct代码解释能力实测:逆向工程与文档生成
  • 使用OpenClaw来拯救一个重度脂肪肝患者
  • 阿里云Notebook免费额度别浪费!手把手教你部署通义千问2-VL-2B视觉模型
  • Uniapp评论模块实战:手把手构建嵌套回复与智能展开收起
  • 【AIAgent客服系统架构解密】:SITS2026实战中高并发、低延迟、可解释性三大瓶颈的破局之道
  • 极速精准生图!小红书把Z-Image打造成人人都能本地跑的GPT-4o
  • Motorola DMR设备玩转APRS定位:从零配置到实战避坑指南
  • 生产环境离线部署大模型
  • 通达信筹码大单捉妖指标实战解析:主副图组合精准捕捉庄家动向
  • 为什么你的AIAgent一换场景就失智?揭秘迁移学习中被忽略的3类隐式分布偏移
  • 为什么你的网络总抽风?可能是这个ARP协议漏洞在捣鬼(含防御方案)
  • Calico IPIP 使用指南旅
  • 4月14日直播丨CANNBot 开发进阶:Ascend C算子开发实操
  • Agent 才 1 岁多,市场已经要求 5 年以上经验了
  • KonkerESP8266嵌入式MQTT/HTTP物联网通信框架解析
  • 告别虚拟机卡顿:用WSL2+Docker高效搭建海思Hi3516CV610交叉编译环境
  • 从洗碗到叠衣:用RECAP算法让机器人学会‘吃一堑长一智’
  • 遥感数字图像处理教程【2.2】
  • 试试建几个 GPTs,看看有没有什么用 - AI
  • 国内环境使用Claude Code的可行路径与聚合平台模式说明
  • 2026届学术党必备的十大降AI率神器实测分析
  • 告别重复代码:Vercel 无服务函数中的高阶函数封装技巧(含认证/日志实战)
  • 第16章 项目干系人管理
  • 如何解决Kirikiri游戏资源加密难题:全功能KirikiriTools实战指南