Emscripten:调用修改数组元素的C函数

时间:2015-01-28 21:28:54

标签: javascript c arrays node.js emscripten

我有一个简单的C函数来修改整数数组的元素。我可以使用Emscripten(emcc)将其转换为JavaScript而不会出现问题。但是当我在JS数组上调用该函数时,其中的值似乎没有改变。请帮忙。

这是C函数定义:

/* modify_array.c */
void modify_array(int X[8]) {
  int i;
  for (i = 0; i < 8; ++i) {
    X[i] += 1;
  }
}

这是我用来将C代码转换为JS的命令:

emcc modify_array.c -o modify_array.js -s EXPORTED_FUNCTIONS="['_modify_array']"

这是用于调用已转换的JS代码的JavaScript(Node.js)代码:

var mod = require("./modify_array.js");
var f = mod.cwrap("modify_array", "undefined", ["array"]);

var X = [0, 1, 2, 3, 4, 5, 6, 7];
var bytesX = new Uint8Array(new Int32Array(X).buffer);

/* Invoke the emscripten-transpiled function */
f(bytesX);


console.log(new Int32Array(bytesX.buffer));

运行JS代码后,缓冲区包含的值与原始值相同,而不是递增的值。为什么?如何获取更新的值?

1 个答案:

答案 0 :(得分:8)

Emscripten的内存模型是单个平面阵列。这意味着当您向已编译的C方法提供数据数组时,它将被复制到单个数组中,这是唯一可以访问它的地方(ccall / cwrap方法为您执行此操作)。换句话说,您传递的所有参数都是按值而不是通过引用,即使它们是数组(在JS中通常通过引用传递)。

要在emscripten内存模型中工作,您可以在单个平面阵列中使用内存

var ptr = Module._malloc(8);
var view = Module.HEAPU8.subarray(ptr, ptr+8);
var f = Module.cwrap("modify_array", "undefined", ["number"]);
f(ptr);

在单个数组中保留一个空格,并在单个数组上使用子数组,我们可以访问它的值。请注意使用number作为类型。我们正在传递ptr,它是指向缓冲区的指针。作为指针,它只是一个数字,指的是单个数组中的位置。

(请注意,您应该调用free以在适当的时间释放内存。)