5

我正在尝试使用 Linux 下的 termios 框架通过 UART(usbserial)连接非接触式智能卡读卡器。该代码在 PC 上运行良好,但是当我在 ARM9 目标上进行交叉编译并试用时,它能够打开设备甚至将命令写入设备,但读取命令会无限期地阻塞。这是代码片段:

int mifare_rdr_init(struct mifare_1K * ptr, char *rdr_devnode)
{   
    bzero(ptr, sizeof(struct mifare_1K));           // zero the entire structure
    // open serial device
    int fd = open(rdr_devnode, O_RDWR|O_NOCTTY );
    if (fd == -1) {
    perror("Failed to open serial device ");
    return 1;
    }
    ptr->serialfd = fd;                 // save file descriptor

    ptr->serialdev.c_iflag = 0;                 // no i/p flags
    ptr->serialdev.c_oflag = 0;                 // o/p flags
    ptr->serialdev.c_cflag = ( CS8 | CREAD | B38400 );      // 8 bits, receive enable, baud for rdr
    ptr->serialdev.c_lflag = ( ICANON );                // CANONICAL mode, means read till newline char '\n'.
    // control chars 
        // commented below line as suggested by A.H below, since it's not needed in CANONICAL mode
    // ptr->serialdev.c_cc[VMIN] = 1;               // read unblocks only after at least one received char.

    // flush all i/o garbage data if present
    tcflush(ptr->serialfd,TCIOFLUSH);

    int ret = 0;
    // apply settings
    ret = tcsetattr(ptr->serialfd,TCSANOW,&ptr->serialdev);
    if (ret == -1) {
        perror("tcsetattr() failed ");
        return 2;
    }
    return 0;
    }

int get_mifare_rdr_version(struct mifare_1K *ptr, char *data)
{
    // flush all i/o garbage data if present
    tcflush(ptr->serialfd,TCIOFLUSH);

    int chars_written = write(ptr->serialfd,"$1V\n",4);
    if( chars_written < 0 ) {
        perror("Failed to write serial device ");
        return 1;
    }
        printf("cmd sent, read version...\n");   // this prints, so I know cmd sent...
    int chars_read = read(ptr->serialfd,ptr->data_buf,14);
    if( chars_read < 0 ) {
        perror("Failed to read serial device ");
        return 2;
    }
    // copy data to user buffer
        printf("reading done.\n");    // this doesn't print...
    return 0;
}

mifare_1K 结构包含串行设备的文件描述符、termios 结构和我正在使用的各种缓冲区。我提到的设备是usb-to-serial(模块:ftdi_sio)设备。它在 termios 的规范模式下配置为 38400@8-N-1。

规范模式,因为来自阅读器的响应以“\n”结尾,因此在规范模式下处理更好,因为它会读取设备直到收到“\n”(如果我错了,请纠正我)。

首先我调用 init() fn,然后调用 get_rdr_version()。字符串“cmd sent, read version...”被打印出来,所以我知道它可以写,但不打印字符串“reading done”。在那之后。

另一件事是,如果我卸下读卡器并将该端口连接到另一台 PC 上的 gtkterm(串行端口终端程序),我不会在该 gtkterm 上收到“$1V\n”??!!. 然后经过一点 RnD,我发现如果我重新启动连接读卡器的系统,那么只有我在另一个 Gtkterm 上得到那个 cmd "$1V\n"。如果我在不重新启动的情况下重试,则在该 Gkterm 上看不到该 cmd ...一个线索,但尚未弄清楚。

是否类似于将 cmd 写入设备文件,但没有排入实际设备?有什么办法可以检查这个吗?

任何帮助都深表感谢,因为我已经被困在这个 fr 一段时间了......谢谢。

更新 :

好的,我通过稍微修改代码来让它工作,如下所示。

// open serial device
    int fd = open(rdr_devnode, O_RDWR|O_NOCTTY|O_NDELAY );  // O_NDELAY ignores the status of DCD line, all read/write calls after this will be non-blocking
    fcntl(fd,F_SETFL,0);   // restore read/write blocking behavior
    if (fd == -1) {
    perror("Failed to open serial device ");
    return 1;
    }

这是在我的 init() 函数中打开端口的代码的修改部分。两个变化:

1) O_NDELAY 添加到 open() 调用中的标志,它忽略数据载波检测 (DCD) 线,以查看另一端是否已连接并准备好进行通信。这本来是用于MODEM的,我不需要,事实上,因为我使用的是usbserial,所以我根本没有它。但是这个标志也会进一步调用 read() 和 write() 作为非阻塞。需要注意的是,我曾认为可以通过将 CLOCAL 添加到 termios 结构的 cflag 来解决这个问题,我尝试过但没有奏效。

2) fcntl(fd,F_SETFL,0) 恢复进一步 read() 和 write() 调用的阻塞行为。

这个组合对我来说非常有效。我没有将此作为答案发布的唯一原因是我还不明白为什么它在没有这种修改的情况下在 PC 上工作,因为它是相同的硬件。事实上,我可以使用minicom从 ARM9 TARGET 上的智能卡读卡器读取数据,但不是我的程序。我将检查 FT232BL 文档,看看默认情况下 DCD 的状态是什么。

无论如何,我在POSIX 操作系统的串行编程指南中找到了一些信息。解释任何人???当然,当我找到答案时,我会发布答案..

干杯:)

4

2 回答 2

4

刚刚在带有 Telegesis USB 模块的 Raspberry Pi 上遇到了相同的症状,我将其添加为另一个数据点。

在我的情况下,原因原来是缺少 RTS 标志。Telegesis 需要 CRTSCTS 流控制,并且不会在没有看到 RTS 的情况下向 Raspberry 发送任何数据。这里令人困惑的方面是 a) 相同的代码在 PC 上运行良好,b) 在第一次插入 Telegesis 时它在 Raspberry 上运行良好,但在随后打开 /dev/ttyUSB0 时不会看到任何数据由覆盆子。

出于某种原因,似乎在 ARM 上,RTS 标志在设备关闭时被清除,但没有再次设置,而在 x86/x64 上,RTS 标志保持设置。这里的修复只是设置 RTS 标志,如果它还没有 - 例如

#include <sys/ioctl.h>
//...
int rtscts = 0;
if (ioctl (fd, TIOCMGET, &rtscts) != 0)
{
  // handle error
}
else if (!(rtscts & TIOCM_RTS))
{
  rtscts |= TIOCM_RTS;
  if (ioctl (fd, TIOCMSET, &rtscts) != 0)
  {
    // handle error
  }
}

我确实注意到,在您的情况下,您不使用流量控制,因此上述内容很可能不适用。然而,正是您的问题和提到 minicom 的工作使我们找到了解决问题的方法 - 所以谢谢您!

于 2013-01-07T06:30:24.033 回答
1

您可能会检查三件事:

规范模式/非规范模式混合 1

您正在混合规范模式和非规范模式的东西:

ptr->serialdev.c_lflag = ( ICANON );
// ...
ptr->serialdev.c_cc[VMIN] = 1;

手册页说明了termios(3)VMIN:

VMIN规范读取 的最小字符数。

所以很明显你的超时不会像你想象的那样工作。

规范模式/非规范模式混合 2

此外,手册页进一步说明如下:

这些符号下标值都是不同的,除了 VTIME, VMIN 可能分别与 VEOL, VEOF具有相同的值。在非规范模式下,特殊字符含义被超时含义所取代。有关 VMIN 和 VTIME 的说明,请参见下面的非规范模式说明。

因此,请检查您的两个平台的这些常量的定义是否不同。错误的设置可能会破坏 EOL/EOF 逻辑。EOL 和 EOF 都可能导致从read.

未初始化c_cc

您的代码未显示c_cc数组的正确初始化。您既没有阅读现有设置,也没有为规范模式所需的值提供合适的默认值。到目前为止显示的代码甚至没有清除这些值。因此可能会使用不可预测的值。

于 2011-12-28T11:38:14.630 回答