android: Register NetworkManager as BroadcastReceiver and relay events via JNI

This commit is contained in:
Tobias Brunner 2012-10-10 14:14:30 +02:00
parent 38bbca587f
commit ef3d1a1ba9
5 changed files with 184 additions and 8 deletions

View File

@ -23,6 +23,7 @@
<uses-sdk android:minSdkVersion="14" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<application
android:icon="@drawable/ic_launcher"

View File

@ -414,7 +414,7 @@ static void charonservice_init(JNIEnv *env, jobject service, jobject builder)
.attr = android_attr_create(),
.creds = android_creds_create(),
.builder = vpnservice_builder_create(builder),
.network_manager = network_manager_create(),
.network_manager = network_manager_create(service),
.vpn_service = (*env)->NewGlobalRef(env, service),
);
charonservice = &this->public;

View File

@ -15,7 +15,9 @@
#include "network_manager.h"
#include "../android_jni.h"
#include "../charonservice.h"
#include <debug.h>
#include <threading/mutex.h>
typedef struct private_network_manager_t private_network_manager_t;
@ -35,6 +37,19 @@ struct private_network_manager_t {
* Java class for NetworkManager
*/
jclass cls;
/**
* Registered callback
*/
struct {
connectivity_cb_t cb;
void *data;
} connectivity_cb;
/**
* Mutex to access callback
*/
mutex_t *mutex;
};
METHOD(network_manager_t, get_local_address, host_t*,
@ -70,11 +85,99 @@ failed:
return NULL;
}
JNI_METHOD(NetworkManager, networkChanged, void,
bool disconnected)
{
private_network_manager_t *nm;
nm = (private_network_manager_t*)charonservice->get_network_manager(
charonservice);
nm->mutex->lock(nm->mutex);
if (nm->connectivity_cb.cb)
{
nm->connectivity_cb.cb(nm->connectivity_cb.data, disconnected);
}
nm->mutex->unlock(nm->mutex);
}
METHOD(network_manager_t, add_connectivity_cb, void,
private_network_manager_t *this, connectivity_cb_t cb, void *data)
{
this->mutex->lock(this->mutex);
if (!this->connectivity_cb.cb)
{
JNIEnv *env;
jmethodID method_id;
androidjni_attach_thread(&env);
method_id = (*env)->GetMethodID(env, this->cls, "Register", "()V");
if (!method_id)
{
androidjni_exception_occurred(env);
}
else
{
(*env)->CallVoidMethod(env, this->obj, method_id);
if (!androidjni_exception_occurred(env))
{
this->connectivity_cb.cb = cb;
this->connectivity_cb.data = data;
}
androidjni_detach_thread();
}
}
this->mutex->unlock(this->mutex);
}
/**
* Unregister the NetworkManager via JNI.
*
* this->mutex has to be locked
*/
static void unregister_network_manager(private_network_manager_t *this)
{
JNIEnv *env;
jmethodID method_id;
androidjni_attach_thread(&env);
method_id = (*env)->GetMethodID(env, this->cls, "Unregister", "()V");
if (!method_id)
{
androidjni_exception_occurred(env);
}
else
{
(*env)->CallVoidMethod(env, this->obj, method_id);
androidjni_exception_occurred(env);
}
androidjni_detach_thread();
}
METHOD(network_manager_t, remove_connectivity_cb, void,
private_network_manager_t *this, connectivity_cb_t cb)
{
this->mutex->lock(this->mutex);
if (this->connectivity_cb.cb == cb)
{
this->connectivity_cb.cb = NULL;
unregister_network_manager(this);
}
this->mutex->unlock(this->mutex);
}
METHOD(network_manager_t, destroy, void,
private_network_manager_t *this)
{
JNIEnv *env;
this->mutex->lock(this->mutex);
if (this->connectivity_cb.cb)
{
this->connectivity_cb.cb = NULL;
unregister_network_manager(this);
}
this->mutex->unlock(this->mutex);
androidjni_attach_thread(&env);
if (this->obj)
{
@ -85,13 +188,14 @@ METHOD(network_manager_t, destroy, void,
(*env)->DeleteGlobalRef(env, this->cls);
}
androidjni_detach_thread();
this->mutex->destroy(this->mutex);
free(this);
}
/*
* Described in header.
*/
network_manager_t *network_manager_create()
network_manager_t *network_manager_create(jobject context)
{
private_network_manager_t *this;
JNIEnv *env;
@ -102,8 +206,11 @@ network_manager_t *network_manager_create()
INIT(this,
.public = {
.get_local_address = _get_local_address,
.add_connectivity_cb = _add_connectivity_cb,
.remove_connectivity_cb = _remove_connectivity_cb,
.destroy = _destroy,
},
.mutex = mutex_create(MUTEX_TYPE_DEFAULT),
);
androidjni_attach_thread(&env);
@ -114,12 +221,12 @@ network_manager_t *network_manager_create()
}
this->cls = (*env)->NewGlobalRef(env, cls);
method_id = (*env)->GetMethodID(env, cls, "<init>",
"()V");
"(Landroid/content/Context;)V");
if (!method_id)
{
goto failed;
}
obj = (*env)->NewObject(env, cls, method_id);
obj = (*env)->NewObject(env, cls, method_id, context);
if (!obj)
{
goto failed;

View File

@ -29,7 +29,19 @@
typedef struct network_manager_t network_manager_t;
/**
* NetworkManager, used to retrieve local IP addresses.
* Callback called if connectivity changes somehow.
*
* Implementation should be quick as the call is made by the Java apps main
* thread.
*
* @param data data supplied during registration
* @param disconnected TRUE if currently disconnected
*/
typedef void (*connectivity_cb_t)(void *data, bool disconnected);
/**
* NetworkManager, used to listen for network changes and retrieve local IP
* addresses.
*
* Communicates with NetworkManager via JNI
*/
@ -43,6 +55,25 @@ struct network_manager_t {
*/
host_t *(*get_local_address)(network_manager_t *this, bool ipv4);
/**
* Register a callback that is called if connectivity changes
*
* @note Only the first registered callback is currently used
*
* @param cb callback to register
* @param data data provided to callback
*/
void (*add_connectivity_cb)(network_manager_t *this, connectivity_cb_t cb,
void *data);
/**
* Unregister a previously registered callback for connectivity changes
*
* @param cb previously registered callback
*/
void (*remove_connectivity_cb)(network_manager_t *this,
connectivity_cb_t cb);
/**
* Destroy a network_manager_t instance
*/
@ -52,8 +83,9 @@ struct network_manager_t {
/**
* Create a network_manager_t instance
*
* @param context Context object
* @return network_manager_t instance
*/
network_manager_t *network_manager_create();
network_manager_t *network_manager_create(jobject context);
#endif /** NETWORK_MANAGER_H_ @}*/

View File

@ -22,12 +22,48 @@ import java.net.NetworkInterface;
import java.net.SocketException;
import java.util.Enumeration;
public class NetworkManager
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
public class NetworkManager extends BroadcastReceiver
{
public NetworkManager()
private final Context mContext;
private boolean mRegistered;
public NetworkManager(Context context)
{
mContext = context;
}
public void Register()
{
mContext.registerReceiver(this, new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION));
}
public void Unregister()
{
mContext.unregisterReceiver(this);
}
@Override
public void onReceive(Context context, Intent intent)
{
ConnectivityManager cm = (ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo info = cm.getActiveNetworkInfo();
networkChanged(info == null || !info.isConnected());
}
/**
* Notify the native parts about a network change
*
* @param disconnected true if no connection is available at the moment
*/
public native void networkChanged(boolean disconnected);
/**
* Function that retrieves a local address of the given family.
*