免费注册 查看新帖 |

Chinaunix

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

请教串口奇偶校验位(mark和space)的设置 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-10-28 09:33 |只看该作者 |倒序浏览
如何设置串口是(8data,1stop,mark)和SPACE(8data,1stop,space),多谢!

论坛徽章:
1
黑曼巴
日期:2020-02-27 22:54:26
2 [报告]
发表于 2010-10-28 09:44 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽

论坛徽章:
0
3 [报告]
发表于 2010-10-28 11:14 |只看该作者
本帖最后由 herocsz 于 2010-10-28 11:16 编辑

回复 2# c/unix
    这里哪里是设置MARK的?
c/unix 发表于 2010-10-28 09:44 [img]http://bbs.chinaunix.net/images/common/back.gif
int set_parity(int fd, int databits, int stopbits, int parity)
{
    struct termios options;

    if (tcgetattr(fd, &options) !=  0)
    {
        perror("SetupSerial 1");
        return 1;
    }

    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 1;
    }
  
    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 1;
    }

    switch (stopbits)
    {
        case 1:
            options.c_cflag &= ~CSTOPB;
            break;
        case 2:
            options.c_cflag |= CSTOPB;
            break;
        default:
            fprintf(stderr, "Unsupported stop bits\n");
            return 1;
    }

    //Set input parity option
    if (parity != 'n')
        options.c_iflag |= INPCK;

    tcflush(fd, TCIFLUSH);
    options.c_cc[VTIME] = 150; //timeout 15 seconds
    options.c_cc[VMIN] = 0; //Update the options and do it NOW

    //set Raw Mode
    options.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);  //Input
    options.c_oflag  &= ~OPOST;   //Output

    //clear Software Flow Control
    options.c_iflag &= ~(IXON | IXOFF | IXANY);

    //clear 0x0d and 0x0a map
    options.c_oflag &= ~(ONLCR | OCRNL);
    options.c_iflag &= ~(INLCR | ICRNL);

    if (tcsetattr(fd, TCSANOW, &options) != 0)
    {
        perror("SetupSerial 3");
        return 1;
    }
   
    return 0;
}[/img]

论坛徽章:
0
4 [报告]
发表于 2012-08-05 17:00 |只看该作者
转一篇文章给你看:

前一阵子因为工作需要摸索的一些linux下得串口通信,总结下结果, 有点乱。。。主要针对linux串口校验方式mark, space的摸索。。。
参考文档:

文档一:Serial Programming Guide for POSIX Operating Systems
文档二:Serial Programming Howto;
文档三:Serial Howto;
说明:由于当前Linux串口通讯方面的文档比较少,网上相关文档大都以这三篇为参考,第一个文档对串口设置和异步通讯介绍的都比较全面,从中能了解串口设置的全面知识,及一些细节性问题;第二个文档有一个非阻塞接收端的例子,写得很好;第三个文档主要从硬件方面介绍串口。
串口通讯程序可能用到的:

1. 打开串口

        fd = open("/dev/ttyS0", O_RDWR | O_NOCTTY | O_NDELAY);
O_RDWR 读写方式打开
O_NOCTTY 不允许进程管理串口
O_NDELAY 非阻塞
2.写串口

        n = write(fd, "ATZ\r", 4);
               n实际写入个数;
3.设置串口为非阻塞方式

        fcntl(fd, F_SETFL, FNDELAY);
4.设置串口为阻塞方式:

               fcntl(fd, F_SETFL, 0);
5.读串口:

               res = read(fd,buf,len);
6.关闭串口

               Close(fd);
串口设置部分:

波特率:

               Tcgetat tr(fd, &options);
               cfsetispeed(&options, B19200);
                cfsetospeed(&options, B19200);
               options.c_cflag |= (CLOCAL | CREAD);
            tcsetattr(fd, TCSANOW, &options);
校验位等:

No parity (8N1):
options.c_cflag &= ~PARENB
options.c_cflag &= ~CSTOPB
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS8;
Even parity (7E1):
options.c_cflag |= PARENB
options.c_cflag &= ~PARODD
options.c_cflag &= ~CSTOPB
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS7;
Odd parity (7O1):
options.c_cflag |= PARENB
options.c_cflag |= PARODD
options.c_cflag &= ~CSTOPB
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS7;
Space parity is setup the same as no parity (7S1):
options.c_cflag &= ~PARENB
options.c_cflag &= ~CSTOPB
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS8;
硬件流控制:

options.c_cflag |= CNEW_RTSCTS;
options.c_cflag &= ~CNEW_RTSCTS;
原始通讯方式

        options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);
输入控制:

        options.c_iflag |= (INPCK | ISTRIP);
输出控制:

        options.c_oflag |= OPOST;
通讯相关:

每次传送一个字节,顺序:起始电平—〉数据位—〉校验位—〉结束电平
起始电平1个Bit;
数据位一般7或8个bit
校验位1 个或者无
结束电平1,2个bit ;
通讯校验方式总结:

1. even 每个字节传送整个过程中bit为1的个数是偶数个(校验位调整个数)
2. odd 每个字节穿送整个过程中bit为1的个数是奇数个(校验位调整个数)
3. noparity没有校验位
4. space 校验位总为0
5. mark 校验位总为1;

经过上网再找相应文档,找到mark/space校验方式的具体实现,采用伪装的方式来实现;实现代码:
Mark parity is simulated by using 2 stop bits (7M1):
options.c_cflag &= ~PARENB;
options.c_cflag |= CSTOPB;
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS7;
Space parity is setup the same as no parity (7S1):
options.c_cflag &= ~PARENB;
options.c_cflag &= ~CSTOPB;
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS8;
来自:Linux Serial Port Programming Mini-Howto上的介绍
此方法利用停止位的值总是为1,这一点,来达到mark校验的校验位总为1的要求;这个办法在传ASCII码的时候没有问题;

用这个方法单桢下发,仍然接收不到数据,找原因,原来是装置的数据位是8位然后外加校验位;就是说实际是8M1;而不是要上面的7M1;类似猜测地改动8M1的实现方法如下:
Mark parity is simulated by using 2 stop bits (8M1):
options.c_cflag &= ~PARENB;
options.c_cflag |= CSTOPB;
options.c_cflag &= ~CSIZE;
options.c_cflag |= CS8;
需要8S1下发,而当前掌握的校验方式不能实现,只能到7S1,在找相应文档:
Using Linux in the real world
A talk for the Linux Users of Victoria, May 1997
这篇文档,要实现的方式与当前本公司系统要实现的系统功能相同;有一节介绍到:
MARK and SPACE parity settings, however the POSIX standard has no definition for MARK or SPACE, and there is no support for this in the standard Linux serial driver.
Fortunately, we have access to the source and only a couple of lines of code to the Linux serial driver (the diffs are below) are required to enable MARK and SPACE parity on a 80x86 system, allowing us to easily toggle MARK and SPACE.
linux/drivers/char/serial.c
1237,1248d1236
linux/include/asm-i386/termbits.h
125d124
他给出了linux下实现mark, space校验的方法;但是需要重新编译linux下的串口驱动程序;考虑到这篇文档是1997年写的,看了下当前linux下的这两个文件已经添上相应代码;所以问题可以解决;只是不知道实现方法,最终以CMSPAR为关键字google终于找到相应介绍两份:
第一份:来自一个论坛,讨论了重编译内核的可行性;并且作者已经实验成功过,备用;
第二份:来自于Using Linux in the real world
的作者:里面说明了本方法的可行性,只要在自己的应用程序中添上#define CMSPAR 010000000000再通过相应的设置就可使用Mark,space校验了;原文如下:
Re: Linux serial sticky parity (was Linux Serial Question)
Theodore Y. Ts'o
Thu, 2 Mar 2000 18:41:07 -0800
   Date: Thu, 2 Mar 2000 15:47:12 -0800
   From:

   I don't do any serial programming but it looks like there is a problem
   supporting mark parity and space parity.  James M. wants to use space
   parity.  Stty doesn't support it but the termios stucture in the
   serial driver might.  Such parity is specified by a c_cflag named
   CMSPAR in .  The equvalent is in
   as UART_LCR_SPAR 0x20.  This is called "sticky parity".  It looks like
   someone put a ? in the comment in the serial-reg.h file shown above and
   called it "stick" instead of "sticky".  If this "sticky" bit is set,
   then the parity bit is always odd (1) or even (0) depending on how the
   parity flag-bit has been set.

No, it's called "stick" parity, not "sticky".  Check the 8250 / 16550
UART documentation.

The reason why CMSPAR isn't in bits/termios.h is because it's not POSIX,
and so it was dropped during the glibc conversion by the glibc folks.  

Traditionally, the way Unix handled mark and space parity was to set or
clear the eighth bit, and simply used 8 bits no parity as far as termios
was concerned.  Actually, mark and space parity was very rare to begin
with, so it rarely came up.

The problem comes when you want to use 8 bits and also force mark or
space parity.  There is no way to do that under POSIX termios.  So, back
in 1977 I added CMSPAR for the one squeeky wheel that was complaining
that we didn't have the support.  Later, we converted to glibc, and the
glibc folks didn't add the bit.  It's indicative that no one has
complained that this feature was missing until now.

   It would seem that the high-level way to go would be to set the sticky
   parity bit in the termios structure, but will this do the job?  To
   change the termios structure one may use tcsetattr() and tcgetattr()
   as shown in Vern's HOWTO.  It seems that the flags normally used are
   in  and CMSPAR is missing from this.  
   #includes this but  #includes the one you want that
   contains the CMSPAR flags.  Will this flag work OK if one changes the
   #includes in their program to the  directory?  Why isn't it in
   ?

So yes, if you set the CMSPAR bit in the termios, it will do the right
thing.  #including asm/termios.h is a bad idea, though.  In the new
header file religion, it's always a bad thing to include anything in the
asm/ directory.  You should complain to the glibc folks and ask them to
include CMSPAR in bits/termsio.h.

In the meantime, a program who needs this bit badly can simply manually
define CMSPAR, and they should be warned that this is a Linux specific
feature that won't work on any other OS.

参考了一下串口的硬件驱动程序:

Stick     even parity    parity enable        parity
-               -               0                    无校验
0               0               1                    奇校验
0               1               1                    偶校验
1               0               1                    Mark
1               1               1                    space
得到如下的校验设置方式:#define  CMSPAR 010000000000
本句使能了stick parity的校验可行性
Mark校验
options.c_cflag |= PARENB | CS8 | CMSPAR |PARODD;
Space校验

options.c_cflag |= PARENB | CS8 | CMSPAR;

实验结果可行;
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP