我是XS的初学者,花了一些时间在网上寻找这个答案而没有运气。问题是XS更改了函数的名称,当它进行编译时,我将得到一个未定义的引用错误。例如,考虑下面的XS代码:
size_t
matrixIndex (colIndex, rowIndex,nCols,nRows)
size_t colIndex
size_t rowIndex
size_t nCols
size_t nRows
CODE:
size_t register i;
RETVAL = (rowIndex * nCols) + colIndex;
OUTPUT:
RETVAL
然后我尝试在以下函数中使用它,如下所示
int
matrixCopyColumnVector_dbl (colIndex,fromMatrix,nColsMatrix,nRowsMatrix,intoVector,nRowsVector)
size_t colIndex
SV * fromMatrix
size_t nColsMatrix
size_t nRowsMatrix
SV * intoVector
size_t nRowsVector
CODE:
size_t register x, n;
if( nRowsVector != nRowsMatrix) { RETVAL = 0; return RETVAL; }
n = 0;
for(x=0; x<= nRowsMatrix; x++) {
intoVector[n] = fromMatrix[matrixIndex /*USE OF FUNCTION HERE!!*/(colIndex,x,nColsMatrix,nRowsMatrix)];
n++;
}
RETVAL = 1;
return RETVAL;
OUTPUT:
RETVAL
然后我运行make
并进行编译过程,我在undefined reference to 'matrixIndex'
的链接阶段出现错误。
所以我想知道在同一个XS文件中调用函数的标准XS方法是什么?
答案 0 :(得分:5)
XS代码创建 Perl 潜艇。所以调用XS函数与calling任何其他Perl子函数相同。
不是处理那种复杂性和低效率,而是创建一个C函数而不是Perl sub。 (如果需要,可以使用XS独立公开该C函数。)
#define PERL_NO_GET_CONTEXT
#include "EXTERN.h"
#include "perl.h"
#include "XSUB.h"
static UV matrixIndex(UV colIndex, UV rowIndex, UV nCols, UV nRows) {
return (rowIndex * nCols) + colIndex;
}
MODULE = Foo::Bar PACKAGE = Foo::Bar
int
matrixCopyColumnVector_dbl(colIndex, fromMatrix, nColsMatrix, nRowsMatrix, intoVector, nRowsVector)
UV colIndex
SV * fromMatrix
UV nColsMatrix
UV nRowsMatrix
SV * intoVector
UV nRowsVector
PREINIT:
UV register x, n;
CODE:
if (nRowsVector == nRowsMatrix) {
RETVAL = 0;
} else {
n = 0;
for (x=0; x<=nRowsMatrix; x++) {
intoVector[n] = fromMatrix[matrixIndex(colIndex, x, nColsMatrix, nRowsMatrix)];
n++;
}
RETVAL = 1;
}
OUTPUT:
RETVAL
您对return
的使用不正确。如果您想提前返回,请使用XSRETURN*
宏之一。
fromMatrix[...]
和intoVector[...]
完全错误。 fromMatrix
和intoVector
是C数组。 (它们甚至不是Perl阵列,而不是相关的。)
Perl整数的大小为IV
(或UV
为无符号),不一定是size_t
。使用它们以获得最佳兼容性。
如果您想要便携性,则不能假设C99,因此您无法混合声明和代码。您需要在PREINIT
中放置声明(或使用CODE
中的curlies为变量声明创建新范围。)
答案 1 :(得分:2)
XSUB不是C函数。 XS预处理器确实从您的声明中创建了一个C函数,但它具有签名void (CV*)
。然后,XSUB从Perl参数堆栈中获取参数,并使用您的类型映射将它们转换为C值。因此无法直接调用您的XSUB。
相反,您必须像调用任何其他Perl函数一样调用XSUB,即将参数作为SV*
推送到堆栈上。有关详细信息,请参阅perldoc perlcall
。显然,这是乏味且低效的。
更好的解决方案是将您想要在Perl和XS中使用的所有函数声明为XS文件中的static
C函数,然后编写XS粘合代码以将其公开给Perl。这里:
static size_t matrixIndex(size_t colIndex, size_t rowIndex, size_t nCols, size_t nRows)
{
size_t register i; // unneeded
return (rowIndex * nCols) + colIndex;
}
请注意,如果此函数使用Perl API的任何部分,那么您还应该使用pTHX
和aTHX
宏。声明变为
static size_t matrixIndex(pTHX_ size_t colIndex, etc...) ...
当从C调用它时,你会写matrixIndex(aTHX_ colIndex, etc...)
。