调用 enif_free() 时出现分段错误

Segmentation fault when calling enif_free()

我有一些代码可以在 Erlang 中管理具有 O(1) 访问和读取时间的 3 维数组。因此,我正在使用 Erlang NIF。除 release() 函数外,一切正常。调用它时总是出现分段错误,我不知道为什么。

这是我的代码:

#include "erl_nif.h"

static ErlNifResourceType *DATA_RESOURCE;

typedef struct
{
    int size;
    ERL_NIF_TERM *** array;
    ERL_NIF_TERM defaultValue;
} DATA;

static ERL_NIF_TERM new3DimArray(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
    DATA *data = (DATA *)enif_alloc_resource(DATA_RESOURCE, sizeof(DATA));
    int size;
    enif_get_int(env, argv[0], &size);
    if(argc > 1)
    {
        data->defaultValue = argv[1];
    }else{
        data->defaultValue = NULL;
    }
    data->size = size;
    data->array =  (ERL_NIF_TERM ***)enif_alloc(sizeof(ERL_NIF_TERM **) * size);
    int x = 0;
    while(x < size)
    {
        data->array[x] = (ERL_NIF_TERM **)enif_alloc(sizeof(ERL_NIF_TERM *) * size);
        int y = 0;
        while(y < size)
        {
            data->array[x][y] =  (ERL_NIF_TERM *)enif_alloc(sizeof(ERL_NIF_TERM) * size);
            y++;
        }
        x++;
    }
    return enif_make_resource(env, data);
}

static ERL_NIF_TERM get_nif(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
    DATA *data;
    enif_get_resource(env, argv[0], DATA_RESOURCE, &data);
    int x;
    int y;
    int z;
    enif_get_int(env, argv[1], &x);
    enif_get_int(env, argv[2], &y);
    enif_get_int(env, argv[3], &z);
    ERL_NIF_TERM res = data->array[x][y][z];
    if(res == NULL && data->defaultValue != NULL)
    {
        res = data->defaultValue;
    }
    return res;
}

static void set_nif(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
    DATA *data;
    enif_get_resource(env, argv[0], DATA_RESOURCE, &data);
    int x;
    int y;
    int z;
    enif_get_int(env, argv[1], &x);
    enif_get_int(env, argv[2], &y);
    enif_get_int(env, argv[3], &z);
    ERL_NIF_TERM value = argv[4];
    data->array[x][y][z] = value;
}

static void release(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
    DATA *data;
    enif_get_resource(env, argv[0], DATA_RESOURCE, &data);
    int x = 0;
    while(x < data->size)
    {
        int y = 0;
        while(y < data->size)
        {
            enif_free(data->array[x][y]);
            y++;
        }
        enif_free(data->array[x]);
        x++;
    }
    enif_free(data->array);
    enif_release_resource(data);
}

static void cleanup(ErlNifEnv *env, void *obj){}

static int load(ErlNifEnv *env, void **priv_data, ERL_NIF_TERM load_info){
    DATA_RESOURCE = enif_open_resource_type(env, "mutArray", "DATA_RESOURCE", &cleanup, ERL_NIF_RT_CREATE, 0);
    return 0;
}

static ErlNifFunc nif_funcs[] = {
    {"new_3_dim_array", 1, new3DimArray},
    {"new_3_dim_array", 2, new3DimArray},
    {"get", 4, get_nif},
    {"set", 5, set_nif},
    {"release", 1, release}
};

ERL_NIF_INIT(mutArray, nif_funcs, load, NULL, NULL, NULL);

这是我的 Erlang 代码(为了更清晰):

module(mutArray).

%% ====================================================================
%% API functions
%% ====================================================================
-export([init/0, new_3_dim_array/1, new_3_dim_array/2, get/4, set/5, release/1]).

init() ->
    erlang:load_nif("./mutArray", 0).

new_3_dim_array(_Size) ->
    "NIF not loaded yet.".

new_3_dim_array(_Size, _DefaultValue) ->
    "NIF not loaded yet.".

get(_Array, _X, _Y, _Z) ->
    "NIF not loaded yet.".

set(_Array, _X, _Y, _Z, _Value) ->
    "NIF not loaded yet.".

release(_Array) ->
    "NIF not loaded yet.".

顺便说一句,这是我的测试代码:

mutArray:init(),
A = mutArray:new_3_dim_array(100),
mutArray:release(A).

编辑:好的,它变得越来越奇怪......经过一些测试后,我发现如果 enif_free(data->array); 是该函数的最后一次调用,我会得到 ** exception error: [] 。在每个其他位置我仍然遇到分段错误,即使在 enif_free(data->array); 之后只有一个 println()。经过一些调试后,我还发现 enif_free(data->array); 之前的每一行都被调用了。所以异常似乎发生在enif_free(data->array)。有人知道这是什么意思吗?

EDIT2:简单地离开 enif_free(data->array); 也无济于事。我也遇到了分段错误。

通过解决几个问题,我能够正确地获取您的代码 运行。

首先,您的代码假定可以通过将 ERL_NIF_TERMNULL 进行比较来检查其有效性,这是不正确的。您可以通过将所有数组元素初始化为 0(通过调用 enif_make_int(env, 0) 来设置每个元素)或使用结构数组来解决此问题,其中每个结构包含一个 ERL_NIF_TERM 和一个 unsigned char 标志来指示该术语是否有效。如果您选择后一种方法,您可以简单地 memset 将结构值设置为 0,如果调用者通过 mutArray:get/4 请求未初始化的元素,只需 return enif_make_badarg(env) 来指示它们向调用传递了错误的参数。

其次,当您需要 return ERL_NIF_TERM 时,您的 set_nifrelease 函数都被声明为 return void。要解决此问题,您可以更正它们的 return 类型,然后 return argv[4] 来自 set_nifenif_make_int(env, 0) 来自 release.

最后,您的 enif_open_resource_type 调用的第二个参数需要是 NULL 而不是您传递的 "mutArray" 值,如 the erl_nif man page 所示。