3

不要问我怎么做的——我一点也没有。

以下代码使我的终端和我使用的任何运行时分析工具崩溃,但不会引发任何静态检查工具警告。Valgrind、cppcheck 和 gdb 对我没什么好处。g++ -Wall 没有给我任何有用的东西。

代码的目标是通过 USB 串行连接将字符写入 audrino。audrino 地址作为第一个参数传递。传递一个 unsigned int 并将其强制转换为一个 unsigned char。

#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <termios.h>
#include <iostream>

using namespace std;

int main(int argc,char** argv){

  struct termios stdio;
  int tty_fd;

  unsigned char ctrlChar;
  unsigned int ctrlInt;

  tty_fd=open(argv[1], O_RDWR | O_NONBLOCK);
  tcgetattr(tty_fd, &stdio);
  cfmakeraw(&stdio);
  stdio.c_cc[VMIN]  = 1;
  stdio.c_cc[VTIME] = 0;
  tcsetattr(tty_fd,TCSANOW, &stdio);
  tcsetattr(tty_fd,TCSAFLUSH, &stdio);
  fcntl(tty_fd, F_SETFL, fcntl(STDIN_FILENO, F_GETFL, 0) | O_NONBLOCK);

  cfsetospeed(&stdio,B9600);            // 9600 baud
  cfsetispeed(&stdio,B9600);            // 9600 baud

  cout << "ready on " << argv[1] << endl;

  scanf("%u", ctrlInt);
  cout <<"recieved initial read.\n";
  while (ctrlInt < 256){
    ctrlChar = ((char) ctrlInt);
    cout << ctrlInt << ctrlChar << endl;
    write(tty_fd,&ctrlChar,1);
    cout << "sent to audrino.\n";
    scanf("%u", ctrlInt);
  }

  cout << "done!" << endl;

  close(tty_fd);
  return 0;
}

现在,如果这看起来很正常,我将提交一份错误报告。

系统规格:最新的 Arch linux 64 位,使用“g++ -g -Wall code.c”编译

4

2 回答 2

5

如果“我的终端崩溃”是指终端停止工作,那么原因很可能是您在标准输出(也就是您的终端)上将各种标志设置为零。您正在调用tcsetattr()并将基本上所有终端控制标志清零。

您应该使用 获取当前终端标志tcgetattr(),修改您要修改的那些,然后使用该结构来调用tcsetattr(). 不这样做甚至是未定义的行为

如果指向的结构的值不是从调用on的结果派生的,则的效果tcsetattr()是不确定的;应用程序应该只修改在调用和之间由本规范定义的字段和标志,而不修改所有其他字段和标志。termiostermios_ptcgetattr()fildestcgetattr()tcsetattr()

于 2012-06-07T03:17:40.240 回答
2

如果你使用linux,你可以使用strace

strace ./yourProgram

您还可以调试 coredump 文件,

ulimit -c unlimited
g++ -g ...

当程序崩溃时,将创建 coredump 文件。

gdb ./yourProgram coredump

使用bt命令查看崩溃信息。

于 2012-06-07T03:12:19.833 回答