搜索
bottom↓
回复: 15

C51+CH375读U盘实例

[复制链接]

出0入0汤圆

发表于 2012-4-11 00:33:25 | 显示全部楼层 |阅读模式
本帖最后由 PCA 于 2012-4-11 20:43 编辑

找到了一个现成的例子,对于读写U盘,文件系统是个大麻烦。现在一直还没有弄明白,先收藏一下这个程序先,等到后面慢慢研究。
作者这个编程规范不错,注释很丰富,值得学习。

#include  
#include "CH375INC.H"
#include   /* 以下定义适用于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, %02X
", (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 disk
" );
    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: %s
", 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: %s
", DISK_BUFFER );  /* 显示首行  
*/
        Cluster = mLinkCluster( Cluster );  /* 获取链接簇,返回0说明错误 */
      }
    }
    while ( mWaitInterrupt( ) != USB_INT_DISCONNECT );  /* 等待U盘拔出 */
    mDelaymS( 250 );
  }
}

阿莫论坛20周年了!感谢大家的支持与爱护!!

曾经有一段真挚的爱情摆在我的面前,我没有珍惜,现在想起来,还好我没有珍惜……

出0入0汤圆

发表于 2012-4-11 08:16:11 | 显示全部楼层
楼主把字体设置好大,看得眼花

出0入0汤圆

发表于 2012-4-11 08:19:16 来自手机 | 显示全部楼层
资料只有代码吗? 源出处应该有点介绍说明吧

出0入0汤圆

发表于 2012-4-11 08:21:47 | 显示全部楼层
mark一下,以后有时间再学习。

出0入0汤圆

发表于 2012-4-11 08:59:38 | 显示全部楼层
字体也太大了

出0入0汤圆

发表于 2012-4-11 09:04:01 | 显示全部楼层
字体太大了

出0入0汤圆

发表于 2012-4-11 09:06:51 | 显示全部楼层
楼主位把字体编辑下,不知道你这样做是为了更醒目 还是什么……

出0入0汤圆

发表于 2012-4-11 12:57:29 | 显示全部楼层
mark一下,刚好我有个375

出0入0汤圆

发表于 2012-4-11 13:33:24 | 显示全部楼层
这么好的资料,对新手来说太好了。谢谢!

出0入0汤圆

发表于 2012-4-11 23:30:01 | 显示全部楼层
这个,看起来挺费劲的..

出0入0汤圆

发表于 2012-4-12 00:01:20 | 显示全部楼层
先收藏慢慢看

出0入0汤圆

发表于 2012-4-12 08:02:26 | 显示全部楼层
mark,回去慢慢琢磨

出0入0汤圆

发表于 2012-4-12 08:44:06 | 显示全部楼层
先mark,在看!

出0入0汤圆

发表于 2012-4-12 08:57:49 | 显示全部楼层
顶一个,不错

出0入0汤圆

发表于 2012-4-12 09:05:58 | 显示全部楼层
留名以后在用吧!

出0入0汤圆

发表于 2012-11-9 20:40:47 | 显示全部楼层
这个代码我在csdn里见过,也不知楼主是否也是从那里复制来的。
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片。注意:要连续压缩2次才能满足要求!!】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-8-26 15:11

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表