当前位置:首页 > 技术学院 > 热搜器件
[导读]STC12C5A60S2单片机的串口扩展

STC12C5A60S2单片机的串口从传统的一个扩展到了两个,
而且还增加了一个独立波特率发生器,把定时器1解放了出来,真的不是一般的方便,
还而且能用1T模式,速度大大滴提高了。。。
 
UART.C
 
 
#include <STC12C5A.H>     //STC12C5A系列单片机 
#include <intrins.h> 
#include "UART.H" 
 
#define uchar   unsigned char 
#define uint    unsigned int 
 
//缓存串口1和串口2接收到的字符 
uchar UART1_Recv_Val = 0; 
uchar UART2_Recv_Val = 0; 
 
 
 
void UART1_Init(uchar RELOAD, bit doubleBaud, bit timeMod) 

    SCON |= 0x50;       //串口1方式1,接收充许 
 
    BRT = RELOAD;       //波特率2400 
 
    if (timeMod == 1)       //1T 
    { 
        //T0x12   T1x12   UM0x6   BRTR    S2SMOD  BRTx12  EXTRAM  S1BRS 
        AUXR |= 0x15;       //串口1使用独立波特率发生器,独立波特率发生器1T 
    } 
    else                    //12T 
    { 
        AUXR |= 0x11; 
    } 
 
    if (doubleBaud == 1) 
    { 
        PCON |= 0x80;     //波特率加倍 
    } 
    else 
    { 
        PCON &= 0x7F;     //波特率不加倍 
    } 
 
    EA = 1; 
    ES = 1;             //充许串口1中断 

 
 
 
void UART2_Init(uchar RELOAD, bit doubleBaud, bit timeMod) 

    //S2SM0  S2SM1   S2SM2   S2REN   S2TB8   S2RB8   S2TI     S2RI 
    S2CON |= 0x50;      //串口2,方式1,接收充许 
 
    BRT = RELOAD; 
 
    if (timeMod == 1)       //1T 
    { 
        //T0x12   T1x12   UM0x6   BRTR    S2SMOD  BRTx12  EXTRAM  S1BRS 
        AUXR |= 0x14;       //串口1使用独立波特率发生器,独立波特率发生器1T 
    } 
    else                    //12T 
    { 
        AUXR = (AUXR | 0x10) & 0xFB; 
    } 
 
    if (doubleBaud == 1) 
    { 
        AUXR |= 0x08;       //波特率加倍 
    } 
    else 
    { 
        AUXR &= 0xF7;       //波特率不加倍 
    } 
 
    EA = 1;  
    //-       -       -       -       -       -       ESPI    ES2 
    IE2 |= 0x01;            //充许串口2中断            

 
 
 
void UART1_SendOneChar(uchar val) 

    //ES = 0;                   //关闭串口1中断 
 
    SBUF = val; 
    while(TI == 0); 
    TI = 0; 
 
    //ES = 1;                  //恢复串口1中断 
}                           
 
 
 
void UART2_SendOneChar(uchar val) 

    //IE2 = 0x00;                 //关闭串口2中断 
 
    S2BUF = val;     
    while ((S2CON & 0x02) == 0); 
    S2CON &= 0xFD; 
 
    //IE2 = 0x01;                //恢复串口2中断 

 
 
 
void UART1_SendStr(uchar *str) 

    while( (*str)!='/0' ) 
    { 
        UART1_SendOneChar(*str); 
        str++; 
    } 

 
 
 
void UART2_SendStr(uchar *str) 

    while( (*str)!='/0' ) 
    { 
        UART2_SendOneChar(*str); 
        str++; 
    } 

 
 
 
void UART1_Int(void) interrupt 4 

    if (RI == 1) 
    { 
        RI = 0; 
        UART1_Recv_Val = SBUF; 
    }    

 
 
 
void UART2_Int(void) interrupt 8 

    if ((S2CON & 0x01) == 1) 
    { 
        S2CON &= 0xFE; 
        UART2_Recv_Val = S2BUF; 
    }    

 
 
 
UART.H
 
 
#ifndef _UART_H_ 
#define _UART_H_ 
 
#define uchar   unsigned char 
#define uint    unsigned int 
 
//定义串口1口开关,关闭则不能接收数据 
#define OpenUART1()     ES=1 
#define CloseUART1()    ES=0 
#define OpenUART2()     IE2|=0x01 
#define CloseUART2()    IE2&=0xFE 
 
//缓存串口1和串口2接收到的字符 
extern uchar UART1_Recv_Val; 
extern uchar UART2_Recv_Val; 
 
 
 
void UART1_Init(uchar RELOAD, bit doubleBaud, bit timeMod); 
 
 
 
void UART2_Init(uchar RELOAD, bit doubleBaud, bit timeMod); 
 
 
 
void UART1_SendOneChar(uchar val); 
 
 
 
void UART2_SendOneChar(uchar val); 
 
 
 
void UART1_SendStr(uchar *str); 
 
 
 
void UART2_SendStr(uchar *str); 
 
 
#endif 
 
 
 
main.c
 
 
#include <STC12C5A.H>     //STC12C5A系列单片机 
#include <intrins.h> 
#include "UART.H" 
 
#define uchar   unsigned char 
#define uint    unsigned int 
 
 
//独立波特率发生器初值,1T模式 
//Fosc = 晶振频率, Baud0 = 标准波特率 
//RELOAD = 256 - INT(Fosc/Baud0/32 + 0.5)        
//Baud = Fosc/(256 - RELOAD)/32 
//error = (Baud - Baud0)/Baud0 * 100% 
uchar RELOAD = 0xD9;                    //Fosc = 12MHz, Baud0 = 9600 
 
//波特率是否加倍,0不倍,1加倍 
bit doubleBaud = 0; 
 
//独立波特率发生器,0为12T模式,1为1T模式 
bit timeMod = 1; 
 
 
 
sbit LED1 = P1^0; 
sbit LED2 = P1^1; 
 
 
 
void main(void) 

    //串口标志位,0使用串口1,1使用串口2 
    bit UART_flag = 1; 
 
    LED1 = 1; 
    LED1 = 1; 
 
    //串口1和串口2初始化 
    UART1_Init(RELOAD, doubleBaud, timeMod); 
    UART2_Init(RELOAD, doubleBaud, timeMod); 
 
    //先用串口1接收字符 
    OpenUART1(); 
    CloseUART2(); 
 
    UART1_SendOneChar(0x0C);            //超级终端清屏 
    UART1_SendStr("/r/n"); 
    UART1_SendStr("/r/n"); 
    UART1_SendStr("1.串口1通讯/r/n"); 
    UART2_SendStr("2.串口2通讯/r/n"); 
 
    while (UART1_Recv_Val == 0); 
 
    UART1_SendStr("/r/n");          //换行 
 
    if (UART1_Recv_Val == '1') 
    { 
        OpenUART1(); 
        CloseUART2(); 
        UART1_SendStr("Light LED(UART1):/r/n"); 
 
        UART_flag = 0; 
    } 
    else 
    { 
        CloseUART1(); 
        OpenUART2(); 
        UART2_SendStr("Light LED(UART2):/r/n"); 
 
        UART_flag = 1; 
    } 
    UART1_Recv_Val = 0;         //缓存清零 
    UART2_Recv_Val = 0;         //缓存清零 
     
    while (1) 
    { 
        if (UART_flag == 0)             //串口1接收字符 
        { 
            if (UART1_Recv_Val != 0) 
            { 
                switch (UART1_Recv_Val) 
                { 
                    case '1': 
                        LED1 = ~LED1; 
                        break; 
                    case '2': 
                        LED2 = ~LED2; 
                        break; 
                    default: 
                        LED1 = 1; 
                        LED1 = 1; 
                        break; 
                } 
                UART1_Recv_Val = 0;            //缓存清零 
            } 
        } 
        else                               //串口2接收字符 
        { 
            if (UART2_Recv_Val != 0) 
            { 
                switch (UART2_Recv_Val) 
                { 
                    case '1': 
                        LED1 = ~LED1; 
                        break; 
                    case '2': 
                        LED2 = ~LED2; 
                        break; 
                    default: 
                        LED1 = 1; 
                        LED1 = 1; 
                        break; 
                } 
                UART2_Recv_Val = 0;            //缓存清零 
            } 
        } 
    } 

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

串行通信需要传输的数据通过调制器(Modulator)将数据转换为模拟信号,经过信号调制(Modulation)后在传输线上传输,接收端通过解调器(Demodulator)将信号解码还原成原始数据。

关键字: 串口 串行通信 并行通信

51 单片机内部有一个全双工串行接口。什么叫全双工串口呢?一般来说,只能接受或只能发送的称为单工串行;既可接收又可发送,但不能同时进行的称为半双工;能同时接收和发送的串行口称为全双工串行口。串行通信是指数据一位一位地按顺...

关键字: 单片机 全双工 串口

串口是“串行接口”的简称,即采用串行通信方式的接口。串行通信将数据字节分成一位一位的形式在一条数据线上逐个传送,其特点是通信线路简单,但传输速度较慢。因此串口广泛应用于嵌入式、工业控制等领域中对数据传输速度要求不高的场合...

关键字: 串口 RS232 同步传输

串口作为单片机开发的一个常用的外设,应用范围非常广。大部分时候,串口需要接收处理的数据长度是不定的。那么怎么才能判断一帧数据是否结束呢,今天就以STM32单片机为例,介绍几种接收不定长数据的方法。

关键字: 单片机 串口 STM32

这是FPGA之旅设计的第十例啦,在上一例中,已经成功驱动了OLED屏幕,本例将结合上一例,以及第四例多bytes串口通信做一个有趣的例程。

关键字: FPGA OLED屏 串口

接下来测试烧写功能,本次采用串口和USB烧写方式。使用ISP串口烧写这是51单片机常用的方案,本次测试比较顺利,没有遇到什么问题。但是USB烧写没有测试成功,USB烧写方式不需要任何的驱动和硬件支持,直接将USB线和ST...

关键字: PCB控制板 USB 串口

摘要:多功能电能表在配电系统中应用广泛,其计量的准确度对企业管理和考核至关重要,因此在设计多功能电能表时需要对其进行校准,满足一定应用等级。常规的多功能电能表校准方法是以电能脉冲校准为主,现提出一种基于C#和功率校表法的...

关键字: 多功能电能表 串口 波特率

摘 要:为了能通过串口采集电能参数,完成一种基于串口的三相电能采集设备的研制,设计了电能采集设备的硬件和软件部分。其中硬件采用MCU+专用电能计量芯片的结构,结构简单;软件则用于实现输入、输出、三相电能参数的采集和串行通...

关键字: 电能采集 ATT7022B MSP430 串口

如何确定时基假如要测量的波特率为9600,则每一比特位的时间为:1/9600≈104μs,一般示波器横向上每个大格子里5个小格子,要想看清一比特位一般需要一个小格子就够了,则时基为:104μs*5=520μs,也就是说时...

关键字: 串口

VivadoML最新版2021下载方法:《安装Vivado2021.1ML版,编译时间真的会减少吗?》今天我们通过zedboard串口使用的实例来简单介绍vivado和vitis的使用步骤。1,首先打开软件,新建一个空白...

关键字: 串口 zedboard vi
关闭
关闭