C8051F020 UART0自发自收数据 c语言

时间: 2023-09-01 21:11:39 浏览: 64
以下是C8051F020的UART0自发自收数据的C语言代码示例: ```c #include <C8051F020.h> #define SYSCLK 22118400 // 系统时钟频率,单位为Hz #define BAUDRATE 9600 // 波特率 #define SCALE_FACTOR 256 // 定时器预分频因子 unsigned char buffer[64]; // 缓冲区 unsigned char buf_head = 0; // 缓冲区头指针 unsigned char buf_tail = 0; // 缓冲区尾指针 void UART0_ISR() interrupt 4 // UART0中断服务程序 { if (RI0 == 1) // 接收到了数据 { buffer[buf_tail] = SBUF0; // 将接收到的数据存入缓冲区 buf_tail = (buf_tail + 1) % 64; // 更新缓冲区尾指针 RI0 = 0; // 清除接收中断标志位 } if (TI0 == 1) // 发送完了数据 { TI0 = 0; // 清除发送中断标志位 } } void UART0_init() { SCON0 = 0x50; // 8位数据,无校验,1位停止位 TMOD &= ~0xf0; // 清除定时器1模式位 TMOD |= 0x20; // 定时器1工作在模式2 TH1 = -(SYSCLK / (SCALE_FACTOR * BAUDRATE)); // 计算定时器重载值 TL1 = TH1; // 初值等于重载值 TR1 = 1; // 启动定时器1 ET0 = 0; // 关闭定时器0中断 ES0 = 1; // 允许UART0中断 EA = 1; // 打开总中断开关 } void main() { UART0_init(); // 初始化UART0 while (1) { while (buf_head != buf_tail) // 当缓冲区不为空时 { SBUF0 = buffer[buf_head]; // 发送缓冲区中的数据 buf_head = (buf_head + 1) % 64; // 更新缓冲区头指针 while (TI0 == 0); // 等待发送完成 } } } ``` 该代码中,首先定义了系统时钟频率、波特率和定时器预分频因子等常量,然后定义了一个长度为64的缓冲区和两个指针,分别指向缓冲区的头和尾。接着编写了一个UART0中断服务程序,用于处理接收和发送数据。在主函数中,首先调用了初始化UART0的函数,然后进入一个死循环,不断检查缓冲区是否为空。如果缓冲区不为空,则将缓冲区中的数据发送出去,并等待发送完成。如果缓冲区为空,则继续等待。

相关推荐

doc
//;UART0收发一组字符(48个)(通过) // rxd_str必须是全局变量,切使用前要赋值。 // rxd_str不能是局部变量,局部变量会清零, //因为中断一次只能收一个字符rxd_str不可能>=LENM /********************************************** 文件描述: 三相电测量上报 功能说明: 测量ATT7022B完成 PCF8563上报定时(需校时) FM24C256转换数据存储 UART0口用于RS232读数据 UART1控制无线模块上报数据 创建:2006年12月5日 /********************************* *函数:main. *入口:无 *出口:无 *功能说明:接收UART端口命令 执行读ATT7022B的数据 进行处理、存储和回发数据 ********************************/ //***************************** //包含的文件 //****************************** #include <c8051f020.h> #include <main.h> #include <intrins.h> //************************************************* //函数:init_sysclk(void) //功能:时钟初始化 //入口:无 //出口: 无 //说明:使用外部时钟12M //************************************************* void init_sysclk (void) { uint i=0; OSCXCN=0x67; //external oscillator with 12MHz crystal for(i=0;i<256;i++); // XTLVLD blanking while(!(OSCXCN & 0x80)); // Wait for crystal osc. to settle OSCICN=0x88; //时钟丢失检测,选择外部时钟 CKCON = 0x00; //时钟分频 } /********************************* 函数:void enable_wdog(void) void disable_wdog(void) 功能:使能、禁止wdog 入口:无 出口:无 说明: ********************************/ void enable_wdog(void) { WDTCN=0xA5; //允许看门狗定时器工作 } void disable_wdog(void) //禁止看门狗定时器工作 { WDTCN=0xDE; WDTCN=0xAD; } /*********************************** 函数:void init_ioport() 功能:端口配置及端口位定义 入口:无 出口:无 说明: *************************************/ void init_ioport() { XBR0=0x07; //TXD0-P0.0 RXD0-P0.1,SPI_SCK-P0.2,SPI_MISO-P0.3 //SPI_MOSI-P0.4,SPI_NSS-P0.5,SDA-P0.6,SCL-P0.7, XBR2=0x44; //端口I/O弱上拉允许,TX1-P1.0,RXD-P1.1交叉开关允许 XBR1=0X10; //INT1使能INT1--P1.2 P0MDOUT=0x00; //端口0输出方式寄存器:0--漏极开路 P0=0xff; P1MDIN=0XFF; //端口1输入方式寄存器:0--配置为数字输入 P1MDOUT=0x00; //端口1输出方式寄存器,0--漏极开路 P1=0xff; P2MDOUT=0x00; //端口2输出方式寄存器:0--漏极开路 P2=0xff; P3MDOUT=0X00; //端口3输出方式寄存器:0--漏极开路 P3=0xff; P74OUT=0x00; //端口7-4输出方式寄存器:0--漏极开路 P4=0xff; P5=0xff; P6=0xff; P7=0xff; } //-------------------------------------------------------------- //函数:void Delay_ms (unsigned ms) //功能:实现延时功能 Timer0_ms //--------------------------------------------------------------- // /* Configure Timer0 to delay <ms> */ void Delay_ms (unsigned ms) { uchar i; // millisecond counter TCON &= ~0x30; // STOP Timer0 and clear overflow flag TMOD |= 0x01; // configure Timer0 to 16-bit mode CKCON |= 0x08; // Timer0 counts SYSCLKs for (i = 0; i < ms; i++) // count milliseconds { TR0 = 0; // STOP Timer0 TH0 = (-SYSCLK/1000) >> 8; // set Timer0 to overflow in 1ms TL0 = -SYSCLK/1000; TR0 = 1; // START Timer0 while (TF0 == 0); // wait for overflow TF0 = 0; // clear overflow indicator } } //------------------------------------------------------- //函数:void Delay_us (unsigned us) //功能:实现延时功能 Timer0_us //------------------------------------------------------- // /* Configure Timer0 to delay <us>*/ void Delay_us (unsigned us) { uchar i; // microseconds counter TCON &= ~0x30; // STOP Timer0 and clear overflow flag TMOD |= 0x01; // configure Timer0 to 16-bit mode CKCON |= 0x08; // Timer0 counts SYSCLKs for (i = 0; i < us; i++) // count microseconds { TR0 = 0; // STOP Timer0 TH0 = (-SYSCLK/1000000) >> 8; // set Timer0 to overflow in 1us TL0 = -SYSCLK/1000000; TR0 = 1; // START Timer0 while (TF0 == 0); // wait for overflow TF0 = 0; // clear overflow indicator } } //*************************************** //函数:unsigned char my_add(uchar my_add) //功能:读开关状态确定子地址 //入口:无 //出口:子地址 //说明:子地址存于myadd中(即设备号) //**************************************** void my_add(void) { P74OUT |=0XD0; myadd =P5; } //****************************************** //函数:void jiaob(ATT_JB[]) //功能:写校表寄存器 //说明:UART口接收校表时间并存于FM24C256中 //***************************************** void jiaob (void) { } main() { disable_wdog(); //关看门狗 init_sysclk (); //时钟初始化 init_ioport(); //交叉开关配置 my_add(); //读设备子地址设置 UART0_Init(); //UART0初始化 EX1= 1; //开INT1 EA = 1; //开中断 rxd_str=0; while(1) { if(uart0_flag) { rxd_str=0; uart0_flag = 0; m=ur0_rxd; txd_string(m,LENM); } } //---------判断本设备命令及命令内容执行命令------- } //----------------------------------------------------- //串口初始化 //-------------------------------------------------------- void UART0_Init(void) { SCON0 = 0x50; //串口方式1,波特率可变 PCON |= 0x00; //SMOD = 0 TMOD = 0x20; //选择T1方式2, TH1 = 0xe8; //T1初值, TL1 = 0xe8; ES0 = 1; //UART0中断开启 TR1 = 1; //启动定时器T1 } //---------------------------------------------------------- //发送单个字符 //--------------------------------------------------------- void txd_char(unsigned char ch) { SBUF0 = ch; //送入缓冲区 while(TI0 == 0); //等待发送完毕 TI0 = 0; //软件清零 } //----------------------------------------------------- //发送字符串,调用Send_Char() len字符串长度 //---------------------------------------------------- void txd_string(unsigned char * str,unsigned char len) { unsigned char k = 0; do { txd_char(*(str+k) ); k++; } while(k < len); } //-------------------------------------------------------- //UART0中断服务程序. 接收字符 //-------------------------------------------------------- // rxd_str必须是全局变量,切使用前要赋值。 // rxd_str不能是局部变量,局部变量会清零, //因为中断一次只能收一个字符rxd_str不可能>=LENM void uart0_isr(void) interrupt 4 using 1 { unsigned char rxch; if(RI0) //中断标志 RI0=1 数据完整接收 { RI0 = 0; //软件清零 rxch = SBUF0; //读缓冲 if(rxd_str>=LENM) { uart0_flag=1; rxd_str=0; } ur0_rxd[rxd_str] = rxch; //存入数组,供发送 rxd_str++; } } //********************** //main.h //********************* #ifndef _main_h #define _main_h //***************************** //全局常量 //**************************** #define uchar unsigned char #define uint unsigned int #define SYSCLK 11059200 #define CMD_RESET 0X11 #define CMD_TIME 0X12 #define CMD_DATA 0X13 #define ATT_R 0x00 // ATT Read command #define ATT_W 0x80 // ATT Write command #define fm_Write_add 0xA0 #define fm_Read_add 0xA1 #define Fm_add 0xA0 //fm24c256器件从地?#define LENM 0x30 unsigned char idata r_commond=0x01; //读命令代码 unsigned char idata w_time_commond=0x02; //校时代码 unsigned char xdata jb_commond=0xdd; //校表命令代码 //---------------------------------------------------- //全局变量 //---------------------------------------------------- unsigned char rxd_str; unsigned char * m; unsigned char xdata fm_read_buf[60]={0}; unsigned char xdata fm_write_buf[60]={0}; unsigned char xdata pcf_d[16]={0}; //data of pcf8563 unsigned char xdata *d_ptr; unsigned char xdata ur0_rxd[60]; //每次接收字符串 unsigned char xdata ur0_txd[60]; //要发送的字符串 unsigned char xdata att_rd[90]; unsigned char xdata att_wd[60]; // att7022 of data unsigned char xdata att_jb[40]; //校表数组(myadd=0) unsigned char idata period; //时间间隔 unsigned char idata myadd; unsigned char *str; unsigned char chksum; unsigned char ATT_W_ADD; // ATT Read status register unsigned char ATT_R_ADD; // ATT Write status register uchar slave_add,fm_ram_add,send_byte,write_num,read_num; uchar fm_send_count,fm_receive_count,fm_send_len,fm_receive_len; sbit SDA=P1^6; //*模拟I2C数据传送位*/ sbit SCL=P1^7; //*模拟I2C时钟控制位*/ sbit ATT_CS = P3^0; // ATT CS signal sbit D_E = P1^3; //485收发控制?bit ack; bit uart0_flag; //中断接收完成标志 bit uart1_flag; //中断接收完成标志 bit sm_busy; //收发开始置1,操作结束后由中断清0 bit fm_err_flag; //--------子函数声明------------------------------------- void uart0_isr(); //串口中断服务程序,接收字符 void UART0_Init(void); void txd_char(uchar ch); void txd_string(uchar * str,uchar len); void init_sysclk (void); void enable_wdog(void); void disable_wdog(void); void init_ioport(); void Delay_ms (unsigned ms); void Delay_us (unsigned us) ; void my_add(void); void jiaob (void); void ATT_Write (uchar ATT_WADD, uchar att_wd[]); unsigned char ATT_Read (uchar ATT_RADD,uchar att_rd[]); void SPI0_Init (void); void Start_I2c(); void Stop_I2c(); void Ack_I2c(); void SendByte(uchar x); uchar RcvByte(); bit ISendStr(uchar sla,uchar suba,uchar *d_ptr,uchar no); bit IRcvStr(uchar sla,uchar suba,uchar *d_ptr,uchar no); void WriteClock(void); void StartClock(void); void init_sysclk (void); void init_ioport(); void init_smbus(void); void smbus_receive (uchar chip_select,byte_address,receive_num); void smbus_send (uchar chip_select,byte_address,write_num); void enable_wdog(void); void disable_wdog(void); #endif

最新推荐

recommend-type

C8051f020 UART0

#include &lt;c8051f020.h&gt; #include #include //************************************************* //函数:init_sysclk(void) //功能:时钟初始化 //入口:无 //出口: 无 //说明:使用外部时钟12M //**********...
recommend-type

C8051f330单片机UART编程

C8051F系列单片机的UART接口通常涉及到以下几个寄存器: 1. 功能控制寄存器SCON0:用于设置UART0的工作模式,如是否启用多机通信、接收和发送允许、第九位数据传输以及接收和发送的标志位。 2. 数据缓冲寄存器SBUF0...
recommend-type

I2C,SPI,UART和CAN等常见通信协议的区别

SPI--Serial Peripheral Interface,(Serial Peripheral Interface:串行外设接口)串行外围设备接口,是Motorola公司推出的一种同步串行通讯方式,是一种三线同步总线,因其硬件功能很强,与SPI有关的软件就相当简单...
recommend-type

中颖SH88F2051 串口UART 设置

中颖SH88F2051串口UART设置是基于.keil C中的代码选项,选第一项用内部的16.6M,通过软件设置CLKCON = 0x20,将其2分频。下面是相关知识点的详细说明: 1. 串口UART设置:UART(Universal Asynchronous Receiver-...
recommend-type

UART串口字符串接收程序

在UART0_GetData()函数中,我们可以看到while循环的使用,以等待接收数据。查寻方式可以降低CPU占用率,提高系统的实时性。 知识点6:中断处理 在UART串口字符串接收程序中,中断处理是非常重要的。在UART0_Get...
recommend-type

京瓷TASKalfa系列维修手册:安全与操作指南

"该资源是一份针对京瓷TASKalfa系列多款型号打印机的维修手册,包括TASKalfa 2020/2021/2057,TASKalfa 2220/2221,TASKalfa 2320/2321/2358,以及DP-480,DU-480,PF-480等设备。手册标注为机密,仅供授权的京瓷工程师使用,强调不得泄露内容。手册内包含了重要的安全注意事项,提醒维修人员在处理电池时要防止爆炸风险,并且应按照当地法规处理废旧电池。此外,手册还详细区分了不同型号产品的打印速度,如TASKalfa 2020/2021/2057的打印速度为20张/分钟,其他型号则分别对应不同的打印速度。手册还包括修订记录,以确保信息的最新和准确性。" 本文档详尽阐述了京瓷TASKalfa系列多功能一体机的维修指南,适用于多种型号,包括速度各异的打印设备。手册中的安全警告部分尤为重要,旨在保护维修人员、用户以及设备的安全。维修人员在操作前必须熟知这些警告,以避免潜在的危险,如不当更换电池可能导致的爆炸风险。同时,手册还强调了废旧电池的合法和安全处理方法,提醒维修人员遵守地方固体废弃物法规。 手册的结构清晰,有专门的修订记录,这表明手册会随着设备的更新和技术的改进不断得到完善。维修人员可以依靠这份手册获取最新的维修信息和操作指南,确保设备的正常运行和维护。 此外,手册中对不同型号的打印速度进行了明确的区分,这对于诊断问题和优化设备性能至关重要。例如,TASKalfa 2020/2021/2057系列的打印速度为20张/分钟,而TASKalfa 2220/2221和2320/2321/2358系列则分别具有稍快的打印速率。这些信息对于识别设备性能差异和优化工作流程非常有用。 总体而言,这份维修手册是京瓷TASKalfa系列设备维修保养的重要参考资料,不仅提供了详细的操作指导,还强调了安全性和合规性,对于授权的维修工程师来说是不可或缺的工具。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

【进阶】入侵检测系统简介

![【进阶】入侵检测系统简介](http://www.csreviews.cn/wp-content/uploads/2020/04/ce5d97858653b8f239734eb28ae43f8.png) # 1. 入侵检测系统概述** 入侵检测系统(IDS)是一种网络安全工具,用于检测和预防未经授权的访问、滥用、异常或违反安全策略的行为。IDS通过监控网络流量、系统日志和系统活动来识别潜在的威胁,并向管理员发出警报。 IDS可以分为两大类:基于网络的IDS(NIDS)和基于主机的IDS(HIDS)。NIDS监控网络流量,而HIDS监控单个主机的活动。IDS通常使用签名检测、异常检测和行
recommend-type

轨道障碍物智能识别系统开发

轨道障碍物智能识别系统是一种结合了计算机视觉、人工智能和机器学习技术的系统,主要用于监控和管理铁路、航空或航天器的运行安全。它的主要任务是实时检测和分析轨道上的潜在障碍物,如行人、车辆、物体碎片等,以防止这些障碍物对飞行或行驶路径造成威胁。 开发这样的系统主要包括以下几个步骤: 1. **数据收集**:使用高分辨率摄像头、雷达或激光雷达等设备获取轨道周围的实时视频或数据。 2. **图像处理**:对收集到的图像进行预处理,包括去噪、增强和分割,以便更好地提取有用信息。 3. **特征提取**:利用深度学习模型(如卷积神经网络)提取障碍物的特征,如形状、颜色和运动模式。 4. **目标
recommend-type

小波变换在视频压缩中的应用

"多媒体通信技术视频信息压缩与处理(共17张PPT).pptx" 多媒体通信技术涉及的关键领域之一是视频信息压缩与处理,这在现代数字化社会中至关重要,尤其是在传输和存储大量视频数据时。本资料通过17张PPT详细介绍了这一主题,特别是聚焦于小波变换编码和分形编码两种新型的图像压缩技术。 4.5.1 小波变换编码是针对宽带图像数据压缩的一种高效方法。与离散余弦变换(DCT)相比,小波变换能够更好地适应具有复杂结构和高频细节的图像。DCT对于窄带图像信号效果良好,其变换系数主要集中在低频部分,但对于宽带图像,DCT的系数矩阵中的非零系数分布较广,压缩效率相对较低。小波变换则允许在频率上自由伸缩,能够更精确地捕捉图像的局部特征,因此在压缩宽带图像时表现出更高的效率。 小波变换与傅里叶变换有本质的区别。傅里叶变换依赖于一组固定频率的正弦波来表示信号,而小波分析则是通过母小波的不同移位和缩放来表示信号,这种方法对非平稳和局部特征的信号描述更为精确。小波变换的优势在于同时提供了时间和频率域的局部信息,而傅里叶变换只提供频率域信息,却丢失了时间信息的局部化。 在实际应用中,小波变换常常采用八带分解等子带编码方法,将低频部分细化,高频部分则根据需要进行不同程度的分解,以此达到理想的压缩效果。通过改变小波的平移和缩放,可以获取不同分辨率的图像,从而实现按需的图像质量与压缩率的平衡。 4.5.2 分形编码是另一种有效的图像压缩技术,特别适用于处理不规则和自相似的图像特征。分形理论源自自然界的复杂形态,如山脉、云彩和生物组织,它们在不同尺度上表现出相似的结构。通过分形编码,可以将这些复杂的形状和纹理用较少的数据来表示,从而实现高压缩比。分形编码利用了图像中的分形特性,将其转化为分形块,然后进行编码,这在处理具有丰富细节和不规则边缘的图像时尤其有效。 小波变换和分形编码都是多媒体通信技术中视频信息压缩的重要手段,它们分别以不同的方式处理图像数据,旨在减少存储和传输的需求,同时保持图像的质量。这两种技术在现代图像处理、视频编码标准(如JPEG2000)中都有广泛应用。