电路图是按照资料提供的串口模式连接直接接MCU,5V供电,现在问题是确认硬件连接无误,不插U盘时ACT#为高,INT#也为高,RST为低,RST#为高,TXD为5V,RXD为3.75,晶振2.2V左右用示波器测得振荡波形正常,但查了U盘后,U盘里面的灯闪烁很正常,但是,ACT#还是为高,INT#也为高,RST还是为低,RST#为高,用KEIL4编译,程序总是在unsigned char xReadCH375Data( void ) { while ( RI0 == 0 );//每次都卡在这,等待UART0接受中断 RI0 = 0; return( SBUF0 ); } 程序用的FILELIB5里面的CH375HF5.LIB和CH375HF5.H及EXAM7里面的CH375HFT.C,只是将并口的读写屏蔽了用的串口的读写,然后写了串口初始化程序,同时添加了C8051F060初始化的几个函数,如下 /* CH375 主机文件系统接口 */ /* 支持: FAT12/FAT16/FAT32 */
/* MCS-51单片机C语言的U盘文件读写示例程序 */ /* 用于以下情况的MCS51单片机 1. 高速MCS51单片机,机器周期小于0.3uS,或者在机器周期为12个时钟时的时钟频率大于40MHz 2. 非总线MCS51单片机,用普通I/O引脚模拟8位并行总线读写,与CH375之间采用并口连接 3. 单片机与CH375之间采用串口连接 */ /* 本程序用于演示将ADC模数采集的数据保存到U盘文件MY_ADC.TXT中 */ /* CH375的INT#引脚采用查询方式处理,本例用普通I/O引脚模拟8位并行总线读写,同时提供串口连接示例, 以字节为单位读写U盘文件,读写速度较扇区模式慢,但是由于字节模式读写文件不需要文件数据缓冲区FILE_DATA_BUF, 所以总共只需要600字节的RAM,适用于单片机硬件资源有限、数据量小并且读写速度要求不高的系统 */
#include #include #include "CH375HF5.H" #define BAUDRATE 9600 // UART波特率bps #define SYSCLK 22118400 // System clock frequency in Hz
#define MAX_BYTE_IO 35 /* 以字节为单位单次读写文件时的最大长度,默认值是29,值大则占用内存多,值小则超过该长度必须分多次读写 */
/* 以下定义的详细说明请看CH375HF5.H文件 */ #define LIB_CFG_FILE_IO 1 /* 文件读写的数据的复制方式,0为"外部子程序",1为"内部复制" */ #define LIB_CFG_INT_EN 0 /* CH375的INT#引脚连接方式,0为"查询方式",1为"中断方式" */ /*#define LIB_CFG_UPD_SIZE 1*/ /* 在添加数据后是否自动更新文件长度: 0为"不更新",1为"自动更新" */ /* 默认情况下,如果扇区数/字节数不为0那么CH375FileWrite/CH375ByteWrite只负责写入数据而不修改文件长度, 如果需要每次写完数据后会自动修改/更新文件长度,那么可以使全局变量CH375LibConfig的位4为1, 如果长时间不写入数据则应该更新文件长度,防止突然断电后前面写入的数据与文件长度不相符, 如果确定不会突然断电或者后面很快有数据不断写入则不必更新文件长度,可以提高速度并减少U盘损耗(U盘内部的内存寿命有限,不宜频繁改写) */
/* 只使用单片机内置的1KB外部RAM: 0000H-01FFH 为磁盘读写缓冲区, 以字节为单位读写文件不需要文件数据读写缓冲区FILE_DATA_BUF */ #define DISK_BASE_BUF_ADDR 0x0000 /* 外部RAM的磁盘数据缓冲区的起始地址,从该单元开始的缓冲区长度为SECTOR_SIZE */ #define DISK_BASE_BUF_LEN 2048 /* 默认的磁盘数据缓冲区大小为512字节,建议选择为2048甚至4096以支持某些大扇区的U盘,为0则禁止在.H文件中定义缓冲区并由应用程序在pDISK_BASE_BUF中指定 */ #define FILE_DATA_BUF_ADDR 0x0000 /* 外部RAM的文件数据缓冲区的起始地址,缓冲区长度不小于一次读写的数据长度,字节模式不用该缓冲区 */ /* 由于单片机内置的外部RAM只有1KB, 有些单片机还要去掉256字节内部RAM, 只剩下768字节的外部RAM, 其中前512字节由CH375子程序用于磁盘数据缓冲 */ #define FILE_DATA_BUF_LEN 0x0800 /* 外部RAM的文件数据缓冲区,缓冲区长度不小于一次读写的数据长度,字节模式不用该缓冲区 */
#define CH375_INT_WIRE P3^2 /* P3.2, INT0, CH375的中断线INT#引脚,连接CH375的INT#引脚,用于查询中断状态 */
#define NO_DEFAULT_CH375_F_ENUM 1 /* 未调用CH375FileEnumer程序故禁止以节约代码 */ #define NO_DEFAULT_CH375_F_QUERY 1 /* 未调用CH375FileQuery程序故禁止以节约代码 */
//#include "..\CH375HF5.H"
/* 在P1.4连接一个LED用于监控演示程序的进度,低电平LED亮,当U盘插入后亮 */ sbit P1_4 = P1^4; #define LED_OUT_INIT( ) { P1_4 = 1; } /* P1.4 高电平 */ #define LED_OUT_ACT( ) { P1_4 = 0; } /* P1.4 低电平驱动LED显示 */ #define LED_OUT_INACT( ) { P1_4 = 1; } /* P1.4 低电平驱动LED显示 */ sbit P1_5 = P1^5; /* 在P1.5连接一个LED用于监控演示程序的进度,低电平LED亮,当对U盘操作时亮 */ #define LED_RUN_ACT( ) { P1_5 = 0; } /* P1.5 低电平驱动LED显示 */ #define LED_RUN_INACT( ) { P1_5 = 1; } /* P1.5 低电平驱动LED显示 */ sbit P1_6 = P1^6; /* 在P1.6连接一个LED用于监控演示程序的进度,低电平LED亮,当对U盘写操作时亮 */ #define LED_WR_ACT( ) { P1_6 = 0; } /* P1.6 低电平驱动LED显示 */ #define LED_WR_INACT( ) { P1_6 = 1; } /* P1.6 低电平驱动LED显示 */
void SYSCLK_Init (void) { char old_SFRPAGE = SFRPAGE; int i; SFRPAGE = CONFIG_PAGE; // Switch to Configuration Page OSCXCN = 0x67; // start external oscillator with // 22.1184MHz crystal on TB for (i=0; i <5000; i++) ; // XTLVLD blanking interval (>1ms) while (!(OSCXCN & 0x80)) ; // Wait for crystal osc. to settle RSTSRC = 0x04; // enable missing clock detector reset CLKSEL = 0x01; // change to external crystal OSCICN = 0x00; // disable internal oscillator SFRPAGE = old_SFRPAGE; // restore SFRPAGE }
void Port_Init (void) { char old_SFRPAGE = SFRPAGE; SFRPAGE = CONFIG_PAGE; // Switch to configuration page XBR0 = 0x07; // Enable UART0 on crossbar XBR2 = 0x44; // Enable crossbar and weak pull-ups P1MDOUT |= 0xF0; // Set P1.4 P1.6 1.7(LED) to push-pull P3MDOUT =0x00; P7MDOUT =0xff; // Set P7 to push-pull SFRPAGE = old_SFRPAGE; // restore SFRPAGE } void EMIF_Init (void) { char SFRPAGE_SAVE = SFRPAGE; // Save Current SFR page SFRPAGE = EMI0_PAGE; // Save SFR_PAGE EMI0CF = 0x00; // upper ports; non-muxed mode; 选择内部存储器 // split mode w/o bank select EMI0TC = 0x45; // timing (7-cycle MOVX) SFRPAGE = CONFIG_PAGE; P3MDOUT =0x00; P4MDOUT |= 0xFF; // all EMIF pins configured as P5MDOUT |= 0xFF; // push-pull P6MDOUT |= 0xFF; P7MDOUT |= 0xFF;
SFRPAGE = SFRPAGE_SAVE; // restore SFR_PAGE } //----------------------------------------------------------------------------- // UART0 初始化 //----------------------------------------------------------------------------- // // 配置UART0 使用定时器1为波特率发生器 // /*void UART0_Init (void) { SCON0 = 0xD0; // SCON0: 模式9, 9 位UART, 使能RX //SCON0 = 0x50; // SCON0: 模式1, 8 位UART, 使能RX TMOD = 0x20; // TMOD: 定时器1, 模式2, 8 位重装 TH1 = -(SYSCLK/BAUDRATE/16); // 根据波特率的值设定定时器1重装值 TR1 = 1; // 启动定时器1 CKCON |= 0x10; // 定时器1使用系统时钟作为时基 PCON |= 0x80; // SMOD00 = 1 TI0 = 1; // 表示TX0就绪 }*/ void T0_Wait_us (unsigned char us) { SFRPAGE = TIMER01_PAGE; TCON &= ~0x30; // 停止定时器T0并清除溢出标志 TMOD &= ~0x0f; // 配置定时器T0为16位模式 TMOD |= 0x01; CKCON |= 0x08; // 定时器T0计数系统时钟 while (us) { TR0 = 0; // Stop Timer0 TH0 = -(SYSCLK/1000000 >> 8); // Overflow in 1us TL0 = -(SYSCLK/1000000); TF0 = 0; // Clear overflow indicator TR0 = 1; // Start Timer0 while (!TF0); // Wait for overflow us--; // Update ms counter } TR0 = 0; } void T0_Wait_ms (unsigned char ms) { char SFRPAGE_SAVE = SFRPAGE; // Save Current SFR page SFRPAGE = TIMER01_PAGE; TCON &= ~0x30; // Stop Timer0; Clear TF0 TMOD &= ~0x0f; // 16-bit free run mode TMOD |= 0x01; CKCON |= 0x08; // Timer0 counts SYSCLKs while (ms) { TR0 = 0; // Stop Timer0 TH0 = -(SYSCLK/1000 >> 8); // Overflow in 1ms