Java本机访问代码错误:"内存访问无效"

时间:2017-07-28 22:54:24

标签: java c++ c pointers jna

我可以访问test.h和test.so文件:

test.h

int insert_data(void *location_data, uint64_t device_id, uint64_t event_id,
                uint64_t motion_seq, uint64_t utc_time, uint32_t sensor_seq,
                uint32_t device_seq, uint32_t tick_mark, uint16_t ble,
                int movement, int path_id);

我现在正在编写JNA代码来调用上面的方法。

interface TestCLibrary extends Library {
    TestCLibrary INSTANCE = (TestCLibrary) Native.loadLibrary(“test”, TestCLibrary);

    int insert_data(Pointer test_data, long device_id, long event_id,
                    long motion_seq, long utc_time, int sensor_seq,
                    int device_seq, int tick_mark, short ble,
                    int movement, int path_id);
 }
public class JNATest {
    public static void main(String[] args) throws IOException{ 
        Native.setProtected(true); 
        int path_id = 1, movement = 1, tick_mark = 1, device_seq = 1;
        long utc_time = System.currentTimeMillis();
        int sensor_seq = 100;
        short ble = 25;
        long motion_seq = 22222299;
        long device_id = 111122333;
        long event_id = 44445555;  

        Pointer test_data = Pointer.NULL; 

        TestCLibrary.INSTANCE.insert_data(test_data, device_id, event_id, motion_seq, utc_time, sensor_seq, device_seq, tick_mark, ble, movement, path_id);
   }
}

在此代码位置运行JNATest时面临以下异常:

  
    

TestCLibrary.INSTANCE.insert_data

  
     

线程中的异常" main" java.lang.Error:无效的内存访问   com.sun.jna.Native.invokeInt(Native Method)at   com.sun.jna.Function.invoke(Function.java:419)at   com.sun.jna.Function.invoke(Function.java:354)at   com.sun.jna.Library $ Handler.invoke(Library.java:244)at   com.test.apps。$ Proxy0.insert_data(未知来源)....

如何避免此类错误? 以及如何使用默认值初始化JNA中的指针?

2 个答案:

答案 0 :(得分:0)

错误正在发生,因为我假设该函数将数据写入传递给它的指针所指向的内存中。您需要分配函数可以写入的内存。它无法写入NULL指针。

应该做什么取决于数据是什么,如何构建以及如何使用它。

如果数据只是一个整数或字节数组,请将其视为(相应地更改数组大小):

int insert_data(int[] data, // ... etc.
// ...

int[] data = new int[4096];
insert_data(data, /* ... */

但是,如果您知道该函数生成的数据是structclass,请为其创建Java class。 JNA将自动将由本机函数写入的数据“编组”到有意义的Java类实例中:

由于函数接受指针,因此您需要使用Structure.ByReference

int insert_data(Structure.ByReference data, // ...

答案 1 :(得分:0)

执行以下步骤后,JVM崩溃问题得以解决:

1)Native.setProtected(true);

2)使用Native库释放内存 Native.free(Pointer.nativeValue(my_pointer_by_ref.getValue()));

3)使用以下CLI选项运行程序 - -XX:-UseLoopPredicate -DLD_PRELOAD = / usr / lib / jvm / java-8-oracle / jre / lib / amd64 / libjsig.so