Emscripten: как получить массив символов в c из javascript

Я новичок в Emscripten / javascript, поэтому заранее прошу прощения, если моя ситуация уже решена. У меня есть приложение в ionic3, и я хочу прочитать массив в c, который имеет 3 позиции, и у этих позиций есть другой массив, объявленный в структуре:

struct struct_name {
    char text1[10];
    char text2[30];
    char text3[50];
    int int1;
    int int2;
    int int3;
};

float EMSCRIPTEN_KEEPALIVE calcArray(struct struct_name miArray[3]){
    int value = 0;
    for (int x = 0; x < 3; x ++) {
        if (strcmp(miArray[x].text3, “PRUEBA”) == 0)
            value += miArray[x].int3;
    }
    return value;
}

В части javascript я помещаю массив в ascii, а затем выделяю память в куче, используя функцию malloc, и передаю длину функции c, но файл wasm не прочитал это так, как я хотел.

let miArray = [
    {
        text1: "lorem ipsum ",
        text2: "lorem ipsum lorem ipsum",
        text3: "lorem ipsum PRUEBA",
        int1: 1,
        int2: 2,
        int3: 3
    }
];

// Init the typed array with the same length as the number of items in the array parameter
const typed_miArray = new Float32Array(miArray.length);
// Populate the array with the values
for (let i = 0; i < miArray.length; i++) {
  typed_miArray[i] = miArray[i];
}

//allocate memory on wasm heap for the array
const miArray_buffer = Module._malloc(typed_miArray.length * typed_miArray.BYTES_PER_ELEMENT);
Module.HEAPF32.set(typed_miArray, miArray_buffer >> 2);

const value = Module._calcArray(miArray_buffer, miArray_buffer.length);

Для этого теста я изменяю функцию c:

float EMSCRIPTEN_KEEPALIVE calcArray(struct struct_name miArray, int arraySize){
    int value = 0;
    for (int x = 0; x < arraySize; x ++) {
        if ( x == 2 ){
            //text3
            if (strcmp(miArray[x].text3, “PRUEBA”) == 0)
            value += miArray[x].int3;
        }
    }
    return value;
}

Я создал такие файлы .js и .wasm (некоторые файлы .c включают заголовки .h):

emcc data.c data2.c data3.c -O0 -s WASM=1 -s MODULARIZE=1 -s ERROR_ON_UNDEFINED_SYMBOLS=0 -s ASSERTIONS=1 -s EMBIND_STD_STRING_IS_UTF8=1 -s ALLOW_MEMORY_GROWTH=1 -s NO_FILESYSTEM=1 -s -o results.js`

Любая помощь приветствуется. Простите за неправильное форматирование (это мой первый пост здесь). Спасибо!


person Jorge GH    schedule 28.11.2019    source источник


Ответы (1)


Во-первых, вы не можете назначить объект типизированному массиву. Вы также, вероятно, не захотите использовать массив с плавающей запятой. Наивный способ кодирования строки JavaScript в типизированный массив:

const myString = "test";
// clamped array will convert anything above 255 to 255 instead of overflowing
// Note that we leave the last byte as zero
const myString_c_style = new Uint8ClampedArray(myString.length + 1);
for (let i = 0; i < myString.length; ++i) {
    myString_c_style[i] = myString.charCodeAt(i);
}

Но это не все. Вы также не должны пытаться разместить C struct самостоятельно в каком-либо типизированном массиве. Это принесет вам много головной боли в будущем. Вместо этого я рекомендую использовать функцию преобразования, например:

struct struct_name
{
  char text1[10];
  char text2[30];
  char text3[50];
  int int1;
  int int2;
  int int3;
};

float EMSCRIPTEN_KEEPALIVE calcArrayFromJS(const char* text1, const char* text2, const char* text3, int i1, int i2, int i3)
{
  struct_name tmp;
  // copy strings
  strncpy(&(tmp.text1[0]), text1, sizeof(tmp.text1) / sizeof(char));
  strncpy(&(tmp.text2[0]), text2, sizeof(tmp.text2) / sizeof(char));
  strncpy(&(tmp.text3[0]), text3, sizeof(tmp.text3) / sizeof(char));
  // Copy integers
  tmp.int1 = i1;
  tmp.int2 = i2;
  tmp.int3 = i3;
  return calcArray(tmp);
}

Я опустил тот факт, что вы используете массив struct_name, но он показывает общее направление. В качестве альтернативы подумайте, не предоставляет ли emscripten API для написания структур на JavaScript.

person Tomáš Zato - Reinstate Monica    schedule 20.12.2019