PHP扩展中的SEG错误

时间:2013-01-02 15:36:16

标签: php c segmentation-fault php-extension php-internals

我写了一个PHP扩展来访问静态库中的函数,我将PHP构建为CGI,一切似乎都有效(经过几天的工作......)

一旦所有工作都兴奋,我重新编译PHP而没有调试我在其中的消息。 (php_printf("here111"); .... php_printf("sending arguments...");

然后,它就停止了工作。我在静态库中调用的函数有效,我通过直接从另一个可执行文件中调用它来测试它。

我使用调试符号(--enable-debug)构建了PHP,并且可以在gdb中将其调试到一定程度。

我仍然在努力弄清楚出了什么问题。似乎lib(diffFst)中的函数似乎无法读取输入参数。

268     if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ssssssd",
269        &filA, &filA_len,
270        &nomvarA, &nomvarA_len,
271        &filB, &filB_len,
272        &nomvarB, &nomvarB_len,
273        &filO, &filO_len,
274        &newnomvar, &newnomvar_len,
275        &mult
276        ) == FAILURE) {
277         RETURN_LONG(-100);
278     }
279 
280     php_printf("Read arguments:\nfilA: %s, nomvara: %s\nfilB: %s, nomvarB: %s\nfilO: %s, nomvarO: %s\nMult: %0.3f\n",
281        filA,nomvarA, filB,nomvarB, filO,newnomvar, mult);
282 
285     ier = difffst_(filA,nomvarA, filB,nomvarB, filO,newnomvar, mult);

当我调用此函数时,php_printf()语句有效并打印出正确的值。但是当我让它调用difffst_函数时,我在尝试读取输入变量时会遇到段错误。

diffFst函数用fortran编写:

  5 function diffFst(filA, nomvara, filB, nomvarb, filO, newnomvar, change, write_tictac, in_verbose) result(ier)
 10     implicit none
 11 

 12     character (len=*), intent(IN) :: filA, filB, filO
 13     character (len=*), intent(IN) :: nomvara, nomvarb, newnomvar
 14 
 16     real, intent(IN) :: change
 17     logical, intent(IN) :: write_tictac
 18 
 19     logical, intent(IN), optional :: in_verbose
 21     logical :: verbose = .false.
 27     integer :: ier
...
117     ier = fstouv(iuna, 'RND')
118     IF (ier < 0) THEN
119         if (verbose) write(stderr,'(2A)') "Could not fstouv FST file ", trim(filA)
120     ELSE
121         nmax = fstnbr(iuna);
122         if (verbose) write(stdout,'(3A,I6,A)') "Succesfully opened ", trim(filA), ' with ', nmax, ' records'
123         allocate(liste(nmax))
124     END IF

具体来说,当它尝试阅读filA时,它在第122行(根据调试器)失败。

我不知道为什么,我试过了:

  • 使该功能成为子程序
  • 使该功能成为一项功能
  • 使该功能成为'纯'功能
  • 有返回值(即现在的那些,ier = ..
  • 在代码中包含return个语句,删除return语句
  • 尝试将内容打印到stdout以及记录文件

似乎数据没有正确传递。即使在调试器中,我也无法读取参数。

令人沮丧的是,有一点,这只是工作..我已经检查了文件权限,检查路径等。我可以从fortran包装器可执行文件运行该函数就好了。

我缺少一个技巧吗?

谢谢

2 个答案:

答案 0 :(得分:2)

花了一段时间,需要额外的帮助(issues like this

基本上有两件事需要改变:

  • 通过引用传递整数
  • 正确接受字符串

第一个很简单,只需ier=func(..., &integer_var, ...)

第二个涉及传递字符串的长度。可能有一种更简单的方法(通过查找\ 0感知字符串长度),但它们没有成功。所以,现在我通过

ier = func(str,strlen,...)

然后在Fortran中,我接受字符串

character(kind=c_char,len=strlen), intent(IN) :: str

上面对fortran代码的具体更改是

11     use, intrinsic :: iso_c_binding
12     use interfaces_rmnutils
13     implicit none
16     integer(kind=c_int), intent(IN) :: len_filA, len_filB, len_filO
17     character (kind=c_char,len=len_filA), intent(IN) :: filA
18     character (kind=c_char,len=len_filB), intent(IN) :: filB
19     character (kind=c_char,len=len_filO), intent(IN) :: filO

当它工作时,它必须在我尝试将字符串读入(len=*)之前,并且整数作为引用传入,因此它们本质上具有随机值。

感谢所有人!

答案 1 :(得分:1)

你的程序似乎会破坏堆栈。堆栈损坏通常伴随着指针的不正确使用。在可疑函数调用之前,请仔细检查(使用调试器或简单的封装来调查变量的内容)。