IO模拟SPI操作SD卡系列之六:实现shell文件操作命令集

原创 嵌入式Lee 2024-08-16 08:00

一. 前言

我们现在已经实现了通过shell使用xmodem协议进行文件的导入导出,现在我们继续来实现文件操作的一些常见shell命令,比如查看文件,ls,移动我呢见mv等。

二. 实现ls

shell_func.c中申明实现函数

static void lsfilefunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"ls",       lsfilefunc,       (uint8_t*)"ls path"},

实现函数如下

/* List contents of a directory */static FRESULT list_dir (const char *path){    FRESULT res;    DIR dir;    FILINFO fno;    int nfile, ndir;
    res = f_opendir(&dir, path);                       /* Open the directory */    if (res == FR_OK) {        nfile = ndir = 0;        for (;;) {            res = f_readdir(&dir, &fno);                   /* Read a directory item */            if (res != FR_OK || fno.fname[0] == 0) break;  /* Error or end of dir */            if (fno.fattrib & AM_DIR) {            /* Directory */                xprintf("     %s\n", fno.fname);                ndir++;            } else {                               /* File */                xprintf("%10u %s\n", fno.fsize, fno.fname);                nfile++;            }        }        f_closedir(&dir);        printf("%d dirs, %d files.\n", ndir, nfile);    } else {        xprintf("Failed to open \"%s\". (%u)\n", path, res);    }    return res;}
void lsfilefunc(uint8_t* param){  char path[128];  if(1 == sscanf((const char*)param, "%*s %s", path))  {    list_dir((const char *)path);  }  else  {    printf("param err");  }}

Help查看命令

ls查看文件

三. 实现mv

使用rename实现

四. 实现cp

shell_func.c中申明实现函数

static void cpfilefunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"cp",       cpfilefunc,       (uint8_t*)"cp srcpath dstpath"},

实现函数如下

void cpfilefunc(uint8_t* param){  char srcpath[128];  char dstpath[128];  uint8_t tmp[32];  FIL srcfil;  FIL dstfil;  UINT bw;  UINT br;  FRESULT res;  if(2 == sscanf((const char*)param, "%*s %s %s", srcpath, dstpath))  {    xprintf("cp %d to %s\r\n",srcpath,dstpath);    res = f_open(&srcfil, srcpath, FA_READ);    if(res == FR_OK)    {      res = f_open(&dstfil, dstpath, FA_CREATE_NEW | FA_WRITE);      if(res == FR_OK)      {        do        {          res = f_read(&srcfil, tmp, sizeof(tmp), &br);          if(res == 0)          {            res = f_write(&dstfil, tmp, br, &bw);            if ((bw != br) || (res != 0))            {              xprintf("write err %d %d\r\n",bw,res);              break;            }          }          else          {            xprintf("read err %d\r\n",res);             break;          }        }while(br > 0);
        f_close(&dstfil);        f_close(&srcfil);      }      else      {        f_close(&srcfil);        xprintf("open %s err %d\r\n",dstpath,res);      }    }    else    {      xprintf("open %s err %d\r\n",srcpath,res);    }  }  else  {    xprintf("param err");  }}

Help查看命令

ls查看文件,cp复制一份,ls再查看

五. 实现rm

shell_func.c中申明实现函数

static void rmfilefunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"rm",       rmfilefunc,       (uint8_t*)"rm path"},

实现函数如下

void rmfilefunc(uint8_t* param){  FRESULT res;  char path[128];  if(1 == sscanf((const char*)param, "%*s %s", path))  {    if(FR_OK != (res = f_unlink((const char *)path)))    {      printf("unlink %s err %d\r\n",path,res);    }  }  else  {    printf("param err");  }}

help查看命令

ls查看文件

rm删除文件

ls再查看是否删除

六. 实现rename

shell_func.c中申明实现函数

static void renamefilefunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"rename",       renamefilefunc,       (uint8_t*)"rename path newpath"},

实现函数如下

void renamefilefunc(uint8_t* param){  FRESULT res;  char path[128];  char newpath[128];  if(2 == sscanf((const char*)param, "%*s %s %s", path, newpath))  {    if(FR_OK != (res = f_rename((const char *)path, (const char*)newpath)))    {      xprintf("rename %s to %s err %d\r\n",path,newpath,res);    }  }  else  {    xprintf("param err");  }}

help查看命令

 

ls查看文件

rename修改名字

ls看名字是否修改

七. 实现mkdir

shell_func.c中申明实现函数

static void mkdirfunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"mkdir",       mkdirfunc,       (uint8_t*)"mkdir path"},

实现函数如下

void mkdirfunc(uint8_t* param){  char path[128];  FRESULT res;  if(1 == sscanf((const char*)param, "%*s %s", path))  {    if(FR_OK != (res=f_mkdir((const char*)path)))    {      xprintf("mkdir %s err %d",path,res);    }  }  else  {    xprintf("param err");  }}

help查看命令

mkdir创建命令,然后ls查看

八. 实现touch

shell_func.c中申明实现函数

static void touchfunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"touch",       touchfunc,       (uint8_t*)"touch path size"},

实现函数如下

需要配置宏

#define FF_USE_EXPAND   1

void touchfunc(uint8_t* param){  FIL fil;  char path[128];  int size;  if(2 == sscanf((const char*)param, "%*s %s %d", path, &size))  {    FRESULT res = f_open(&fil, path, FA_CREATE_NEW | FA_WRITE);    if (FR_OK == res)    {      if(FR_OK != (res = f_expand(&fil,size,1)))      {        xprintf("expand %s size to %d err %d\r\n",path,size,res);       }
      if(FR_OK != (res = f_close(&fil)))      {        xprintf("close %s err %d\r\n",path,res);       }    }    else    {      xprintf("open %s err %d\r\n",path,res);    }  }  else  {    xprintf("param err");  }}

Help查看命令

touch创建一个文件

ls查看

九. 实现pwd

shell_func.c中申明实现函数

static void pwdfunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"pwd",       pwdfunc,       (uint8_t*)"pwd"},

实现函数如下

以下宏要配置为2

#define FF_FS_RPATH    2

void pwdfunc(uint8_t* param){  FRESULT fr;  TCHAR str[128];  if(FR_OK != (fr = f_getcwd(str, 128)))  /* Get current directory path */  {    xprintf("getcwd err %d\r\n",fr);  }  else  {    xprintf("%s\r\n",str);  }}

help查看命令

pwd查看当前目录

十. 实现cd

shell_func.c中申明实现函数

static void cdfunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"cd",       cdfunc,       (uint8_t*)"cd path"},

实现函数如下

以下宏要大于等于1 ,这里还要使用f_getcwd所以设置为2

#define FF_FS_RPATH     2

void cdfunc(uint8_t* param){  char path[128];  FRESULT res;  if(1 == sscanf((const char*)param, "%*s %s", path))  {    if(FR_OK != (res=f_chdir((const char*)path)))    {      xprintf("chdir to %s err %d",path,res);    }  }  else  {    xprintf("param err");  }}

help查看命令

cd进入子目录,pwd再查看当前目录

十一. 实现hexdump

shell_func.c中申明实现函数

static void hexdumpfunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"hexdump",       hexdumpfunc,       (uint8_t*)"hexdump path addr size"},

实现函数如下

void hexdumpfunc(uint8_t* param){  char path[128];  uint8_t tmp[16];  uint32_t addr;  uint32_t size;  FRESULT res;  FIL fil;  UINT br;  uint32_t offset = 0;  if(3 == sscanf((const char*)param, "%*s %s %lx %ld", path, &addr, &size))  {    xprintf("hexdump %s 0x%x %d\r\n",path,addr,size);    if(FR_OK == (res=f_open(&fil,(const char*)path, FA_READ)))    {      xprintf("\r\n");      do      {        br = 0;        if(FR_OK == (res = f_read(&fil,tmp,(size>16)?16:size,&br)))        {          xprintf("%08x ",offset);          offset+=br;          for(uint32_t i=0;i          {            xprintf("%02x",(uint32_t)tmp[i]);          }          xprintf(":");          for(uint32_t i=0;i          {            xprintf("%c",((tmp[i]>0x1F)&&(tmp[i]<0x7F))?(char)tmp[i]:'.');          }          xprintf("\r\n");          size -= br;        }        else        {          break;        }      }while(br > 0);
      if(FR_OK != (res = f_close(&fil)))      {        xprintf("close %s err %d\r\n",path,res);       }    }    else    {      xprintf("open %s err %d",path,res);    }  }  else  {    xprintf("param err");  }}

help查看命令

hexdump查看内容

十二. 实现hexwrite

shell_func.c中申明实现函数

static void hexwritefunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"hexwrite",       hexwritefunc,       (uint8_t*)"hexwrite path addr[hex] hexstr"},

实现函数如下

static int ascii2uc(const char c, unsigned char *uc){    if ((c >= '0') && (c <= '9')) {        *uc = c - '0';    } else if ((c >= 'a') && (c <= 'f')) {        *uc = c - 'a' + 10;    } else if ((c >= 'A') && (c <= 'F')) {        *uc = c - 'A' + 10;    } else {        return -1;    }
    return 0;}
static uint32_t str2hex(const char* str, unsigned char *buff, uint32_t len){  uint32_t num = 0;  uint8_t hex = 0;  while(1)  {    uint8_t tmp1;    uint8_t tmp2;    if(ascii2uc(*str++, &tmp1) < 0)    {      break;    }    if(ascii2uc(*str++, &tmp2) < 0)    {      break;    }    hex = tmp1*16 + tmp2;    *buff++ = hex;    num++;    if(num >= len)    {      break;    }  }  return num;}
void hexwritefunc(uint8_t* param){  char path[128];  uint8_t hexstr[32+1];  uint8_t tmp[16];  uint32_t hexnum = 0;  uint32_t addr;  FRESULT res;  FIL fil;  UINT bw;  if(3 == sscanf((const char*)param, "%*s %s %lx %s", path, &addr, hexstr))  {    xprintf("hexwrite %s 0x%x %s\r\n",path,addr,hexstr);    if(FR_OK == (res=f_open(&fil,(const char*)path, FA_WRITE)))    {      xprintf("\r\n");      hexnum = str2hex((const char*)hexstr,tmp,32);      if(hexnum > 0)      {        if(FR_OK == (res=(f_lseek(&fil,addr))))        {          if(FR_OK != (res = f_write(&fil,tmp,hexnum,&bw)))          {            xprintf("write err %d\r\n",res);          }        }        else        {          xprintf("seek %d err %d\r\n",addr,res);        }      }
      if(FR_OK != (res = f_close(&fil)))      {        xprintf("close %s err %d\r\n",path,res);       }    }    else    {      xprintf("open %s err %d",path,res);    }  }  else  {    xprintf("param err");  }}

Help查看命令

Hexwrite修改内容,再hexdump查看

十三. 实现md5sum

shell_func.c中申明实现函数

static void md5sumfunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"md5sum",       md5sumfunc,       (uint8_t*)"md5sum path"},

实现函数如下

/* md5算法参考 https://www.rfc-editor.org/rfc/rfc1321 */
void md5sumfunc(uint8_t* param){  char path[128];  uint8_t tmp[32];  uint8_t out[16];  FIL fil;  UINT br;  FRESULT res;
  MD5_CTX ctx;  MD5Init(&ctx);  if(1 == sscanf((const char*)param, "%*s %s",path))  {    res = f_open(&fil, path, FA_READ);    if(res == FR_OK)    {      do      {        res = f_read(&fil, tmp, sizeof(tmp), &br);        if(res == 0)        {          MD5Update(&ctx, tmp, br);        }        else        {          xprintf("read err %d\r\n",res);           break;        }      }while(br > 0);
      MD5Final(out, &ctx);      for(int i=0; i<16;i++)      {         xprintf("%02x",out[i]);      }      xprintf("\r\n");      f_close(&fil);    }    else    {      xprintf("open %s err %d\r\n",path,res);    }  }  else  {    xprintf("param err");  }}

help查看命令

md5sum计算,和电脑上计算一致

十四. 实现free

shell_func.c中申明实现函数

static void freefunc(uint8_t* param);

g_shell_cmd_list_ast中添加一行

{ (uint8_t*)"free",       freefunc,       (uint8_t*)"free path"},

实现函数如下

void freefunc(uint8_t* param){  char path[128];  FRESULT res;  if(1 == sscanf((const char*)param, "%*s %s", path))  {    FATFS *fs;    DWORD fre_clust, fre_sect, tot_sect;    /* Get volume information and free clusters of drive 1 */    res = f_getfree(path, &fre_clust, &fs);    if(res == FR_OK)    {      /* Get total sectors and free sectors */      tot_sect = (fs->n_fatent - 2) * fs->csize;      fre_sect = fre_clust * fs->csize;
      /* Print the free space (assuming 512 bytes/sector) */      xprintf("%10lu KiB total drive space.\n%10lu KiB available.\n", tot_sect / 2, fre_sect / 2);    }    else    {      xprintf("getfree %s err %d\r\n",path,res);    }  }  else  {    xprintf("param err");  }}

Help查看命令

Free查看空间大小,我这里是4GB的卡

十五.总结

以上实现了比较常用的文件操作shell命令,可以方便的进行文件操作了。


评论
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 138浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球无人机锂电池产值达到2457百万美元,2024-2030年期间年复合增长率CAGR为9.6%。 无人机锂电池是无人机动力系统中存储并释放能量的部分。无人机使用的动力电池,大多数是锂聚合物电池,相较其他电池,锂聚合物电池具有较高的能量密度,较长寿命,同时也具有良好的放电特性和安全性。 全球无人机锂电池核心厂商有宁德新能源科技、欣旺达、鹏辉能源、深圳格瑞普和EaglePicher等,前五大厂商占有全球
    GIRtina 2025-01-07 11:02 95浏览
  • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
    知白 2025-01-06 12:04 188浏览
  • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
    优思学院 2025-01-06 12:03 135浏览
  • 随着市场需求不断的变化,各行各业对CPU的要求越来越高,特别是近几年流行的 AIOT,为了有更好的用户体验,CPU的算力就要求更高了。今天为大家推荐由米尔基于瑞芯微RK3576处理器推出的MYC-LR3576核心板及开发板。关于RK3576处理器国产CPU,是这些年的骄傲,华为手机全国产化,国人一片呼声,再也不用卡脖子了。RK3576处理器,就是一款由国产是厂商瑞芯微,今年第二季推出的全新通用型的高性能SOC芯片,这款CPU到底有多么的高性能,下面看看它的几个特性:8核心6 TOPS超强算力双千
    米尔电子嵌入式 2025-01-03 17:04 55浏览
  • 村田是目前全球量产硅电容的领先企业,其在2016年收购了法国IPDiA头部硅电容器公司,并于2023年6月宣布投资约100亿日元将硅电容产能提升两倍。以下内容主要来自村田官网信息整理,村田高密度硅电容器采用半导体MOS工艺开发,并使用3D结构来大幅增加电极表面,因此在给定的占位面积内增加了静电容量。村田的硅技术以嵌入非结晶基板的单片结构为基础(单层MIM和多层MIM—MIM是指金属 / 绝缘体/ 金属) 村田硅电容采用先进3D拓扑结构在100um内,使开发的有效静电容量面积相当于80个
    知白 2025-01-07 15:02 99浏览
  • 大模型的赋能是指利用大型机器学习模型(如深度学习模型)来增强或改进各种应用和服务。这种技术在许多领域都显示出了巨大的潜力,包括但不限于以下几个方面: 1. 企业服务:大模型可以用于构建智能客服系统、知识库问答系统等,提升企业的服务质量和运营效率。 2. 教育服务:在教育领域,大模型被应用于个性化学习、智能辅导、作业批改等,帮助教师减轻工作负担,提高教学质量。 3. 工业智能化:大模型有助于解决工业领域的复杂性和不确定性问题,尽管在认知能力方面尚未完全具备专家级的复杂决策能力。 4. 消费
    丙丁先生 2025-01-07 09:25 93浏览
  • 本文介绍Linux系统更换开机logo方法教程,通用RK3566、RK3568、RK3588、RK3576等开发板,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。制作图片开机logo图片制作注意事项(1)图片必须为bmp格式;(2)图片大小不能大于4MB;(3)BMP位深最大是32,建议设置为8;(4)图片名称为logo.bmp和logo_kernel.bmp;开机
    Industio_触觉智能 2025-01-06 10:43 87浏览
  •     为控制片内设备并且查询其工作状态,MCU内部总是有一组特殊功能寄存器(SFR,Special Function Register)。    使用Eclipse环境调试MCU程序时,可以利用 Peripheral Registers Viewer来查看SFR。这个小工具是怎样知道某个型号的MCU有怎样的寄存器定义呢?它使用一种描述性的文本文件——SVD文件。这个文件存储在下面红色字体的路径下。    例:南京沁恒  &n
    电子知识打边炉 2025-01-04 20:04 105浏览
  • By Toradex 秦海1). 简介嵌入式平台设备基于Yocto Linux 在开发后期量产前期,为了安全以及提高启动速度等考虑,希望将 ARM 处理器平台的 Debug Console 输出关闭,本文就基于 NXP i.MX8MP ARM 处理器平台来演示相关流程。 本文所示例的平台来自于 Toradex Verdin i.MX8MP 嵌入式平台。  2. 准备a). Verdin i.MX8MP ARM核心版配合Dahlia载板并
    hai.qin_651820742 2025-01-07 14:52 62浏览
  • 本文介绍编译Android13 ROOT权限固件的方法,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。关闭selinux修改此文件("+"号为修改内容)device/rockchip/common/BoardConfig.mkBOARD_BOOT_HEADER_VERSION ?= 2BOARD_MKBOOTIMG_ARGS :=BOARD_PREBUILT_DTB
    Industio_触觉智能 2025-01-08 00:06 42浏览
  • 根据Global Info Research项目团队最新调研,预计2030年全球封闭式电机产值达到1425百万美元,2024-2030年期间年复合增长率CAGR为3.4%。 封闭式电机是一种电动机,其外壳设计为密闭结构,通常用于要求较高的防护等级的应用场合。封闭式电机可以有效防止外部灰尘、水分和其他污染物进入内部,从而保护电机的内部组件,延长其使用寿命。 环洋市场咨询机构出版的调研分析报告【全球封闭式电机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球封闭式电机总体规
    GIRtina 2025-01-06 11:10 112浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 172浏览
  • PLC组态方式主要有三种,每种都有其独特的特点和适用场景。下面来简单说说: 1. 硬件组态   定义:硬件组态指的是选择适合的PLC型号、I/O模块、通信模块等硬件组件,并按照实际需求进行连接和配置。    灵活性:这种方式允许用户根据项目需求自由搭配硬件组件,具有较高的灵活性。    成本:可能需要额外的硬件购买成本,适用于对系统性能和扩展性有较高要求的场合。 2. 软件组态   定义:软件组态主要是通过PLC
    丙丁先生 2025-01-06 09:23 88浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦