单片机读写U盘闪盘超精简C源程序,不用子程序库

发布者:BlissfulBliss最新更新时间:2015-02-03 来源: newmaker关键字:单片机  读写U盘闪盘  C源程序 手机看文章 扫描二维码
随时随地手机看文章

/* 这个程序用180行C代码就能够读取FAT16文件系统U盘的根目录,可以看到根目录下的文件名,并可显示
首文件内容,不过,该程序很不严谨,也没有任何错误处理,对U盘兼容性较差,只是用于简单试验,作为参考.
这个程序可以支持WINDOWS按FAT16格式化的U盘,因为程序精简,所以只兼容超过50%以上的U盘品牌,如果换
成CH375A芯片则兼容性可提高到85%,当然,如果使用公司的子程序库或者正式版本的C源程序兼容性更好。
下面贴出的程序注释较少,如果需要加注释的C源程序,请到楼下的跟贴中下载,
欢迎各位将本源程序转贴到其它网站,比较一下,如何做出最简洁的单片机读写U盘文件的程序 :-)
测试以下U盘通过:郎科/超稳经典64M/超稳迷你128M/U160-64M/超稳普及128M,爱国者/迷你王16M/邮箱型,
黑匣子/64M,微闪/64M,飙王/32M/64M/128M,晶彩/C200-64M,新科/256M,昂达/128M...,欢迎提供测试结果
未通过U盘:爱国者/智慧棒128M,清华普天/USB2.0-128M,当然,使用子程序库或CH375A都可以测试通过 */

 

#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\n", (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\n" );
    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\n", 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\n", DISK_BUFFER );  /* 显示首行 */
        Cluster = mLinkCluster( Cluster );  /* 获取链接簇,返回0说明错误 */
      }
    }
    while ( mWaitInterrupt( ) != USB_INT_DISCONNECT );  /* 等待U盘拔出 */
    mDelaymS( 250 );
  }
}

关键字:单片机  读写U盘闪盘  C源程序 引用地址:单片机读写U盘闪盘超精简C源程序,不用子程序库

上一篇:用RS-485构成总线型多点数据采集系统
下一篇:单片机处理密码的简单方法

推荐阅读最新更新时间:2024-03-16 13:52

基于单片机的加热炉温度模糊控制系统
   1 引 言  热处理加热炉具有大惯性、纯滞后等非线性以及时变的特点,开关炉门、加热材料、环境温度以及电网电压等都影响控制过程,基于精确数学模型的常规控制例如PID控制难以保证加热工艺曲线要求。作为非线性控制的一大分支,模糊控制在上述温度控制系统中可以得到较好的应用。   模糊控制是智能控制的分支之一,他具有以下特点:    2模糊控制器的设计   本控制系统主要完成数据采集、温度显示、炉温控制、故障检测以及报警等功能,智能模糊控制器由单片机完成,采用规则自寻优的控制算法进行过程控制。加热炉采用双向可控硅控制,由单片机输出通断率控制信号,产生可控硅的过零触发脉冲。   整个系统的核心是模糊控制器,8098单片机
[单片机]
基于<font color='red'>单片机</font>的加热炉温度模糊控制系统
基于LJD-ZN-8400T智能触摸液晶显示终端与单片机的接口
  作为一种常见的显示设备,LCD具有功耗低,体积小,重量轻,显示信息量大等特点,尤其是在便携设备中,更占据着绝对主流的地位。对于字符或数字的简单信息显示,通常可用段码式LCD驱动器。对于复杂的信息显示,则必须由点阵式LCD驱动器来显示。最常见的点阵式LCD控制/驱动器中,有不带中文字库的LCD驱动器,也有带中文字库的LCD驱动器。随着嵌入式系统的广泛应用,出现了大量的16位和32位的嵌入式处理器。然而传统的8位单片机长期用于生产实践中,制造工艺成熟,性能更加可靠,因此仍然占有相当大的市场,特别是在汽车电子等对可靠性要求极高的领域。液晶显示器具有显示信息量丰富,功耗低,体积小,质量轻,无辐射等优点。触摸屏作为一种特殊的计算机外设,
[单片机]
基于LJD-ZN-8400T智能触摸液晶显示终端与<font color='red'>单片机</font>的接口
51单片机 | 串口通信实验(模拟串口通信/多机通信实例)
实验7.4:RS232串口应用实例 设计要求: 通过串口发送接收数据(串口工作方式1) 实现思路: 定时器设置 串口设置 工作方式 波特率 开启定时/计数器 串口执行 接收数据 发送数据 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 实现代码: 1 #include reg51.h 2 main() 3 { 4 TMOD = 0x20; 5
[单片机]
51<font color='red'>单片机</font> | 串口通信实验(模拟串口通信/多机通信实例)
ATMega16单片机外部中断程序分享
//Crystal:7.3728MHz,功能:学习外部中断0的程序 #include #include #defineLED_COMPORTA^=(1《voidport_init(void) { PORTA=0x40; DDRA=0x40; PORTB=0x00; DDRB=0x00; PORTC=0x00;//m103outputonly DDRC=0x00; PORTD=0x04; DDRD=0x00; } #pragmainterrupt_handlerint0_isr:2 voidint0_isr(void) { LED_COM; } //callthisroutinetoiniTIalizeallperiphera
[单片机]
ATMega16<font color='red'>单片机</font>外部中断程序分享
PIC16C74单片机SPI方式读写串行EEPROM程序
; list p=16C74, st=off ; PORTC PIN DESCRIPTION ; SCK bit 3, SDI bit 4, SDO bit 5, CS bit 7 ; Fosc = 10.0 MHz, thus each instr. cycle = 400ns ;***************Ram Register Definitions******************************* rxdata equ 25h addr equ 26h loops equ 27h ;***************Bit Definitions***************************
[单片机]
STM32单片机通信协议操作步骤及注意事项
STM32单片机通信协议是一种用于在不同设备之间进行数据传输的协议,它可以帮助设备之间进行高效的通信。STM32单片机通信协议可以用于实现多种不同的应用,如智能家居、智能安防、智能交通等。 STM32单片机通信协议的使用方法主要包括以下几个步骤: 1. 首先,需要确定使用的协议类型,如UART、I2C、SPI等,并确定使用的协议的具体参数,如波特率、数据位、停止位等。 2. 然后,需要在STM32单片机上配置相应的通信接口,并将其与外部设备连接起来。 3. 接着,需要编写相应的程序,实现STM32单片机与外部设备之间的数据传输。 4. 最后,需要将程序烧录到STM32单片机上,并运行程序,实现STM32单片机与外部设备之间的数
[单片机]
基于STM8S207单片机的测试架系统设计
0 引言 随着信息科学技术的发展,工业生产扩大和产品种类的增加,在科研与生产中,检测成为每一个电子产品必不可缺少的一个环节。对测试的要求也越来越高,有时因为对一块电路板要同时测试好多功能,而无法同时检测而影响产品质量。基于STM8S207单片机的强大功能,最高fCPU可达24 MHz,基于哈佛结构并带有3级流水线扩展指令集,最高20MIPS@24 MHz。程序存储器:最多128KB FLASH,10千次擦写后在55℃环境下数据可保存20年;数据存储器:最多2 KB真正的数据E2PROM;可达30万次擦写RAM:最多6KB。时钟、复位和电源管理2.95~5.5 V工作电压灵活的时钟控制,4个主时钟源,低功率晶体振荡器,外部时钟输
[单片机]
基于STM8S207<font color='red'>单片机</font>的测试架系统设计
8051单片机的历史_8051单片机的应用领域
  目前,有关8051单片机的说法很多:8051单片机是Intel发明的,可与8位系列处理器一起使用。现在,8051单片机已广泛应用于各个行业以及家庭的各种家用电器上。      一、8051单片机的历史   如果我们回顾历史,8051单片机是由微处理器巨头Intel于1980年首次发明的,并且逐渐被全世界所接受,并且随着未来的每一天,8051单片机的重要性正在不断提高。当它由英特尔发明时,它是通过NMOS技术开发的,但作为NMOS技术却不是很有效。   为了提高效率和生产率,英特尔通过实施CMOS技术对它进行了改型,并出现了一个新版本,名称中带有字母“C”,比如:“80C51”。新版8051单片机具有两条总线,一条总线用于程序
[单片机]
8051<font color='red'>单片机</font>的历史_8051<font color='red'>单片机</font>的应用领域
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved