最近在将 HevSocks5Client 移植到 Android 上了,在经过增加 signalfd 和 timerfd 相关的系统调用支持后,就可以直接使用 NDK 编译出 executable 了。直接的 native exectuable 在 Android 系统总还是不太方便用哦。还是做成一个 apk 吧,暂定只写一个 service 并开机自动启用,无 activity 的。
Java 中调用 native 程序我选择使用 JNI 方式,直接在 JNI_OnLoad 方法中调用 pthread_create 创建个线程跑原来的 main 就行啦。
...
#if defined(ANDROID)
#include <jni.h>
#include <pthread.h>
#endif
int
main (int argc, char *argv[])
{
...
}
#if defined(ANDROID)
static void *
thread_handler (void *data)
{
main (0, NULL);
return NULL;
}
jint
JNI_OnLoad (JavaVM *vm, void *reserved)
{
pthread_t thread;
pthread_create (&thread, NULL, thread_handler, NULL);
return JNI_VERSION_1_4;
}
#endif
Android 服务
服务主要是加载 JNI 接口的 hev-socks5-client 库,使服务跑起来。
package hev.socks5;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.util.Log;
public class MainService extends Service {
static {
System.loadLibrary("hev-socks5-client");
}
public IBinder onBind(Intent intent) {
return null;
}
}
BroadcastReceiver
ServiceReceiver 的功能就是监听系统上的 BOOT_COMPLETED 事件,用于实现自动启动服务。
package hev.socks5;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
public class ServiceReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(Intent.ACTION_BOOT_COMPLETED)) {
Intent i = new Intent(context, MainService.class);
context.startService(i);
}
}
}
AndroidManifest.xml
最后,要在 Manifest 中注册 Service 和 Receiver,增加上访问 Internet 和 Boot completed 的权限。
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="hev.socks5"
android:versionCode="1"
android:versionName="1.0">
<application android:label="@string/app_name" >
<service android:name=".MainService">
<intent-filter>
<action android:name="hev.socks5.MainService" />
</intent-filter>
</service>
<receiver android:enabled="true" android:name=".ServiceReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
</receiver>
</application>
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
</manifest>
Tips
此方法仅在 Android 2.3.3 及之前版本有效,之后版本如果此应用在安装后从没运行过,Broadcast Receiver 将接收不到 boot completed 的 action,解决方法是使用命令手动启动一下 service。