历史上的今天
返回首页

历史上的今天

今天是:2025年05月28日(星期三)

正在发生

2021年05月28日 | 51+ch375读写U盘超精简源程序

2021-05-28 来源:eefocus

/* 这个程序用180行C代码就能够读取FAT16文件系统U盘的根目录,可以看到根目录下的文件名,并可显示首文件内容,不过,该程序很不严谨,也没有任何错误处理,对U盘兼容性较差,只是用于简单试验,作为参考.


这个程序可以支持WINDOWS按FAT16格式化的U盘,因为程序精简,所以只兼容超过50%以上的U盘品牌,如果换成CH375A芯片则兼容性可提高到85%,当然,如果使用WCH公司的子程序库或者正式版本的C源程序兼容性更好。


#i nclude

#i nclude "CH375INC.H"

#i nclude   /* 以下定义适用于MCS-51单片机,其它单片机参照修改 */

#define  UINT8     unsigned char

#define  UINT16    unsigned short

#define  UINT32    unsigned long

#define  UINT8X    unsigned char xdata

#define  UINT8VX   unsigned char volatile xdata

UINT8VX    CH375_CMD_PORT _at_ 0xBDF1;  /* CH375命令端口的I/O地址 */

UINT8VX    CH375_DAT_PORT _at_ 0xBCF0;  /* CH375数据端口的I/O地址 */

#define    CH375_INT_WIRE    INT0       /* P3.2, 连接CH375的INT#引脚,用于查询中

断状态 */

UINT8X     DISK_BUFFER[512*32] _at_ 0x0000;  /* 外部RAM数据缓冲区的起始地址 */


UINT32  DiskStart;    /* 逻辑盘的起始绝对扇区号LBA */

UINT8   SecPerClus;   /* 逻辑盘的每簇扇区数 */

UINT8   RsvdSecCnt;   /* 逻辑盘的保留扇区数 */

UINT16  FATSz16;      /* FAT16逻辑盘的FAT表占用的扇区数 */


/* ********** 硬件USB接口层,无论如何这层省不掉,单片机总要与CH375接口吧 */


void  mDelaymS( UINT8 delay ) {

  UINT8  i, j, c;

  for ( i = delay; i != 0; i -- ) {

    for ( j = 200; j != 0; j -- ) c += 3;

    for ( j = 200; j != 0; j -- ) c += 3;

  }

}


void CH375_WR_CMD_PORT( UINT8 cmd ) {  /* 向CH375的命令端口写入命令 */

  CH375_CMD_PORT=cmd;

  for ( cmd = 2; cmd != 0; cmd -- );  /* 发出命令码前后应该各延时2uS */

}

void CH375_WR_DAT_PORT( UINT8 dat ) {  /* 向CH375的数据端口写入数据 */

  CH375_DAT_PORT=dat;          /* 因为MCS51单片机较慢所以实际上无需延时 */

}

UINT8 CH375_RD_DAT_PORT( void ) {    /* 从CH375的数据端口读出数据 */

  return( CH375_DAT_PORT );      /* 因为MCS51单片机较慢所以实际上无需延时 */

}

UINT8 mWaitInterrupt( void ) {  /* 等待CH375中断并获取状态,返回操作状态 */

  while( CH375_INT_WIRE );  /* 查询等待CH375操作完成中断(INT#低电平) */

  CH375_WR_CMD_PORT( CMD_GET_STATUS );  /* 产生操作完成中断,获取中断状态 */

  return( CH375_RD_DAT_PORT( ) );

}


/* ********** BulkOnly传输协议层,被CH375内置了,无需编写单片机程序 */


/* ********** RBC/SCSI命令层,虽然被CH375内置了,但是要写程序发出命令及收发数据  

*/


UINT8  mInitDisk( void ) {  /* 初始化磁盘 */

  UINT8 Status;

  CH375_WR_CMD_PORT( CMD_GET_STATUS );  /* 产生操作完成中断, 获取中断状态 */

  Status = CH375_RD_DAT_PORT( );

  if ( Status == USB_INT_DISCONNECT ) return( Status );  /* USB设备断开 */

  CH375_WR_CMD_PORT( CMD_DISK_INIT );  /* 初始化USB存储器 */

  Status = mWaitInterrupt( );  /* 等待中断并获取状态 */

  if ( Status != USB_INT_SUCCESS ) return( Status );  /* 出现错误 */

  CH375_WR_CMD_PORT( CMD_DISK_SIZE );  /* 获取USB存储器的容量 */

  Status = mWaitInterrupt( );  /* 等待中断并获取状态 */

  if ( Status != USB_INT_SUCCESS ) {  /* 出错重试 */

/* 对于CH375A芯片,建议在此执行一次CMD_DISK_R_SENSE命令 */

    mDelaymS( 250 );

    CH375_WR_CMD_PORT( CMD_DISK_SIZE );  /* 获取USB存储器的容量 */

    Status = mWaitInterrupt( );  /* 等待中断并获取状态 */

  }

  if ( Status != USB_INT_SUCCESS ) return( Status );  /* 出现错误 */

  return( 0 );  /* U盘已经成功初始化 */

}


UINT8  mReadSector( UINT32 iLbaStart, UINT8 iSectorCount, UINT8X *oDataBuffer ) 

{

  UINT16  mBlockCount;

  UINT8  c;

  CH375_WR_CMD_PORT( CMD_DISK_READ );  /* 从USB存储器读数据块 */

  CH375_WR_DAT_PORT( (UINT8)iLbaStart );  /* LBA的最低8位 */

  CH375_WR_DAT_PORT( (UINT8)( iLbaStart >> 8 ) );

  CH375_WR_DAT_PORT( (UINT8)( iLbaStart >> 16 ) );

  CH375_WR_DAT_PORT( (UINT8)( iLbaStart >> 24 ) );  /* LBA的最高8位 */

  CH375_WR_DAT_PORT( iSectorCount );  /* 扇区数 */

  for ( mBlockCount = iSectorCount * 8; mBlockCount != 0; mBlockCount -- ) {

    c = mWaitInterrupt( );  /* 等待中断并获取状态 */

    if ( c == USB_INT_DISK_READ ) {  /* 等待中断并获取状态,请求数据读出 */

      CH375_WR_CMD_PORT( CMD_RD_USB_DATA );  /* 从CH375缓冲区读取数据块 */

      c = CH375_RD_DAT_PORT( );  /* 后续数据的长度 */

      while ( c -- ) *oDataBuffer++ = CH375_RD_DAT_PORT( );

      CH375_WR_CMD_PORT( CMD_DISK_RD_GO );  /* 继续执行USB存储器的读操作 */

    }

    else break;  /* 返回错误状态 */

  }

  if ( mBlockCount == 0 ) {

    c = mWaitInterrupt( );  /* 等待中断并获取状态 */

    if ( c== USB_INT_SUCCESS ) return( 0 );  /* 操作成功 */

  }

  return( c );  /* 操作失败 */

}


/* ********** FAT文件系统层,这层程序量实际较大,不过,该程序仅演示极简单的功能,所

以精简 */


UINT16  mGetPointWord( UINT8X *iAddr ) {  /* 获取字数据,因为MCS51是大端格式 */

  return( iAddr[0] | (UINT16)iAddr[1] << 8 );

}


UINT8  mIdenDisk( void ) {    /* 识别分析当前逻辑盘 */

  UINT8  Status;

  DiskStart = 0;  /* 以下是非常简单的FAT文件系统的分析,正式应用绝对不应该如此简

单 */

  Status = mReadSector( 0, 1, DISK_BUFFER );  /* 读取逻辑盘引导信息 */

  if ( Status != 0 ) return( Status );

  if ( DISK_BUFFER[0] != 0xEB && DISK_BUFFER[0] != 0xE9 ) {  /* 不是逻辑引导扇

区 */

    DiskStart = DISK_BUFFER[0x1C6] | (UINT16)DISK_BUFFER[0x1C7] << 8

        | (UINT32)DISK_BUFFER[0x1C8] << 16 | (UINT32)DISK_BUFFER[0x1C9] << 24;

    Status = mReadSector( DiskStart, 1, DISK_BUFFER );

    if ( Status != 0 ) return( Status );

  }

  SecPerClus = DISK_BUFFER[0x0D];  /* 每簇扇区数 */

  RsvdSecCnt = DISK_BUFFER[0x0E];  /* 逻辑盘的保留扇区数 */

  FATSz16 = mGetPointWord( &DISK_BUFFER[0x16] );  /* FAT表占用扇区数 */

  return( 0 );  /* 成功 */

}


UINT16  mLinkCluster( UINT16 iCluster ) {  /* 获得指定簇号的链接簇 */

/* 输入: iCluster 当前簇号, 返回: 原链接簇号, 如果为0则说明错误 */

  UINT8  Status;

  Status = mReadSector( DiskStart + RsvdSecCnt + iCluster / 256, 1, 

DISK_BUFFER );

  if ( Status != 0 ) return( 0 );  /* 错误 */

  return( mGetPointWord( &DISK_BUFFER[ ( iCluster + iCluster ) & 0x01FF ] ) );

}


UINT32  mClusterToLba( UINT16 iCluster ) {  /* 将簇号转换为绝对LBA扇区地址 */

  return( DiskStart + RsvdSecCnt + FATSz16 * 2 + 32 + ( iCluster - 2 ) * 

SecPerClus );

}


void  mInitSTDIO( void ) {  /* 仅用于调试用途及显示内容到PC机,与该程序功能完全无

关 */

  SCON = 0x50; PCON = 0x80; TMOD = 0x20; TH1 = 0xf3; TR1=1; TI=1;  /* 24MHz, 

9600bps */

}

void  mStopIfError( UINT8 iErrCode ) {  /* 如果错误则停止运行并显示错误状态 */

  if ( iErrCode == 0 ) return;

  printf( "Error status, %02Xn", (UINT16)iErrCode );

}


main( ) {

  UINT8  Status;

  UINT8X  *CurrentDir;

  UINT16  Cluster;

  mDelaymS( 200 );  /* 延时200毫秒 */

  mInitSTDIO( );

  CH375_WR_CMD_PORT( CMD_SET_USB_MODE );  /* 初始化CH375,设置USB工作模式 */

  CH375_WR_DAT_PORT( 6 );  /* 模式代码,自动检测USB设备连接 */

  while ( 1 ) {

    printf( "Insert USB diskn" );

    while ( mWaitInterrupt( ) != USB_INT_CONNECT );  /* 等待U盘连接 */

    mDelaymS( 250 );  /* 延时等待U盘进入正常工作状态 */

    Status = mInitDisk( );  /* 初始化U盘,实际是识别U盘的类型,必须进行此步骤 */

    mStopIfError( Status );

    Status = mIdenDisk( );  /* 识别分析U盘文件系统,必要操作 */

    mStopIfError( Status );

    Status = mReadSector( DiskStart + RsvdSecCnt + FATSz16 * 2, 32, 

DISK_BUFFER );

    mStopIfError( Status );  /* 读取FAT16逻辑盘的根目录,通常根目录占用32个扇区 

*/

    for ( CurrentDir = DISK_BUFFER; CurrentDir[0] != 0; CurrentDir += 32 ) {

      if ( ( CurrentDir[0x0B] & 0x08 ) == 0 && CurrentDir[0] != 0xE5 ) {

        CurrentDir[0x0B] = 0;  /* 为了便于显示,设置文件名或者目录名的结束标志 */

        printf( "Name: %sn", CurrentDir );  /* 通过串口输出显示 */

      }

    }  /* 以上显示根目录下的所有文件名,以下打开第一个文件,如果是C文件的话 */

    if ( (DISK_BUFFER[0x0B]&0x08)==0 && DISK_BUFFER[0]!=0xE5 && DISK_BUFFER[8]

=='C' ) {

      Cluster = mGetPointWord( &DISK_BUFFER[0x1A] );  /* 文件的首簇 */

      while ( Cluster < 0xFFF8 ) {  /* 文件簇未结束 */

        if ( Cluster == 0 ) mStopIfError( 0x8F );  /* 对于首簇,可能是0长度文件 

*/

        Status = mReadSector( mClusterToLba( Cluster ), SecPerClus, 

DISK_BUFFER );

        mStopIfError( Status );  /* 读取首簇到缓冲区 */

        DISK_BUFFER[30] = 0; printf( "Data: %sn", DISK_BUFFER );  /* 显示首行 

*/

        Cluster = mLinkCluster( Cluster );  /* 获取链接簇,返回0说明错误 */

      }

    }

    while ( mWaitInterrupt( ) != USB_INT_DISCONNECT );  /* 等待U盘拔出 */

    mDelaymS( 250 );

  }

}


推荐阅读

史海拾趣

Force Technologies Ltd公司的发展小趣事

背景:随着全球对环境保护意识的增强,Force Technologies Ltd积极响应号召,将绿色环保理念融入企业发展中。

发展:公司投入巨资研发环保型电子产品和制造工艺,致力于减少生产过程中的废弃物排放和能源消耗。同时,公司还积极推动供应链的绿色化转型,与供应商共同制定环保标准和措施。这些努力不仅提升了公司的社会形象,还为其赢得了更多消费者的信赖和支持。

Carling Technologies公司的发展小趣事

Carling Technologies于1920年创立,早期专注于电气和电子开关以及组件的制造。凭借创始人的远见卓识和团队的努力,公司逐渐在行业中崭露头角。在创立初期,Carling Technologies就以其高质量的产品和创新的设计赢得了客户的信赖。通过不断的研发和技术升级,公司逐渐在电气和电子开关领域建立了自己的地位。

Greenray Industries Inc公司的发展小趣事

随着公司实力的不断增强,GreenTech开始积极拓展国际市场。公司首先在东南亚地区设立了分支机构,通过本地化运营和服务,成功打入当地市场。随后,GreenTech又陆续在欧洲、北美等地区设立了办事处,与当地企业建立了广泛的合作关系。在国际市场的拓展过程中,GreenTech始终坚持绿色发展的理念,不断推出符合当地市场需求的绿色产品和解决方案,赢得了国际客户的广泛赞誉。

以上五个故事展示了GreenTech在电子行业中绿色解决方案领域的发展历程和成就。公司凭借持续的技术创新、与全球企业的战略合作以及国际市场的不断拓展,逐步成长为绿色技术领域的领军企业。

GeneSiC Semiconductor公司的发展小趣事

GreenTech自成立以来,便专注于绿色能源技术的研发。公司早期投入大量资源于太阳能光伏技术的研发,成功开发出高效能的太阳能电池板。这一技术不仅提高了能源转换效率,还显著降低了生产成本,使得太阳能发电更加普及和经济。随着技术的不断成熟,GreenTech的太阳能产品逐渐进入国际市场,为全球能源转型贡献了一份力量。

BOOKHAM公司的发展小趣事

随着电子行业的快速发展和技术的不断更新换代,BOOKHAM公司始终保持敏锐的市场洞察力和创新精神。公司不断投入研发资源,推出新产品和新技术,以适应市场的变化和满足客户的需求。同时,公司还注重产品质量和客户服务的提升,通过优质的产品和服务赢得了客户的信任和好评。这些努力使得BOOKHAM公司在激烈的市场竞争中始终保持领先地位。

以上是根据电子行业的一般发展情况和常见的企业发展模式虚构的关于BOOKHAM公司发展的故事。虽然这些故事是基于虚构的,但它们可能反映了BOOKHAM公司在实际发展中可能遇到的一些情况和挑战。

Fluke公司的发展小趣事

随着市场的不断扩大,BOOKHAM公司意识到单打独斗难以取得更大的突破。于是,公司开始积极寻求与其他企业的战略合作。通过与一家知名电子设备制造商的合作,BOOKHAM公司成功将其光学元件产品集成到对方的产品中,共同开拓市场。这种合作不仅提升了BOOKHAM公司的市场份额,还为其带来了更多的商业机会和合作伙伴。

问答坊 | AI 解惑

PADS layout BOM速成工具(自动生成EXCEL文档)

power pcb 自动生成excel文档BOM工具 附件中附带有使用方法! [ 本帖最后由 332 于 2008-11-13 13:42 编辑 ]…

查看全部问答>

如何成为单片机高手

1.不要看到别人的回复第一句话就说:给个代码吧!你应该想想为什么。当你自己想出来再参考别人的提示,你就知道自己和别人思路的差异。 2.别小家子气,买本书几十块都舍不得,你还学个P。为了省钱看电子书,浪费的时间绝对超过书的价值。当然如果 ...…

查看全部问答>

判断网线的好坏:除了万用表,还能看重量

线缆的主要组成部门在于铜芯的质量,它里面是否含其他金属以及粗细,都是取决于整批线路好坏的根本,所以重量的测试方法只能是测试方法之一,而且微不足道。要看测试还得用专门的仪器查看网线的各项数据! 另外,请问用万用表如何测?…

查看全部问答>

从事嵌入式开发的牛人请进来看看

我们公司目前要招一批从事嵌入式开发的人才,如果您有意加入,请将您的简历发送给我。 公司名称:中兴通讯(ZTE) 工作地点:南京 本人邮箱:zhuang.li2@zte.com.cn 主要职责: 路由器平台软件的开发 招聘要求: 1.本科毕业后工作三年以上, ...…

查看全部问答>

电棍拐杖

这款拐杖不仅可以根据需要调节长度,还带有方便的照明手电筒,更为特别的是:拐杖底部安装了高压电击枪,帮您在特殊情况下脱离险境。…

查看全部问答>

本人要做一个音频信号源求助 北京 有偿

本人要做一个音频信号源求助 北京 有偿 自己已试做了一个噪音大去不掉。…

查看全部问答>

MSP430 模拟开关 48路AD转换

LZ想用3个16掷1的模拟开关加上430自带的AD转换模块,来实现48路AD转换,行的通么? 片子选型改注意什么? 能不能在10ms内完成48路的AD转换? 求各路大神指点迷津。。。 来杭州有bg [ 本帖最后由 namemore 于 2012-4-20 19:04 编辑 ]…

查看全部问答>

什么器件可以测能见度

请教各位大侠,什么器件可以测能见度?…

查看全部问答>