没有副本的指针的Perl SV值

时间:2015-06-10 17:06:55

标签: perl xs

如何从没有副本的空终止字符串创建SV值?像newSVpv(const char*, STRLEN)但没有副本和移动所有权到Perl(因此Perl必须释放该字符串内存)。我需要这个来避免大量的内存分配和复制。

我找到了以下示例:

SV *r = sv_newmortal();
SvPOK_on(r);
sv_usepvn_mg(r, string, strlen(string) + 1);

但我对XS内部知识并不了解并有一些疑问。

1 个答案:

答案 0 :(得分:6)

如果您希望Perl管理内存块,它需要知道如何重新分配它并释放它。它知道如何重新分配和释放的唯一内存是使用其分配器Newx分配的内存。 (否则,它必须将重新分配器和解除分配器与每个内存块相关联。)

如果您不能使用Newx分配内存块,那么您最好的选择可能是创建一个SvLEN设置为零的只读SV。这告诉Perl它不拥有内存。 SV可以被祝福到一个具有析构函数的类中,该析构函数将使用适当的解除分配器释放内存。

如果您可以使用Newx分配内存块,则可以使用以下内容:

SV* newSVpvn_steal_flags(pTHX_ const char* ptr, STRLEN len, const U32 flags) {
#define newSVpvn_steal_flags(a,b,c) newSVpvn_steal_flags(aTHX_ a,b,c)
    SV* sv;

    assert(!(flags & ~(SVf_UTF8|SVs_TEMP|SV_HAS_TRAILING_NUL)));

    sv = newSV(0);
    sv_usepvn_flags(sv, ptr, len, flags & SV_HAS_TRAILING_NUL);
    if ((flags & SVf_UTF8) && SvOK(sv)) {
        SvUTF8_on(sv);
    }

    SvTAINT(sv);

    if (flags & SVs_TEMP) {
        sv_2mortal(sv);
    }

    return sv;
}

注意:ptr应指向Newx分配的内存,并且必须指向Newx返回的块的开头。

注意:接受标记SVf_UTF8(以指定ptr是要在Perl中看到的字符串的UTF-8编码),SVs_TEMP(要sv_2mortal呼叫SV)和SV_HAS_TRAILING_NUL(见下文)。

注意:某些代码期望标量的字符串缓冲区具有尾随NUL(即使缓冲区的长度已知,即使缓冲区可以包含NUL)。如果您分配的内存块具有超出数据末尾的尾随NUL(例如,C样式的NUL终止字符串),则传递SV_HAS_TRAILING_NUL标志。如果没有,该函数将尝试扩展缓冲区并添加NUL。