
AJMP MAIN上电,转向主程序
ORG 0023H串行口的中断入口地址
AJMP SERVE 转向中断服务程序
ORG 0040H主程序
MAIN: MOV SP,#60H 设置堆栈指针
MOV SCON ,#50H
MOV PCON ,#00H
MOV TMOD,#20H
MOV TH1,#0F3H
MOV TL1,#0F3H
SETB TR1
MOV R0 ,#20H 置发送数据区首地址
MOV R1 ,#40H ;置接收数据区首地址
MOV R7 ,#10H ;置发送字节长度
MOV R6 ,#10H ;置接收字节长度
SETB ES允许串行口中断
SETB EACPU允许中断
MOV A ,@R0 ;取第一个数据发送
MOV SBUF ,A发送第一个数据
SJMP $ ;等待中断
SERVE: JNB RI ,SEND TI=1,为发送中断
CLR RI
MOV A ,SBUF 读出接收缓冲区内容
MOV @R1 ,A读入接收缓冲区
DJNZ R6 ,L1 判断数据块发送完否
SJMP L2数据块接收完,转L2
L1:INC R1修改数据区指针
L2:RETI中断返回
SEND:
CLR TI清除发送中断标志
DJNZ R7 ,L3 判断数据块发送完否
SJMP L4数据块接收完,转L4
L3: MOV A ,@R0取数据发送
MOV SBUF ,A发送数据
INC R0修改数据地址
L4:
RETI中断返回
END
要求用什么语言?------------------------------
网友“星心晨梦”的回答,是可以满足题目要求的。
我实验了,是成功的,没有任何错误。
楼主应该检查自己的电路、串口设置等等。
------------------------------
网友“星心晨梦”的回答,篇幅稍稍长了一些。
缩减一半就可以正常工作了,可读性更好一些。
建议看看下面的。
------------------------------
#include <AT89X51.h>
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int
//---------------------------------------------------
uchar idata trdata1[] = {
'W','E','L','C','O','M','E',' ','T','O',' ','C','H','I','N','A','!',0x0d,0x0a,0x00}
uchar idata trdata2[] = {
'M','Y',' ','N','A','M','E',' ','I','S',' ','L','I','M','I','N','G',0x0d,0x0a,0x00}
uchar idata trdata3[] = {
'Y','I','N','G','Y','O','N','G','K','E','X','U','E','X','U','E','Y','U','A','N',0x0d,0x0a,0x00}
uchar RxBuf[5], Rx_p, Rx_i, TX_p, Tx_i
//---------------------------------------------------
void UART_Init(void) //串口初始化
{
PCON = 0x00
SCON = 0x50//串口工作方式为1,允许接收.
TMOD = 0x20// T1 定时方式2
TH1 = 0xfd //波特率 9600bps @ fosc = 11.0592MHz
TL1 = 0xfd
TR1 = 1//启动T1
ES = 1 //开串口中断.
EA = 1 //开总中断.
}
//---------------------------------------------------
void main()
{
UART_Init()
Rx_p = 0
Rx_p = 2
while(1) {
if (Rx_p != 0) {
TX_p = Rx_p//字符串1 2 3
Tx_i = 0 //字符指针.
Rx_p = 0 //清零.
Rx_i = 0
TI = 1//启动发送中断.
} }
}
//---------------------------------------------------
void Uart_INT(void) interrupt 4 //串口中断函数
{
uchar Tcv = 0
if(RI) { //接收?.
RI = 0 //标志位清零.
RxBuf[Rx_i] = SBUF
if((RxBuf[Rx_i - 1] == 'g') &&(RxBuf[Rx_i] == 'o')) Rx_p = 1
if((RxBuf[Rx_i - 2] == 'w') &&(RxBuf[Rx_i - 1] == 'h') &&(RxBuf[Rx_i] == 'o')) Rx_p = 2
if((RxBuf[Rx_i - 3] == 'h') &&(RxBuf[Rx_i - 2] == 'o') &&(RxBuf[Rx_i - 1] == 'm') &&(RxBuf[Rx_i] == 'e')) Rx_p = 3
P1 = Rx_p //在P1显示收到的信息.
Rx_i++
Rx_i %= 5
}
else {
TI = 0
if (TX_p == 1) Tcv = trdata1[Tx_i] //取来待发字符.
if (TX_p == 2) Tcv = trdata2[Tx_i]
if (TX_p == 3) Tcv = trdata3[Tx_i]
if (Tcv != 0) {SBUF = Tcv Tx_i++} //不是0就发送.
}
}
//---------------------------------------------------
猴!今儿扯串口,相对于并行——一口气全把数据扔过去,串行显得更加稳重——一位一位来。
串行就是这样,只需要一条数据线(全双工和同步串行时两条),一位一位的传过去。为了让大家在直到你是在给我传数据而不是外面的噪音或者是胡说八道,所以串行数据的各位要组装帧(看正文中的 帧格式 )。乍一看,这种方式跟并行比肯定慢的一腿。但实际上,多亏了它的稳定性,可以在波特率极高的情况下依然保持稳定,这是并行所办不到的(传的快了或距离远了就张牙舞爪了),所以发展到现在,串口已经把并口甩走几条街啦。
并口传输的例子: 《51单片机实战:液晶显示器のLCD1602》
除此之外,串行传输分同步和异步。同步除了传输数据外,还要传输时钟信号,以保持双方同步。另一种,异步,就没这么麻烦了,也是本例中要讲到的,各自走各自的时钟就好,只要帧格式和波特率都商量好是一样的就好。
电平之前在文章 《51单片机实战:液晶显示器のLCD1602》 中介绍过,那里只说了TTL,本例中由于要和计算机打交道,所以多了一种电平:RS-232C
在单片机中是TTL,电脑那边传出和接收都是RS232,所以两种电平需要作转换。
当当当!它就是干这活的。
举个栗子,比如单片机从T1IN输入TTL电平,转换好的RS232电平就从R1OUT输出。其他的照猫画虎,这里不详细说这个东西,因为咱们在Proteus里干活,用不着转换(Proteus光环)。
在此描述串行传输数据速率。
正儿八经的说,波特率乃 码元 的传输速率,即每秒传输的码元个数(码元可以是任意进制的),并不是什么每秒传输的比特数,大家注意。
波特来源于一个人的名字: Jean-Maurice-Émile Baud ot ,因此简写为Baud,单位符号:Bd。波特率可简写成Bd/s。
在串口通信中,其码元就是二进制信号,所以波特率的数值等于比特率数值,但你不能说波特率就是比特率啊!
单片机的串口通信有四种方式(各方式具体是干什么的,别着急,在后面),其中方式0和方式2的波特率是固定的。方式1和方式3的波特率是可变的,其脉冲周期由定时器1溢出产生。
其中 f 是系统晶振频率,T1是计时器1, SMOD 是PCON中的最高位(PCON见相关寄存器的第一个)。
可以从上述公式看出,波特率不可变是因为直接与系统晶振频率相关(晶振频率不可变,除非换晶振),而可变是因为直接与T1的溢出率相关(溢出率可以改变)。
溢出率
在之前定时器应用的例子( 《51单片机实战:定时器与数码管的应用》 )中,我们计算的是溢出周期,也就是多长时间会溢出一次。这次我们用到的溢出率其实是同一个东西,取倒数就可以了。
详见: 《51单片机实战:定时器与数码管的应用》 - 知识点 - 定时器/计数器 - 初值
11.0592MHz
为什么要用这么蹩脚的数字作晶振频率哈,就是跟这里有关。如果你已经用上述公式计算过串口方式1下的12MHz和11.0592MHz在9600波特率下的定时器初值,你就会发现,前者得出一个小数,而后者是个整数。
我们可没办法用小数赋初值,所以你若用近似的整数作初值,就意味着会产生误差。
若用其他的晶振和波特率的话,请自行按前面的公式计算。
串行传输按比特来,一个个比特组成一个帧,帧需要一定的格式才能被双方识别这是一个帧信息。
电源管理 寄存器,用于管理单片机的电源部分。
字节地址: 87H ,不能位寻址, reg52.h 中已定义,单片机复位时全部清零。
上表中出现的“串口方式”见下表的SM0和SM1。
串口控制 寄存器,用于设定串口工作方式。
字节地址: 98H ,可位寻址, reg52.h 中已定义,单片机复位时全部清零。
上表中波特率可变的方式,都由定时器1的溢出率控制。
当单片机接收到字符 a 时,点亮一个LED灯。传送方式:9600波特率,8数据位,无校验位,1停止位。
本例中我就不写电脑端程序了,直接用现成的。
注意,这里面我没有放转换电平转换芯片(MAX232),只有在Proteus里可以这么干,现实中焊板子还是要做电平转换的,这里这个软件给简化了。
COMPIM
虚拟终端
右下角那个东西是虚拟终端(Virtual Terminal),他可以直接截获串口传来的消息然后显示出来。很方便做这方面调试时使用。
路径: 边栏 → instruments → virtual terminal
如果在调试的时候不小心把它的终端窗口关了,再次打开路径: 菜单 → debug - virtual terminal ,注意是在启动调试的情况下。
大年初二,拜访完姥姥家就该看看单片机怎么玩,你说是吧!这两天快马加鞭了,下一站:一周目大BOSS。各位加油。
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)