面就是用C语言编写控制串行口的程序.pdf
- 格式:pdf
- 大小:63.66 KB
- 文档页数:7
C语言实现串行通信接口程序摘要本文说明了异步串行通信-232的工作方式,探讨了查询和中断两种软件接口利弊,并给出两种方式的语言源程序。
的通道之一,以最简单方式组成的串行双工线路只需两条信号线和一条公共地线,因此串行通信既有线路简单的优点同时也有它的缺点,即通信速率无法同并行通信相比,实际上-232在标准条件下的最大通信速率仅为20。
尽管如此,大多数外设都提供了串行口接口,尤其在工业现场-232的应用更为常见。
及兼容机系列都有-232的适配器,操作系统也提供了编程接口,系统接口分为功能调用和功能调用两种21的03和04号功能调用为异步串行通信的接收和发送功能;而14有4组功能调用为串行通信服务,但和功能调用都需握手信号,需数根信号线连接或彼此间互相短接,最为不便的是两者均为查询方式,不提供中断功能,难以实现高效率的通信程序,为此本文采用直接访问串行口硬件端口地址的方式,用语言编写了串行通信查询和中断两种方式的接口程序。
1串行口工作原理微机串行通信采用-232标准,为单向不平衡传输方式,信号电平标准±12,负逻辑,即逻辑1表示为信号电平-12,逻辑0表示为信号电平+12,最大传送距离15米,最大传送速率196波特,其传送序列如图1,平时线路保持为1,传送数据开始时,先送起始位0,然后传8或7,6,5个数据位0,1,接着可传1位奇偶校验位,最后为1~2个停止位1,由此可见,传送一个字符7位,加上同步信号最少需9位数据位。
@@812300;图1@@串行通信的工作相当复杂,一般采用专用芯片来协调处理串行数据的发送接收,称为通用异步发送接收器,以节省的时间,提高程序运行效率,系列采用8250来处理串行通信。
在数据区中的头8个字节为4个的端口首地址,但只支持2个串行口1基地址00400000和2基地址00400002。
8250共有10个可编程的单字节寄存器,占用7个端口地址,复用地址通过读写操作和线路控制寄存器的第7位来区分。
C语言实现串行通信接口程序串行通信是一种在计算机系统中进行数据传输的常见方式。
C语言提供了许多用于串行通信接口的库函数,如<termios.h>和<fcntl.h>等。
在本文中,我将介绍如何使用这些库函数来实现一个简单的串行通信接口程序。
首先,我们需要打开串行通信端口并进行初始化设置。
在<fcntl.h>中,有一个函数叫做open(,我们可以用它来打开串行端口。
例如,如果我们想打开/dev/ttyS0端口,可以使用以下代码:```cint fd = open("/dev/ttyS0", O_RDWR);if (fd == -1)perror("Error opening serial port");exit(1);```这里的O_RDWR表示我们将以读写模式打开串行端口。
如果打开串行端口失败,open(函数会返回-1,并通过perror(函数打印错误信息到标准错误输出。
在这种情况下,我们使用exit(1)来结束程序。
接下来,我们需要配置串行端口的通信参数。
在<termios.h>中,有一个结构体叫做termios,它包含了一些用于配置串行端口的参数。
我们可以使用tcgetattr(和tcsetattr(函数来获取和设置串行端口的参数。
```cstruct termios options;tcgetattr(fd, &options);//设置串行端口的通信参数cfsetispeed(&options, B9600); // 设置输入波特率为9600cfsetospeed(&options, B9600); // 设置输出波特率为9600options.c_cflag ,= CS8; // 设置数据位为8位options.c_cflag &= ~PARENB; // 禁用奇偶校验tcsetattr(fd, TCSANOW, &options);```在这个示例中,我们将波特率设置为9600,数据位设置为8位,并禁用了奇偶校验。
8. 结构实例6:单片机串口通信虽然那个流水灯游戏的可玩性和按键手感问题还值得再好好提升一下,但小月更希望调剂一下,转而开始了对手头烧写板上关于RS-232转换部分的学习。
小月的做法并不难以理解,毕竟与RS-232转换的相关电路在原理图中还是相当显眼的,甚至于他手头编程器的别名就是RS-232转换器。
图8.1 单片机中负责RS-232通讯的电路在烧写器一端与电脑连接的两个接头中,9针的RS-232接口就是串口通信线,而另一个USB口仅接通了+5V和GND,只有给烧写器供电的作用。
这样就可以知道,电脑可以通过RS-232对单片机的内部程序进行改写。
那么,这就意味着单片机与电脑间必然可以进行数据的交换,这种交换,就叫做通信。
所谓串口通信,就是指这种基于RS-232串口的通信方式。
RS-232(ANSI/EIA-232标准)是IBM-PC及其兼容机上的串行连接标准。
最早是为使电脑通过电话线系统相互通信的调制解调器上而是设计的。
后来发展到连接鼠标或打印机上,目前已经被支持设备的即插即用和热插拔功能的USB所替代,但仍广泛的用于工业仪器仪表中,同时也是单片机最基础和最常见的通信方式。
不过要把“最基础和最常见”这两个最拆开来说,就要在后面加上“之一”了。
虽然目前的通信技术日新月异,但这种说法在今后很长一段时期内都是成立的,也正因为这样的特点,STC的51系列单片机都是默认通过RS-232方式进行烧写的。
作为两台设备之间进行的通信,必然需要共同遵守某种规定或规则,包括交流什么、怎样交流及何时交流。
这个规则就是通信协议。
RS-232通信中通信协议的原则就是串口按位(bit)发送和接收数据。
线路上,RS-232通信使用3根线完成,分别是地线、发送、接收。
端口能够在一根线上发送数据的同时在另一根线上接收数据,即全双工传输。
全双工传输是传输制式的一种分类方式中的一类,除此还有单工传输和半双工传输。
单工传输,是指消息只能单方向传输的工作方式。
串行口1编程的要点和步骤一、串行口1编程的基本概念串行口1是指计算机中的一个通信接口,它可以用来连接外设如打印机、调制解调器等,并通过串行通信协议进行数据传输。
在编程中,我们可以使用串行口1来实现计算机与外设之间的数据传输。
二、串行口1编程的步骤1. 打开串行口1在开始进行串行口1编程之前,我们需要先打开串行口1。
这可以通过访问计算机的端口地址来实现。
具体方法为:向端口地址0x3F8写入数值0x80,然后再向端口地址0x3FB写入数值0x03。
2. 配置波特率和数据格式在打开串行口1之后,我们需要对其进行配置,以确保数据能够正确地传输。
首先需要设置波特率,即数据传输速率。
这可以通过向端口地址0x3F8写入相应的数值来实现。
如果要设置波特率为9600,则需要向该地址写入数值0x01。
还需要设置数据格式,包括数据位、停止位和校验位等参数。
这些参数也可以通过向相应的端口地址写入数值来实现。
3. 发送数据在完成配置之后,就可以开始发送数据了。
这可以通过向端口地址0x3F8写入要发送的数据来实现。
需要注意的是,数据必须按照指定的数据格式进行发送,否则将无法正确接收。
4. 接收数据除了发送数据之外,我们还需要能够接收来自外设的数据。
这可以通过访问端口地址0x3FD来实现。
如果有数据传输过来,就可以从该端口地址读取相应的数值。
5. 关闭串行口1在完成所有操作之后,我们需要关闭串行口1以释放资源。
这可以通过向端口地址0x3F8写入数值0x00来实现。
三、串行口1编程的要点1. 端口地址:在进行串行口1编程时,需要访问计算机中特定的端口地址。
这些地址通常是预定义好的,具体取决于计算机硬件和操作系统等因素。
2. 波特率和数据格式:在配置串行口1时,需要设置波特率和数据格式等参数。
这些参数必须与外设相匹配才能确保数据正确传输。
3. 数据发送和接收:在进行串行通信时,需要注意正确地发送和接收数据。
如果格式不正确或者未按照协议进行操作,则可能会导致通信失败或者出现意想不到的错误。
面就是用C语言编写控制串行口的程序。
#include?"dos.h"#include?"stdlib.h"#include?"stdio.h"#define?PORT?0{if(argc<2){exit(1);}InitPort(PORT,231);if(*argv[1]==''''S'''')?/*检查选择的有效性*/ SendFile(argv[2]);else?if(*argv[1]==''''R'''')ReceiveFile();else{printf("Error?parament.Please?input?again."); exit(1);}}void?SendFile(char?*fname){FILE?*fp;int?ch,s;if((fp=fopen(fname,"rb"))==NULL) {exit(1);}do{if(ferror(fp)){break;}s=ch%16;?/*Send(s);s=ch/16;?/*Send(s);s=46;?/*发送文件结束信息*/ Send(s);Send(s);fclose(fp);}void?Send(s)int?s;{SendPort(PORT,s);G=ReadPort(PORT);?/*等待握手信号*/ if(s!=G)s=s+16;do{SendPort(PORT,s);G=ReadPort(PORT);/*等待握手信号*/ }while(s!=G);}char?*fname;{int?s,ch;s=ch%16;?/*Send(s);s=ch/16;Send(s);?/*}s=32;?/*Send(s);Send(s);}void?ReceiveFile(){FILE?*fp;char?ch;int?G1,G2,G3;char?fname[15];GetFileName(fname);remove(fname);if((fp=fopen(fname,"wb"))==NULL){printf("Can''''t?open?output?file.\n");exit(1);}/*循环为检测每次接受的数据是否为新数据,如果不是,*/ /*则用此次接收的数据覆盖上次接收的数据*/do{?putc(ch,fp);if(ferror(fp)){exit(1);}G1=G3;fclose(fp);}int?Receive(port,G)int?port,*G;{int?GM;SendPort(port,*G);GM=ReadPort(port);if(GM/16==0)else?if(GM/16==1){do{*G=GM;SendPort(port,GM);GM=ReadPort(port);}while(GM/16==1);}return?GM;}{char?ch;do{*f=ch;*f++;G1=G3;printf("File?name?transmit?finished.\n"); }void?InitPort(port,para)int?port;unsigned?char?para;{union?REGS?reg;int86(0x14,?,?);}void?SendPort(port,c)int?port;char?c;{union?REGS?reg;exit(1);}}int?port;{if(kbhit()){/*printf("Press?any?key?to?exit."); getch();exit(1);}}int86(0x14,?,?);printf("\nRead?mistake!"); exit(1);}}int?CheckState(port) int?port;{}。