aboutsummaryrefslogtreecommitdiffstats
path: root/src/frontends/android/jni/libandroidbridge
diff options
context:
space:
mode:
Diffstat (limited to 'src/frontends/android/jni/libandroidbridge')
-rw-r--r--src/frontends/android/jni/libandroidbridge/charonservice.c2
-rw-r--r--src/frontends/android/jni/libandroidbridge/kernel/network_manager.c113
-rw-r--r--src/frontends/android/jni/libandroidbridge/kernel/network_manager.h36
3 files changed, 145 insertions, 6 deletions
diff --git a/src/frontends/android/jni/libandroidbridge/charonservice.c b/src/frontends/android/jni/libandroidbridge/charonservice.c
index 96a4da8f3..591ca90ec 100644
--- a/src/frontends/android/jni/libandroidbridge/charonservice.c
+++ b/src/frontends/android/jni/libandroidbridge/charonservice.c
@@ -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;
diff --git a/src/frontends/android/jni/libandroidbridge/kernel/network_manager.c b/src/frontends/android/jni/libandroidbridge/kernel/network_manager.c
index e7bcf1365..9c97fbb14 100644
--- a/src/frontends/android/jni/libandroidbridge/kernel/network_manager.c
+++ b/src/frontends/android/jni/libandroidbridge/kernel/network_manager.c
@@ -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;
diff --git a/src/frontends/android/jni/libandroidbridge/kernel/network_manager.h b/src/frontends/android/jni/libandroidbridge/kernel/network_manager.h
index cb0da7fa2..634816405 100644
--- a/src/frontends/android/jni/libandroidbridge/kernel/network_manager.h
+++ b/src/frontends/android/jni/libandroidbridge/kernel/network_manager.h
@@ -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
*/
@@ -44,6 +56,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
*/
void (*destroy)(network_manager_t *this);
@@ -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_ @}*/