当字符串太短时,为什么popt会出现段错误?

时间:2015-04-23 05:49:34

标签: c segmentation-fault popt

argDescrip字符串不够长时,为什么popt会成为segfaulting?

采用以下示例:

#include <popt.h>

struct poptOption options[] = {
  POPT_AUTOHELP
  {
    .longName = "color",
    .shortName = '\0',
    .argInfo = POPT_ARG_STRING
               | POPT_ARGFLAG_OPTIONAL
               | POPT_ARGFLAG_SHOW_DEFAULT,
    .arg = (void*) "auto",
    .val = 1,
    .descrip = "Whether or not to use colored output",
    .argDescrip = "always|never|auto (checks if TTY)"
  },
  POPT_TABLEEND
};


int main(int argc, const char** argv) {
  // get popt context
  poptContext ctx = poptGetContext(
      "program",
      argc, argv,
      options,
      POPT_CONTEXT_POSIXMEHARDER);

  // parse
  poptGetNextOpt(ctx);

  return 0;
}

以上段错误:

/tmp$ ./a.out --help
Usage: a.out [OPTION...]
[1]    47468 segmentation fault  ./a.out --help

虽然将.argDescrip更改为

.argDescrip = "always|never|auto (checks if TTY) "
              "........................................."

popt愉快地接受它并显示输出:

/tmp$ ./a.out --help
Usage: a.out [OPTION...]
      --color[=always|never|auto (checks if TTY) .........................................]     Whether or not to use colored output

Help options:
  -?, --help                                                                                

Show this help message
    --usage
Display brief usage message

是什么给出的?我错过了C规范中的某些内容吗? popt man pages未指定所需长度。这是一个错误吗?

1 个答案:

答案 0 :(得分:1)

你的问题是别的。您已将arg成员设置为常量字符串,而您需要将其设置为指向此类的指针(手册页显示,应使用指向char *的指针初始化arg)。您的程序在尝试取消引用非指针时会死亡。

试试这样:

char *garg = "auto";

struct poptOption options[] = {
    POPT_AUTOHELP
    {
        .longName = "color",
        .shortName = '\0',
        .argInfo = POPT_ARG_STRING
                | POPT_ARGFLAG_OPTIONAL
                | POPT_ARGFLAG_SHOW_DEFAULT,
        .arg = &garg, 
        .val = 1,
        .descrip = "Whether or not to use colored output",
        .argDescrip = "always|never|auto (checks if TTY)"
    },
    POPT_TABLEEND
};