Обертывание функции с двумя двунаправленными параметрами с помощью SWIG для использования в java [foo(char ** strs, unsigned int

#java #c #c #java-native-interface #swig

#java #c #c #java-native-interface #swig

Вопрос:

Я использую SWIG 4.0.2, чтобы обернуть библиотеку C для использования из кода Java. У одного из классов есть сложная функция, которую мне сложно обернуть:

 class Bar {
...
public:
    Status foo(char **strs, unsigned intamp; size);
    ...
}
 

Способ использования «foo» заключается в выделении массива строк и вызове функции, присваивающей ей массив и количество элементов в нем. Функция заполнит массив строками, заканчивающимися нулем, и изменит параметр «size», чтобы он соответствовал количеству заполненных элементов (если он записал меньше максимального количества элементов).

Пример использования:

 #define MAX_STR_LENGTH 16 // Max string length is known

unsigned int numElements = 5;
char** strs= new char*[numElements];
for (unsigned int i=0; i<numElements;   i) {
    strs[i] = new char[MAX_STR_LENGTH];
}

Status status = bar.foo(strs, numElements);

if (status == Success) {
    for (unsigned int i=0; i<numElements;   i) {
        std::cout << strs[i] << std::endl;
    }
}
 

Я гибок в отношении того, как будет выглядеть сигнатура функции Java, но мне нужно иметь возможность каким-то образом извлекать строки после вызова foo() .

P.S: «Bar» на самом деле большой класс, который до сих пор SWIG отлично справлялся со мной, поэтому мне бы не хотелось вручную переносить его в код JNI.

Ответ №1:

Я, наконец, нашел решение этой проблемы. Полное решение на самом деле представляет собой смесь кода, взятого из собственного файла various.i SWIG, с небольшой настройкой для обработки параметра «char ** strs» и применения существующей карты типов для другого параметра.

Вот соответствующий код из файла .i:

 %typemap(jni) char **STRING_IN_OUT "jobjectArray"
%typemap(jtype) char **STRING_IN_OUT "String[]"
%typemap(jstype) char **STRING_IN_OUT "String[]"
%typemap(in) char **STRING_IN_OUT (jint size) {
  int i = 0;
  if ($input) {
    size = JCALL1(GetArrayLength, jenv, $input);
#ifdef __cplusplus
    $1 = new char*[size 1];
#else
    $1 = (char **)malloc((size 1) * sizeof(char *));
#endif
    for (i = 0; i<size; i  ) {
      jstring j_string = (jstring)JCALL2(GetObjectArrayElement, jenv, $input, i);
      const char *c_string = JCALL2(GetStringUTFChars, jenv, j_string, 0);
#ifdef __cplusplus
      $1[i] = new char [strlen(c_string) 1];
#else
      $1[i] = (char *)malloc((strlen(c_string) 1) * sizeof(const char *));
#endif
      strcpy($1[i], c_string);
      JCALL2(ReleaseStringUTFChars, jenv, j_string, c_string);
      JCALL1(DeleteLocalRef, jenv, j_string);
    }
    $1[i] = 0;
  } else {
    $1 = 0;
    size = 0;
  }
}

%typemap(argout) char **STRING_IN_OUT {
  for (int i=0; i< (int) size$argnum; i  ) {
    jstring jnewstring = NULL;
    jnewstring = JCALL1(NewStringUTF, jenv, $1[i]);
    JCALL3(SetObjectArrayElement, jenv, $input, i, jnewstring); 
  }
}

%typemap(freearg) char **STRING_IN_OUT {
  int i;
  for (i=0; i<size$argnum; i  )
#ifdef __cplusplus
    delete[] $1[i];
  delete[] $1;
#else
  free($1[i]);
  free($1);
#endif
}

%typemap(out) char **STRING_IN_OUT {
  if ($1) {
    int i;
    jsize len=0;
    jstring temp_string;
    const jclass clazz = JCALL1(FindClass, jenv, "java/lang/String");

    while ($1[len]) len  ;
    $result = JCALL3(NewObjectArray, jenv, len, clazz, NULL);
    /* exception checking omitted */

    for (i=0; i<len; i  ) {
      temp_string = JCALL1(NewStringUTF, jenv, *$1  );
      JCALL3(SetObjectArrayElement, jenv, $result, i, temp_string);
      JCALL1(DeleteLocalRef, jenv, temp_string);
    }
  }
}

%typemap(javain) char **STRING_IN_OUT "$javainput"
%typemap(javaout) char **STRING_IN_OUT {
    return $jnicall;
  }

%apply char **STRING_IN_OUT { char** strs}

%include "typemaps.i"
%apply unsigned intamp; INOUT { unsigned intamp; size}
 

И вот как я вызываю это из Java:

 // For now, this is how I tell the JNI how many buffers to allocate and their length
// I Will probably look for a way to work around this useless allocation in the future.
String[] strs = new String[MAX_NUMBER_STRINGS];
for (int i = 0 ; i < strs.length ;   i) {
  strs[i] = new String(new char[BUFFER_LENGTH]);
}

long[] numOfIds = new long[]{MAX_NUMBER_STRINGS};
Status s = m_faceAuthenticator.QueryUserIds(strs, numOfIds);
 

Мне потребовалось много времени, чтобы решить эту проблему.