我写了一个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
语句似乎数据没有正确传递。即使在调试器中,我也无法读取参数。
令人沮丧的是,有一点,这只是工作..我已经检查了文件权限,检查路径等。我可以从fortran包装器可执行文件运行该函数就好了。
我缺少一个技巧吗?
谢谢
答案 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)
你的程序似乎会破坏堆栈。堆栈损坏通常伴随着指针的不正确使用。在可疑函数调用之前,请仔细检查(使用调试器或简单的封装来调查变量的内容)。