我正在尝试编写一些 XS 代码,将库作为可以写入的流接口公开给 Perl 代码。下面的
get_stream
函数应该是一个构造函数,它准备并返回一个 PerlIO 对象。我认为我只需要
Write
andClose
方法,所以我将所有其他函数槽留空。
typedef struct {
struct _PerlIO base;
mylib_context* ctx;
} PerlIOmylib;
/* [...] */
PERLIO_FUNCS_DECL(PerlIO_mylib_funcs) = {
.fsize = sizeof(PerlIO_funcs),
.name = "mylib",
.size = sizeof(PerlIOmylib,
.Write = mylib_write,
.Close = mylib_close,
};
/* XS below */
PerlIO*
get_stream (SV* context_obj)
CODE:
mylib_context* ctx = (mylib_context*) SvIV (SvRV (context_obj));
PerlIO* f = PerlIO_allocate (aTHX);
f = PerlIO_push (aTHX, f, PERLIO_FUNCS_CAST(&PerlIO_mylib_funcs), "a", NULL);
PerlIOSelf(f, PerlIOmylib)->ctx = ctx;
PerlIOBase(f)->flags |= PERLIO_F_OPEN;
RETVAL = f;
OUTPUT:
RETVAL
当我像这样使用提供的界面时......
{
my $fh = MyLib::get_stream($lib_ctx);
print $fh "x" x 300;
}
...mylib_write
函数被调用,所以到目前为止我还没有完全搞砸。(我通过插入调试 printf 语句验证了这一点。)但是,我希望 PerlIO 对象在
$fh
超出范围时关闭,就像使用open
. 但目前,该mylib_close
函数仅在解释器关闭期间调用。
直接调用close
工作正常,设置$fh
为undef
不行。
更新:按照 ikegami 的建议,我使用Devel::Peek::Dump
并sv_dump
发现句柄返回get_stream
函数是一个“RV”,它指向一个SV = PVGV(...)
. glob ( PVGV
) 的引用计数器设置为 3,这似乎不正确。
我添加了
CLEANUP:
SvREFCNT_dec (SvRV (ST(0)));
SvREFCNT_dec (SvRV (ST(0)));
治愈症状:
在块结束时超出范围时close
调用该函数。$fh
但是我仍然不太了解潜在的问题。
这是为该OUTPUT
部分生成的 C 代码:
ST(0) = sv_newmortal();
{
GV *gv = newGVgen("MyLib");
if (do_open(gv, "+<&", 3, FALSE, 0, 0, RETVAL) )
sv_setsv(ST(0), sv_bless(newRV((SV*)gv), gv_stashpv("MyLib",1)));
else
ST(0) = &PL_sv_undef;
}
XSRETURN(1);
GV 的引用计数如何最终达到 3?