/* CH376芯片 硬件标准SPI串行连接的硬件抽象层 V1.0 */ /* 提供I/O接口子程序 */ #include "HAL.H" #include /* 本例中的硬件连接方式如下(实际应用电路可以参照修改下述定义及子程序) */ /* 单片机的引脚 CH376芯片的引脚 P1.4 SCS P1.5 SDI P1.6 SDO P1.7 SCK */ sbit P14 = P1^4; sbit P16 = P1^6; sbit P10= P1^0; #define CH376_SPI_SCS P14 /* 假定CH376的SCS引脚 */ #define CH376_SPI_SDO P16 /* 假定CH376的SDO引脚 */ #define CH376_SPI_BZ P10 /* 假定CH376的BZ引脚 */ //sfr SPDR = 0x86; /* SPI数据寄存器 */ //sfr SPSR = 0xAA; /* SPI状态寄存器 */ //sfr SPCR = 0xD5; /* SPI控制寄存器 */ #define SPI_IF_TRANS 0x80 /* SPI字节传输完成标志,在SPSR的位7 */ sbit ch376int=P2^0; //#define CH376_INT_WIRE INT0 /* 假定CH376的INT#引脚,如果未连接那么也可以通过查询兼做中断输出的SDO引脚状态实现 */ #define CH376_INT_WIRE ch376int void init_spi(){ printf("init_spi \n"); SPCTL=0x5C; SPSTAT=0xC0; SPDAT=0; //EA=1; //ES=1; } void spi_rec() interrupt 9 { printf(" spi_rec() \n"); SPSTAT=0xC0; } void CH376_PORT_INIT( void ) /* 由于使用SPI读写时序,所以进行初始化 */ { /* 如果是硬件SPI接口,那么可使用mode3(CPOL=1&CPHA=1)或mode0(CPOL=0&CPHA=0),CH376在时钟上升沿采样输入,下降沿输出,数据位是高位在前 */ CH376_SPI_SCS = 1; /* 禁止SPI片选 */ /* 对于双向I/O引脚模拟SPI接口,那么必须在此设置SPI_SCS,SPI_SCK,SPI_SDI为输出方向,SPI_SDO为输入方向 */ //SPCTL = 0x5C; /* 设置SPI模式3, DORD=0(MSB first), CPOL=1, CPHA=1, CH376也支持SPI模式0 */ } void mDelay0_5uS( void ) /* 至少延时0.5uS,根据单片机主频调整 */ { UINT8 i=5,j=0; while(i--){ j=12; while(j--); } } UINT8 Spi376Exchange( UINT8 d ) /* 硬件SPI输出且输入8个位数据 */ { /* 为了提高速度,可以将该子程序做成宏以减少子程序调用层次 */ SPDAT = d; /* 先将数据写入SPI数据寄存器,然后查询SPI状态寄存器以等待SPI字节传输完成 */ printf("Spi376Exchange \n"); while ( ( SPSTAT & SPI_IF_TRANS ) == 0 ); /* 查询SPI状态寄存器以等待SPI字节传输完成 */ SPSTAT=0xC0; /* 清除SPI字节传输完成标志,有的单片机会自动清除 */ printf("SPSTAT & SPI_IF_TRANS \n"); return ( SPDAT ); /* 先查询SPI状态寄存器以等待SPI字节传输完成,然后从SPI数据寄存器读出数据 */ } #define xEndCH376Cmd( ) { CH376_SPI_SCS = 1; } /* SPI片选无效,结束CH376命令,仅用于SPI接口方式 */ void xWriteCH376Cmd( UINT8 mCmd ) /* 向CH376写命令 */ { #ifdef CH376_SPI_BZ UINT8 i; UINT8 rs; #endif printf("ds \n"); CH376_SPI_SCS = 1; /* 防止之前未通过xEndCH376Cmd禁止SPI片选 */ /* 对于双向I/O引脚模拟SPI接口,那么必须确保已经设置SPI_SCS,SPI_SCK,SPI_SDI为输出方向,SPI_SDO为输入方向 */ CH376_SPI_SCS = 0; /* SPI片选有效 */ printf(" for Spi376Exchange mCmd \n"); rs=Spi376Exchange( mCmd ); /* 发出命令码 */ printf("Spi376Exchange mCmd %02X \n",rs); #ifdef CH376_SPI_BZ printf("eb CH376_SPI_BZ \n"); for ( i = 30; i != 0 && CH376_SPI_BZ; -- i ); /* SPI忙状态查询,等待CH376不忙,或者下面一行的延时1.5uS代替 */ #else printf("uneb CH376_SPI_BZ \n"); mDelay0_5uS( ); mDelay0_5uS( ); mDelay0_5uS( ); /* 延时1.5uS确保读写周期大于1.5uS,或者用上面一行的状态查询代替 */ #endif } #ifdef FOR_LOW_SPEED_MCU /* 不需要延时 */ #define xWriteCH376Data( d ) { Spi376Exchange( d ); } /* 向CH376写数据 */ #define xReadCH376Data( ) ( Spi376Exchange( 0xFF ) ) /* 从CH376读数据 */ #else void xWriteCH376Data( UINT8 mData ) /* 向CH376写数据 */ { printf(" for xWriteCH376Data \n"); Spi376Exchange( mData ); printf(" xWriteCH376Data success \n"); mDelay0_5uS( ); /* 确保读写周期大于0.6uS */ } UINT8 xReadCH376Data( void ) /* 从CH376读数据 */ { mDelay0_5uS( ); /* 确保读写周期大于0.6uS */ CH376_SPI_SCS = 1; CH376_SPI_SCS = 0; mDelayuS(10); return( Spi376Exchange( 0xFF ) ); } #endif /* 查询CH376中断(INT#低电平) */ UINT8 Query376Interrupt( void ) { #ifdef CH376_INT_WIRE printf(" Query376Interrupt CH376_INT_WIRE \n"); return( CH376_INT_WIRE ? FALSE : TRUE ); /* 如果连接了CH376的中断引脚则直接查询中断引脚 */ #else return( CH376_SPI_SDO ? FALSE : TRUE ); /* 如果未连接CH376的中断引脚则查询兼做中断输出的SDO引脚状态 */ #endif } UINT8 mInitCH376Host( void ) /* 初始化CH376 */ { UINT8 res; printf("mInitCH376Host \n"); init_spi(); CH376_PORT_INIT( ); /* 接口硬件初始化 */ printf("CH376_PORT_INIT \n"); xWriteCH376Cmd( CMD11_CHECK_EXIST ); /* 测试单片机与CH376之间的通讯接口 */ printf("CMD11_CHECK_EXIST \n"); xWriteCH376Data( 0x65 ); printf("xWriteCH376Data \n"); res = xReadCH376Data( ); printf("xReadCH376Data \n"); xEndCH376Cmd( ); printf("xEndCH376Cmd \n"); if ( res != 0x9A ){ printf("res != 0x9A \n"); return( ERR_USB_UNKNOWN ); /* 通讯接口不正常,可能原因有:接口连接异常,其它设备影响(片选不唯一),串口波特率,一直在复位,晶振不工作 */ } xWriteCH376Cmd( CMD11_SET_USB_MODE ); /* 设备USB工作模式 */ printf("CMD11_SET_USB_MODE \n"); xWriteCH376Data( 0x05 ); printf("xWriteCH376Data \n"); mDelayuS( 20 ); res = xReadCH376Data( ); printf("xReadCH376Data \n"); xEndCH376Cmd( ); printf("xEndCH376Cmd \n"); #ifndef CH376_INT_WIRE #ifdef CH376_SPI_SDO xWriteCH376Cmd( CMD20_SET_SDO_INT ); /* 设置SPI的SDO引脚的中断方式 */ printf("CMD20_SET_SDO_INT \n"); xWriteCH376Data( 0x16 ); printf("xWriteCH376Data \n"); xWriteCH376Data( 0x90 ); /* SDO引脚在SCS片选无效时兼做中断请求输出 */ printf("xWriteCH376Data \n"); xEndCH376Cmd( ); printf("xEndCH376Cmd \n"); #endif #endif if ( res == CMD_RET_SUCCESS ) return( USB_INT_SUCCESS ); else return( ERR_USB_UNKNOWN ); /* 设置模式错误 */ }