[serial]基于select/poll/epoll的串口操作

转自:http://www.cnblogs.com/darryo/p/selectpollepoll-on-serial-port.html

In this article, I will use three asynchronous conferencing--select, poll and epoll on serial port to transmit data between PC and Raspberry pi.


Outline

  1. Character device file of serial port
  2. Naive serial communication
  3. Asynchronous conferencing
  4. Select
  5. Poll
  6. Epoll

Character device of serial port

My device is Raspberry pi with debian system and PC with ubuntu12.04 system.

And I have used a USB-TTL to link the these device.

The character device files on the two device is :

/dev/ttyUSB0 #Ubuntu

/dev/ttyAMA0 #Debian Raspberry pi

These two files are what we must use to achieve the lab.

But there is a little trap of /dev/ttyAMA0.

By default, Raspberry pi uses /dev/ttyAMA0 as a output of serial. Therefor we could use minicom or putty to control our device. However, we have to modify the default function of serial, since we will use our own method to use serial port.

So we should modify two files:

/boot/comdline.txt

dwc_otg.lpm_enable=0 console=ttyAMA0,115200 rootfstype=ext4 elevator=deadline rootwait console=tty1 root=/dev/mmcblk0p2

Delete console=ttyAMA0,115200

/etc/inittab

T0:23:respawn:/sbin/getty -L ttyAMA0 115200 vt100

Comment the line above.


Now, start to code:

Naive version

The naive serial port communication version

Open the device, set the baud rate, and set parity

#include     <stdio.h>      /*标准输入输出定义*/
#include     <string.h>
#include     <stdlib.h>     /*标准函数库定义*/
#include     <unistd.h>     /*Unix标准函数定义*/
#include     <sys/types.h>  /**/
#include     <sys/stat.h>   /**/
#include     <fcntl.h>      /*文件控制定义*/
#include     <termios.h>    /*PPSIX终端控制定义*/
#include     <errno.h>      /*错误号定义*/
#define FALSE 0
#define TRUE 1

void set_speed(int fd) {
    struct  termios Opt;
    tcgetattr(fd, &Opt);
    cfsetispeed(&Opt,B115200);
    cfsetospeed(&Opt,B115200);
    tcsetattr(fd,TCSANOW,&Opt);
    return;
}

void set_Parity(int fd) {
    struct termios options;
    tcgetattr(fd, &options);
    options.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);  /*Input*/
    options.c_oflag  &= ~OPOST;   /*Output*/
    tcsetattr(fd,TCSANOW,&options);
    return;
}

int OpenSerial(char *Dev) {
    int fd = open( Dev, O_RDWR );         //| O_NOCTTY | O_NDELAY
    if (-1 == fd) { /*设置数据位数*/
        perror("Can't Open Serial Port");
        return -1;
    }
    else {
        set_speed(fd);
        set_Parity(fd);
        return fd;
    }

}

int main(){
    int fd; 
    ssize_t length;
    char buff[512];
    char *dev ="/dev/ttyAMA0";
    fd = OpenSerial(dev);
    for(;;){
        length = read(fd,buff,sizeof(buff));
        if(length > 0) {
            buff[length] = 0;
            printf("plain:%s\n",buff);
        }
    }
    close(fd);
    exit(0);
}

Select version

#include <sys/time.h>
#include <sys/types.h>
#include "serial/serial.h"

int main() {
    int fd;
    fd_set rfds;
    struct timeval tv;
    char buff[512];
    ssize_t length;
    fd = OpenSerial("/dev/ttyAMA0");
   
    for(;;) {
        FD_ZERO(&rfds);
        FD_SET(fd, &rfds);

        //timeout = 5s
        tv.tv_sec = 5;
        tv.tv_usec = 0;
        //Wait for 5 seconds, then go
        int n;
        n = select(fd + 1, &rfds, NULL, NULL, &tv);
        //choose the target from set
        if(n > 0) {
            if (FD_ISSET(fd, &rfds)) {
                length = read(fd, &buff, sizeof(buff));
                buff[length] = 0;
                printf("select:%s\n", buff);
            }
        } else {
            printf("No data within 5 seconds.\n");
        }
    }
    return 0;
}

 

Poll version

#include <sys/poll.h>
#include "serial/serial.h"
int main(void) {
    struct pollfd fds[1];
    ssize_t length;
    char buff[512];
    fds[0].fd = OpenSerial("/dev/ttyAMA0");
    fds[0].events = POLLIN ;
    for(;;) {
        int n;
        n = poll( fds, 1, 5000);
        //got data, and look up which fd has data, but we just have 1
        if(n > 0) {
        //if( fds[0].revents & POLLIN ) {
            length = read(fds[0].fd, buff, sizeof(buff) );
            buff[length] = 0;
            printf("poll:%s\n",buff);

        } else {
            printf("No data within 5 seconds.\n");
        }
    }
} 

 

Epoll version

#include <sys/epoll.h>
#include "serial/serial.h"

#define MAXEVENTS 64

int main(void){
    int fd;
    int efd;
    struct epoll_event event;
    struct epoll_event *events;
    int length;
    char buff[512];
    fd = OpenSerial("/dev/ttyAMA0");
    efd = epoll_create1 (0);//initial is 0

    event.data.fd = fd;
    event.events = EPOLLIN | EPOLLET;

    epoll_ctl (efd, EPOLL_CTL_ADD, fd, &event);
    /* Buffer where events are returned */
    events = calloc (MAXEVENTS, sizeof event);

    /* The event loop */
    for(;;) {
        int n;
        n = epoll_wait (efd, events, MAXEVENTS, 5000);
        if(n > 0) {
            length = read(events[0].data.fd, buff, sizeof(buff));
        
            if(length > 0) {
                buff[length] = 0;
                printf("epoll:%s\n", buff);
            }
        } else {
            printf("No data whthin 5 seconds.\n");
        }
    }
    free (events);
    close (fd);
    return 0;
}

 

转载于:https://www.cnblogs.com/aaronLinux/p/9055016.html

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值