C ++程序崩溃终端 - 如何调试?

时间:2012-06-07 02:53:12

标签: c++ terminal serial-port arduino

不要问我怎么做 - 我没有丝毫。

以下代码崩溃了我的终端和我使用的任何运行时分析工具,同时没有引发任何静态检查工具警告。 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”编译

2 个答案:

答案 0 :(得分:5)

如果通过“崩溃我的终端”你的意思是终端停止工作,原因很可能是你在stdout(也就是你的终端)上将各种标志设置为零。您正在调用tcsetattr()并基本上将所有终端控制标志归零。

您应该使用tcgetattr()获取当前终端标志,修改您要修改的标志,然后使用该结构来调用tcsetattr()。不这样做甚至是undefined behavior

  

如果tcsetattr()指向的termios结构的值不是来自termios_p tcgetattr()的调用结果,则fildes的效果未定义1}};应用程序应该在调用tcgetattr()tcsetattr()之间仅修改此规范定义的字段和标志,并保留所有其他字段和标志不被修改。

答案 1 :(得分:2)

如果您使用linux,则可以使用strace

strace ./yourProgram

您也可以调试coredump文件,

ulimit -c unlimited
g++ -g ...
程序崩溃时将创建

而不是coredump文件。

gdb ./yourProgram coredump

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