2019-05-09 单片机IO口模拟UART串口通信

#include

sbit PIN_RXD = P3^0; //接收引脚定义

sbit PIN_TXD = P3^1; //发送引脚定义

bit RxdOrTxd = 0; //指示当前状态为接收还是发送 ,异步,那么在接收,要么在发送

bit RxdEnd = 0; //接收结束标志

bit TxdEnd = 0; //发送结束标志

unsigned char RxdBuf = 0; //接收缓冲器

unsigned char TxdBuf = 0; //发送缓冲器

void ConfigUART(unsigned int baud);

void StartTXD(unsigned char dat);

void StartRXD();

extern void InitLcd1602();

extern void LcdWriteDat(unsigned char dat);

extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str, unsigned char len);

void main(){

  InitLcd1602();

  LcdWriteDat('A');

  LcdWriteDat('B');

    EA = 1; //开总中断

    ConfigUART(9600);   //配置波特率为 9600

    while (1){ //程序一直处于等待信号状态,有数据则接受数据,接受了一个数据后发送出去

        while (PIN_RXD); //等待接收引脚出现低电平,即起始位

        StartRXD(); //启动接收

        while (!RxdEnd); //等待接收完成

//LcdShowStr(0,1,&("data"),10);

LcdWriteDat(RxdBuf);

        StartTXD(RxdBuf); //接收到的数据+1 后,发送回去

        while (!TxdEnd); //等待发送完成

}

}

/* 串口配置函数,baud-通信波特率 */

void ConfigUART(unsigned int baud){

    TMOD &= 0xF0; //清零 T0 的控制位  按位运算: &1的结果不变 &=1就是不改此位,&0的结果是0 &=0就是bit置零 write。|=1的结果是1可以用来把bit改为1 write,|=0结果也不变,

    TMOD |= 0x02; //配置 T0 为模式 2 模式2是自动重装,就是低8位TL0溢出后,自动把高8位TH0保存的值写入TL0,然后又开始计数。

    TH0 = 256 - (11059200/12)/baud; //计算 T0 重载值 根据921600/256=3600,所有这种计数方式波特率最低只能是3600,要想低可能需要改机器周期12这个数值了

}

/* 启动串行接收 */

void StartRXD(){

    TL0 =  256 - ((256-TH0)>>3); //接收启动时的 T0 定时为半个波特率周期

    ET0 = 1; //使能 T0 中断

    TR0 = 1; //启动 T0

    RxdEnd = 0; //清零接收结束标志

    RxdOrTxd = 0; //设置当前状态为接收

}

/* 启动串行发送,dat-待发送字节数据 */

void StartTXD(unsigned char dat){

    TxdBuf = dat; //待发送数据保存到发送缓冲器

    TL0 = TH0; //T0 计数初值为重载值

    ET0 = 1; //使能 T0 中断

    TR0 = 1; //启动 T0

    PIN_TXD = 0; //发送起始位

    TxdEnd = 0; //清零发送结束标志

    RxdOrTxd = 1; //设置当前状态为发送

}

/* T0 中断服务函数,处理串行发送和接收 */

void InterruptTimer0() interrupt 1{

    static unsigned char cnt = 0; //位接收或发送计数

    if (RxdOrTxd){ //串行发送处理

        cnt++;

if (cnt <= 8){ //低位在先依次发送 8bit 数据位

          PIN_TXD = TxdBuf & 0x01; //得到末位的值

          TxdBuf >>= 1;


        }else if (cnt == 9){ //发送停止位

            PIN_TXD = 1;

        }else{ //发送结束

            cnt = 0; //复位 bit 计数器

            TR0 = 0; //关闭 T0

            TxdEnd = 1; //置发送结束标志  

        }

    }else{ //串行接收处理

        if (cnt == 0){ //处理起始位

            if (!PIN_RXD){ //起始位为 0 时,清零接收缓冲器,准备接收数据位

                RxdBuf = 0;

                cnt++;

            }else{ //起始位不为 0 时,中止接收

            TR0 = 0; //关闭 T0

            }

          }else if (cnt <= 8){ //处理 8 位数据位

          // RxdBuf >>= 1; //低位在先,所以将之前接收的位向zuo移

            //接收脚为 1 时,缓冲器最高位置 1,

            //而为 0 时不处理即仍保持移位后的 0

            if (PIN_RXD){

          RxdBuf |=(0x01<<(cnt-1)); //如果数据为1,加到RxdBuf高位上  0b10000000

            }

            cnt++;

        }else if(cnt==9){

cnt++;

}else{ //停止位处理

            cnt = 0; //复位 bit 计数器

            TR0 = 0; //关闭 T0

            if (PIN_RXD){ //停止位为 1 时,方能认为数据有效

                RxdEnd = 1; //置接收结束标志

}

}

    }

}


发0x30收到B0,发0x33到B3,不知为何,时间有限不搞了。。。

你可能感兴趣的:(2019-05-09 单片机IO口模拟UART串口通信)