簡體   English   中英

Char *在JNI調用中損壞

[英]Char* Corrupted Across JNI Calls

我正在使用Android(特別是Google Glass)上的C庫(librtmp),並具有由SWIG創建的綁定。

當我使用包含char *的結構調用本機方法,從該方法返回並調用另一個本機方法時,char *中將沒有相同的數據,很多時候該數據似乎為null,但有時將包含Java或項目路徑的一部分或隨機文本(例如: glassvideostreamer/libr Zunk+Zunk+sun/* ),有時還會包含無效的unicode代碼點。 指針本身始終指向JNI調用之間的相同內存位置,並且在第一個本機方法退出時是正確的,而在它進入另一個本機方法時則已損壞。

將兩個方法都包裝在一個本機函數中,然后調用該函數可以正常工作,但是當調用另一個本機方法時,數據將不再位於內存位置。

這是我的代碼:

test.i

%module test_wrapper
#define NO_CRYPTO
/* Anything in the following section is added verbatim to the .cxx wrapper file*/
%{
#include "rtmp.h"
%}
test* Alloc_Test();
void Init_Test(test* t, char* data, int len);
void Test_String(test* t);
void Init_And_Test_String(test* t, char* data, int len);

rtmp.h

  typedef struct test {
    char* data;
    int len;
  } test;
  test* Alloc_Test();
  void Init_Test(test* t, char* data, int len);
  void Test_String(test* t);
  void Init_And_Test_String(test* t, char* data, int len);

rtmp.c

test* Alloc_Test(){
  return calloc(sizeof(test));
}
void Init_Test(test* t, char* data, int len){
  t->data = data;
  t->len = len;
  __android_log_print(ANDROID_LOG_DEBUG,"TEST", "String: %.*s  (pointer: %p)", t->len, t->data, t);
}
void Test_String(test* t){
 __android_log_print(ANDROID_LOG_DEBUG,"TEST","%String: %.*s  (pointer: %p)", t->len, t->data, t);
}
void Init_And_Test_String(test* t, char* data, int len){
  Init_Test(t, data, len);
  Test_String(t);
}

最終(對於c)這是生成的swig綁定librtmp_wrap.c

SWIGEXPORT jlong JNICALL Java_atellis_glassvideostreamer_librtmp_test_1wrapperJNI_Alloc_1Test(JNIEnv *jenv, jclass jcls) {
  jlong jresult = 0 ;
  test *result = 0 ;

  (void)jenv;
  (void)jcls;
  result = (test *)Alloc_Test();
  *(test **)&jresult = result; 
  return jresult;
}


SWIGEXPORT void JNICALL Java_atellis_glassvideostreamer_librtmp_test_1wrapperJNI_Init_1Test(JNIEnv *jenv, jclass jcls, jlong jarg1, jstring jarg2, jint jarg3) {
  test *arg1 = (test *) 0 ;
  char *arg2 = (char *) 0 ;
  int arg3 ;

  (void)jenv;
  (void)jcls;
  arg1 = *(test **)&jarg1; 
  arg2 = 0;
  if (jarg2) {
    arg2 = (char *)(*jenv)->GetStringUTFChars(jenv, jarg2, 0);
    if (!arg2) return ;
  }
  arg3 = (int)jarg3; 
  Init_Test(arg1,arg2,arg3);
  if (arg2) (*jenv)->ReleaseStringUTFChars(jenv, jarg2, (const char *)arg2);
}


SWIGEXPORT void JNICALL Java_atellis_glassvideostreamer_librtmp_test_1wrapperJNI_Test_1String(JNIEnv *jenv, jclass jcls, jlong jarg1) {
  test *arg1 = (test *) 0 ;

  (void)jenv;
  (void)jcls;
  arg1 = *(test **)&jarg1; 
  Test_String(arg1);
}


SWIGEXPORT void JNICALL Java_atellis_glassvideostreamer_librtmp_test_1wrapperJNI_Init_1And_1Test_1String(JNIEnv *jenv, jclass jcls, jlong jarg1, jstring jarg2, jint jarg3) {
  test *arg1 = (test *) 0 ;
  char *arg2 = (char *) 0 ;
  int arg3 ;

  (void)jenv;
  (void)jcls;
  arg1 = *(test **)&jarg1; 
  arg2 = 0;
  if (jarg2) {
    arg2 = (char *)(*jenv)->GetStringUTFChars(jenv, jarg2, 0);
    if (!arg2) return ;
  }
  arg3 = (int)jarg3; 
  Init_And_Test_String(arg1,arg2,arg3);
  if (arg2) (*jenv)->ReleaseStringUTFChars(jenv, jarg2, (const char *)arg2);
}

在Java方面:我們只有生成的SWIGTYPE_p測試,該測試包裝了結構指針和調用JNI函數的代碼: test_wrapperJNI.java

public class test_wrapperJNI {
  public final static native long Alloc_Test();
  public final static native void Init_Test(long jarg1, String jarg2, int jarg3);
  public final static native void Test_String(long jarg1);
  public final static native void Init_And_Test_String(long jarg1, String jarg2, int jarg3);
}

test_wrapper.java

public class test_wrapper {
  public static SWIGTYPE_p_test Alloc_Test() {
    long cPtr = test_wrapperJNI.Alloc_Test();
    return (cPtr == 0) ? null : new SWIGTYPE_p_test(cPtr, false);
  }

  public static void Init_Test(SWIGTYPE_p_test t, String data, int len) {
    test_wrapperJNI.Init_Test(SWIGTYPE_p_test.getCPtr(t), data, len);
  }

  public static void Test_String(SWIGTYPE_p_test t) {
    test_wrapperJNI.Test_String(SWIGTYPE_p_test.getCPtr(t));
  }

  public static void Init_And_Test_String(SWIGTYPE_p_test t, String data, int len) {
    test_wrapperJNI.Init_And_Test_String(SWIGTYPE_p_test.getCPtr(t), data, len);
  }

}

將它們放在一起,您將得到:

System.loadLibrary("TEST");
SWIGTYPE_p_test test = test_wrapper.Alloc_Test();
test_wrapper.Init_Test(test, "Will this string stay the same?", 31);
test_wrapper.Test_String(test);

test = test_wrapper.Alloc_Test();
test_wrapper.Init_And_Test_String(test, "This string will stay the same", 30);

得到這個結果(在logcat中):

05-11 12:27:33.198    4568-4568/atellis.glassvideostreamer D/dalvikvm﹕ Trying to load lib /data/app-lib/atellis.glassvideostreamer-1/libTEST.so 0x41b597b0
05-11 12:27:33.198    4568-4568/atellis.glassvideostreamer D/dalvikvm﹕ Added shared lib /data/app-lib/atellis.glassvideostreamer-1/libTEST.so 0x41b597b0
05-11 12:27:33.198    4568-4568/atellis.glassvideostreamer D/dalvikvm﹕ No JNI_OnLoad found in /data/app-lib/atellis.glassvideostreamer-1/libTEST.so 0x41b597b0, skipping init
05-11 12:27:38.128    4568-4568/atellis.glassvideostreamer D/TEST﹕ String: Will this string stay the same?  (pointer: 0x59564088)
05-11 12:27:39.940    4568-4568/atellis.glassvideostreamer D/TEST﹕ String:   (pointer: 0x59564088)
05-11 12:27:41.261    4568-4568/atellis.glassvideostreamer D/TEST﹕ String: This string will stay the same  (pointer: 0x59565948)
05-11 12:27:41.261    4568-4568/atellis.glassvideostreamer D/TEST﹕ String: This string will stay the same  (pointer: 0x59565948)

在這種情況下,字符串將變為空白! 如前所述,它有時會包含文件路徑或隨機文本。

typedef struct test {
    char* data;
    int len;
} test;

這沒有指定用於實際存儲字符串data空間,而僅指定了用於保存其地址的指針。

在實踐中,您似乎將其指向可能臨時保留的各種函數參數的存儲器,但這將不起作用,因為以后可以重新使用該存儲器。

void Init_Test(test* t, char* data, int len){
    t->data = data;

實際上,您已明確將您的聲明釋放到該內存中:

Init_And_Test_String(arg1,arg2,arg3); 
if (arg2) (*jenv)->ReleaseStringUTFChars(jenv, jarg2, (const char *)arg2);

您必須顯式保留內存以容納字符串,然后從任何臨時存儲復制到那里。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM