大家都曉得KVM是不支持Native函數調用的,如果要增加一些系統調用的API,那么只能加到KVM內部。同時,不同的J2ME設備,也有不同的系統調用API以及它們的實現。我們從SUN那么下載到原始的KVM源代碼,如何為其增加一個系統調用API呢?本文以具體實踐的步驟一步一步來講解增加KVM系統API的方法。
其實為KVM增加一個系統調用API比為linux增加一個系統調用API還簡單。大致就分成兩步工作就完成。一步是在classes.zip中增加一個你自己新增的class,一步是在KVM的nativeCore.c中實現這個新增的class的native api函數。
下面以org.test.MiniTest這個新增的class為例來實現一個TestInt()系統調用函數。函數的功能很簡單,就是返回一個整數9999。
從SUN那里下載到j2me_cldc 1.1版本的KVM代碼后。在j2me_cldc/api目錄下,增加org/test/MiniTest.java的包目錄以及java文件。然后寫上如下的代碼:
package org.test;
public class MiniTest
{
public static native int TestInt();
}
根據上一篇文章中的KVM編譯方法,在命令提示符下,跳到目錄j2me_cldc/build/win32下,輸入make命令進行第一次整體編譯。不過,這次編譯過程在編譯連接KVM中的*.o文件的時候,會提示一個找到_Java_org_test_MiniTest_TestInt符號的錯誤提示。
這是因為在我們只是在org.test.MiniTest中定義了這個native函數TestInt,但是并沒有在KVM的任何一個c文件中實現其對應的函數。
首先編譯過程是用javac來編譯j2me_cldc/api里面的所有的*.java文件,然后將其class文件打包成一個classes.zip,然后JCC這個工具會默認根據classes.zip生成ROMJavaWin.c和nativeFunctionTableWin.c。而在ROMJavaWin.c聲明這個native函數:
extern void Java_org_test_MiniTest_TestInt(void);
從KVM中的代碼可以看到,KVM默認都是把一些native函數放到了nativeCore.c這個文件里面。你也可以自己去新增一些C程序文件,不過本例就把這個Java_org_test_MiniTest_TestInt放在了nativeCore.c文件。
下面是代碼:
void Java_org_test_MiniTest_TestInt(void)
{
pushStack(9999);
}
這里為什么把返回值使用pushStack這個宏來返回的原因就不好說了,關于JAVA運行的方式其實就是一個堆棧,Java的字節碼其實就是一種棧式語言,這個在編譯原理里面的中間代碼生成那一章可以找到它的原型和其說明。再者,還可以看《Inside Java Virtual Machine》這本書。
第二次編譯就是可以生成真正的kvm.exe文件了。還是跟第一次編譯以及上一篇編譯KVM的方法一樣,敲入make命令即可。
自己寫了一個Test的類,來測試這個API:
import org.test.*;
class Test {
public static void main(String[] args) {
System.out.
(出處:http://www.survivalescaperooms.com)
新聞熱點
疑難解答