免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
123下一页
最近访问板块 发新帖
查看: 6161 | 回复: 22
打印 上一主题 下一主题

望高人帮我看下这代码(一个读串口程序) [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-08-14 14:47 |只看该作者 |倒序浏览
#include     <stdio.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   TRUE 1
#define   FALSE 0


/***@brief  设置串口通信速率
* *@param  fd     类型 int  打开串口的文件句柄
* *@param  speed  类型 int  串口速度
* *@return  void*/

int speed_arr[] = { B38400, B19200, B9600, B4800, B2400, B1200, B300,};
        
int name_arr[] = {38400,  19200,  9600,  4800,  2400,  1200,  300,};
        
void set_speed(int fd, int speed)
{
        int   i;
        int   status;
        struct termios   Opt;
        tcgetattr(fd, &Opt);
        for ( i= 0;  i < sizeof(speed_arr) / sizeof(int);  i++)
        {
                if  (speed == name_arr
)
                {
                        tcflush(fd, TCIOFLUSH);
                        cfsetispeed(&Opt, speed_arr
);
                        cfsetospeed(&Opt, speed_arr
);
                        status = tcsetattr(fd, TCSANOW, &Opt);
                        if  (status != 0)
                                perror("tcsetattr fd1");
                        return;
                }
                tcflush(fd,TCIOFLUSH);
        }
}
/**
* *@brief   设置串口数据位,停止位和效验位
* *@param  fd     类型  int  打开的串口文件句柄*
* *@param  databits 类型  int 数据位   取值 为 7 或者8*
* *@param  stopbits 类型  int 停止位   取值为 1 或者2*
* *@param  parity  类型  int  效验类型 取值为N,E,O,,S
* */
int set_Parity(int fd,int databits,int stopbits,int parity)
{
        struct termios options;
        if  ( tcgetattr( fd,&options)  !=  0)
        {
                perror("SetupSerial 1");
                return(FALSE);
        }
        options.c_cflag &= ~CSIZE;
        switch (databits) /*设置数据位数*/
        {
                case 7:
                        options.c_cflag |= CS7;
                        break;
                case 8:
                        options.c_cflag |= CS8;
                        break;
                default:
                        fprintf(stderr,"Unsupported data size\n");
                        return (FALSE);
        }
        switch (parity)
        {
                case 'n':
                case 'N':
                        options.c_cflag &= ~PARENB;   /* Clear parity enable */
                        options.c_iflag &= ~INPCK;     /* Enable parity checking */
                        break;
                case 'o':
                case 'O':
                        options.c_cflag |= (PARODD | PARENB);  /* 设置为奇效验*/
                        options.c_iflag |= INPCK;             /* Disnable parity checking */
                        break;
                case 'e':
                case 'E':
                        options.c_cflag |= PARENB;     /* Enable parity */
                        options.c_cflag &= ~PARODD;   /* 转换为偶效验*/  
                        options.c_iflag |= INPCK;       /* Disnable parity checking */
                        break;
                case 'S':
                case 's':  /*as no parity*/
                        options.c_cflag &= ~PARENB;
                        options.c_cflag &= ~CSTOPB;
                        break;
                default:
                        fprintf(stderr,"Unsupported parity\n");
                        return (FALSE);
        }
        /* 设置停止位*/   
        switch (stopbits)
        {
                case 1:
                        options.c_cflag &= ~CSTOPB;
                        break;
                case 2:
                        options.c_cflag |= CSTOPB;
                        break;
                default:
                        fprintf(stderr,"Unsupported stop bits\n");
                        return (FALSE);
        }
        /* Set input parity option */
        if (parity != 'n')
                options.c_iflag |= INPCK;
        options.c_cc[VTIME] = 150; // 15 seconds
        options.c_cc[VMIN] = 0;

        tcflush(fd,TCIFLUSH); /* Update the options and do it NOW */
        if (tcsetattr(fd,TCSANOW,&options) != 0)
        {
                perror("SetupSerial 3");
                return (FALSE);
        }
        return (TRUE);
}
/**
* *@breif 打开串口
* */
int OpenDev(char *Dev)
{
        int fd = open( Dev, O_RDWR|O_NOCTTY );         //| O_NOCTTY | O_NDELAY
                printf("open port \n");
        if (-1 == fd)
        { /*设置数据位数*/
                perror("Can't Open Serial Port");
                return -1;
        }
        else
                return fd;

}
/**
* *@breif         main()
* */
int main(int argc, char **argv)
{
        int fd;
        int nread;
        char buff[512];
        char *dev ="/dev/ttyAM1";  //arm开发板的串口2
        fd = OpenDev(dev);
        if (fd>0)
                set_speed(fd,9600);
        else
        {
                printf("Can't Open Serial Port!\n");
                exit(0);
        }
        if (set_Parity(fd,8,1,'N')== FALSE)
        {
                printf("Set Parity Error\n");
                exit(1);
        }
        while(1)
        {
               
                while((nread = read(fd,buff,512))>0)
                {
                        
                        buff[nread+1]='\0';
                        if( write(STDOUT_FILENO,buff,nread) !=nread )
                                perror("write error");

                }
        }
        //close(fd);
        //exit(0);
}

经过我反复测试,我发现,linux系统和开发板通讯是没有问题的,发一个立马显示到屏幕上(开发板屏幕)。但windows好像不会立即显示到屏幕(开发板屏幕)上,而是保存到缓冲区了(造成没有接收到的假象),因为当我再次用linux系统发的时候,屏幕就显示出刚才从windows上发的数据来了。
不过我在windows发送数据的时候加个回车符就能实时显示出来了!!
初步怀疑是上面红色代码有问题!
望高人怎么解决啊!


[ 本帖最后由 wzj071227 于 2008-8-14 16:41 编辑 ]

论坛徽章:
0
2 [报告]
发表于 2008-08-14 15:12 |只看该作者

回复 #1 wzj071227 的帖子

先自己跟踪调试一下,先找到问题出在哪里?

论坛徽章:
0
3 [报告]
发表于 2008-08-14 15:22 |只看该作者
代码我看不出有什么错误的
你串口驱动有没修改过呢?
如果修改过了,用printf打印消息来跟踪一下uart_read

论坛徽章:
0
4 [报告]
发表于 2008-08-14 15:49 |只看该作者
我晕啊,我在PC机上测试什么问题都没有啊!都能读取,都读的好好的啊!arm板没有没法跟踪啊 ,GDB不能用啊!!

串口驱动我没有改过啊!驱动是没有问题的啦,发送能往外发,驱动应该是没有问题的吧!!

[ 本帖最后由 wzj071227 于 2008-8-14 15:51 编辑 ]

论坛徽章:
0
5 [报告]
发表于 2008-08-14 15:57 |只看该作者
你用啥编的?!!???

论坛徽章:
0
6 [报告]
发表于 2008-08-14 15:58 |只看该作者
我指的是编译!!!!!!!

论坛徽章:
0
7 [报告]
发表于 2008-08-14 16:09 |只看该作者
arm-linux-gcc-3.4.3编译的,经过我反复测试,我发现,linux系统和开发板通讯是没有问题的,发一个立马显示到屏幕上(开发板屏幕)。但windows好像不会立即显示到屏幕(开发板屏幕)上,而是保存到缓冲区了(造成没有接收到的假象),因为当我再次用linux系统发的时候,屏幕就显示出刚才从windows上发的数据来了。问题是找到了,不知道怎么解决。而且也不知道缓存区是那的!

[ 本帖最后由 wzj071227 于 2008-8-14 16:13 编辑 ]

论坛徽章:
0
8 [报告]
发表于 2008-08-14 16:10 |只看该作者
原帖由 wzj071227 于 2008-8-14 15:49 发表
我晕啊,我在PC机上测试什么问题都没有啊!都能读取,都读的好好的啊!arm板没有没法跟踪啊 ,GDB不能用啊!!

串口驱动我没有改过啊!驱动是没有问题的啦,发送能往外发,驱动应该是没有问题的吧!!

GDB+GDBServer可以交叉调试

论坛徽章:
0
9 [报告]
发表于 2008-08-14 16:14 |只看该作者
原帖由 wzj071227 于 2008-8-14 16:09 发表
arm-linux-gcc-3.4.3编译的,经过我反复测试,我发现,linux系统和开发板通讯是没有问题的,发一个立马显示到屏幕上(开发板屏幕)。但windows好像不会立即显示到屏幕(开发板屏幕)上,而是保存到缓冲区了(造 ...

试下open用O_NOBLOCK?

论坛徽章:
0
10 [报告]
发表于 2008-08-14 16:32 |只看该作者
不能手动刷缓冲区??????
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP