Imported Upstream version 1.38
[platform/upstream/connman.git] / plugins / vpn.c
index 25711d7..5668c00 100644 (file)
@@ -38,6 +38,7 @@
 #include <connman/dbus.h>
 #include <connman/provider.h>
 #include <connman/ipaddress.h>
+#include <connman/notifier.h>
 #include <connman/vpn-dbus.h>
 #include <connman/inet.h>
 #include <gweb/gresolv.h>
@@ -71,8 +72,10 @@ struct connection_data {
        struct connman_provider *provider;
        int index;
        DBusPendingCall *call;
+       DBusPendingCall *disconnect_call;
        bool connect_pending;
        struct config_create_data *cb_data;
+       char *service_ident;
 
        char *state;
        char *type;
@@ -203,7 +206,7 @@ static void resolv_result(GResolvResultStatus status,
         * We cannot unref the resolver here as resolv struct is manipulated
         * by gresolv.c after we return from this callback.
         */
-       g_timeout_add_seconds(0, remove_resolv, data);
+       g_idle_add(remove_resolv, data);
 
        data->resolv_id = 0;
 }
@@ -249,6 +252,12 @@ static void free_config_cb_data(struct config_create_data *cb_data)
        g_free(cb_data);
 }
 
+static bool provider_is_connected(struct connection_data *data)
+{
+       return data && (g_str_equal(data->state, "ready") ||
+                       g_str_equal(data->state, "configuration"));
+}
+
 static void set_provider_state(struct connection_data *data)
 {
        enum connman_provider_state state = CONNMAN_PROVIDER_STATE_UNKNOWN;
@@ -256,6 +265,11 @@ static void set_provider_state(struct connection_data *data)
 
        DBG("provider %p new state %s", data->provider, data->state);
 
+       if (!provider_is_connected(data)) {
+               g_free(data->service_ident);
+               data->service_ident = NULL;
+       }
+
        if (g_str_equal(data->state, "ready")) {
                state = CONNMAN_PROVIDER_STATE_READY;
                goto set;
@@ -463,6 +477,9 @@ static int errorstr2val(const char *error) {
        if (g_strcmp0(error, CONNMAN_ERROR_INTERFACE ".AlreadyConnected") == 0)
                return -EISCONN;
 
+       if (g_strcmp0(error, CONNMAN_ERROR_INTERFACE ".OperationCanceled") == 0)
+               return -ECANCELED;
+
        return -ECONNREFUSED;
 }
 
@@ -476,27 +493,45 @@ static void connect_reply(DBusPendingCall *call, void *user_data)
        if (!dbus_pending_call_get_completed(call))
                return;
 
+       if (call != data->call) {
+               connman_error("invalid call %p to VPN connect_reply data %p "
+                                       " call %p ", call, data, data->call);
+               dbus_pending_call_unref(call);
+               return;
+       }
+
        DBG("user_data %p path %s", user_data, cb_data ? cb_data->path : NULL);
 
        reply = dbus_pending_call_steal_reply(call);
+       if (!reply)
+               goto out;
 
        dbus_error_init(&error);
 
        if (dbus_set_error_from_message(&error, reply)) {
                int err = errorstr2val(error.name);
-               if (err != -EINPROGRESS) {
+
+               /*
+                * ECANCELED means that user has canceled authentication
+                * dialog. That's not really an error, it's part of a normal
+                * workflow. We also take it as a request to turn autoconnect
+                * off, in case if it was on.
+                */
+               if (err == -ECANCELED) {
+                       DBG("%s connect canceled", data->path);
+                       connman_provider_set_autoconnect(data->provider, false);
+               } else if (err != -EINPROGRESS) {
                        connman_error("Connect reply: %s (%s)", error.message,
                                                                error.name);
-                       dbus_error_free(&error);
-
                        DBG("data %p cb_data %p", data, cb_data);
+
                        if (cb_data) {
                                cb_data->callback(cb_data->message, err, NULL);
                                free_config_cb_data(cb_data);
                                data->cb_data = NULL;
                        }
-                       goto done;
                }
+
                dbus_error_free(&error);
        }
 
@@ -506,32 +541,59 @@ static void connect_reply(DBusPendingCall *call, void *user_data)
         * state.
         */
 
-done:
        dbus_message_unref(reply);
 
-       dbus_pending_call_unref(call);
+out:
+       dbus_pending_call_unref(data->call);
+
+       data->call = NULL;
+       data->connect_pending = false;
 }
 
-static int connect_provider(struct connection_data *data, void *user_data)
+static int connect_provider(struct connection_data *data, void *user_data,
+                       const char *dbus_sender)
 {
        DBusPendingCall *call;
        DBusMessage *message;
        struct config_create_data *cb_data = user_data;
+       struct connman_service *transport = connman_service_get_default();
 
-       DBG("data %p user %p path %s", data, cb_data, data->path);
+       DBG("data %p user %p path %s sender %s", data, cb_data, data->path,
+                                                               dbus_sender);
 
-       data->connect_pending = false;
+       if (!transport) {
+               DBG("no default service, refusing to connect");
+               return -EINVAL;
+       }
 
+       if (data->connect_pending && data->call) {
+               connman_info("connect already pending");
+               return -EALREADY;
+       }
+
+       /* We need to pass original dbus sender to connman-vpnd,
+        * use a Connect2 method for that if the original dbus sender is set.
+        * Connect method requires no parameter, Connect2 requires dbus sender
+        * name to be set.
+        */
        message = dbus_message_new_method_call(VPN_SERVICE, data->path,
                                        VPN_CONNECTION_INTERFACE,
-                                       VPN_CONNECT);
+                                       dbus_sender && *dbus_sender ?
+                                               VPN_CONNECT2 : VPN_CONNECT);
        if (!message)
                return -ENOMEM;
 
+       if (dbus_sender && *dbus_sender)
+               dbus_message_append_args(message, DBUS_TYPE_STRING,
+                                       &dbus_sender, NULL);
+       else
+               dbus_sender = "";
+
        if (!dbus_connection_send_with_reply(connection, message,
                                                &call, DBUS_TIMEOUT)) {
                connman_error("Unable to call %s.%s()",
-                       VPN_CONNECTION_INTERFACE, VPN_CONNECT);
+                       VPN_CONNECTION_INTERFACE, dbus_sender && *dbus_sender ?
+                                               VPN_CONNECT2 : VPN_CONNECT);
                dbus_message_unref(message);
                return -EINVAL;
        }
@@ -541,11 +603,28 @@ static int connect_provider(struct connection_data *data, void *user_data)
                return -EINVAL;
        }
 
+       if (data->call) {
+               dbus_pending_call_cancel(data->call);
+               dbus_pending_call_unref(data->call);
+       }
+
+       data->call = call;
+       data->connect_pending = true;
+
        if (cb_data) {
                g_free(cb_data->path);
                cb_data->path = g_strdup(data->path);
        }
 
+       /*
+        * This is the service which (most likely) will be used
+        * as a transport for VPN connection.
+        */
+       g_free(data->service_ident);
+       data->service_ident =
+               g_strdup(connman_service_get_identifier(transport));
+       DBG("transport %s", data->service_ident);
+
        dbus_pending_call_set_notify(call, connect_reply, data, NULL);
 
        dbus_message_unref(message);
@@ -658,8 +737,15 @@ static void add_connection(const char *path, DBusMessageIter *properties,
                connman_provider_set_domain(data->provider,
                                                data->domain);
 
-       if (data->connect_pending)
-               connect_provider(data, data->cb_data);
+       if (data->connect_pending) {
+               const char *dbus_sender = NULL;
+
+               if (data->cb_data && data->cb_data->message) {
+                       dbus_sender =
+                               dbus_message_get_sender(data->cb_data->message);
+               }
+               connect_provider(data, data->cb_data, dbus_sender);
+       }
 
        return;
 
@@ -857,7 +943,8 @@ static int provider_remove(struct connman_provider *provider)
        return 0;
 }
 
-static int provider_connect(struct connman_provider *provider)
+static int provider_connect(struct connman_provider *provider,
+                                       const char *dbus_sender)
 {
        struct connection_data *data;
 
@@ -865,17 +952,15 @@ static int provider_connect(struct connman_provider *provider)
        if (!data)
                return -EINVAL;
 
-       return connect_provider(data, NULL);
+       return connect_provider(data, NULL, dbus_sender);
 }
 
 static void disconnect_reply(DBusPendingCall *call, void *user_data)
 {
+       struct connection_data *data = user_data;
        DBusMessage *reply;
        DBusError error;
 
-       if (!dbus_pending_call_get_completed(call))
-               return;
-
        DBG("user %p", user_data);
 
        reply = dbus_pending_call_steal_reply(call);
@@ -890,54 +975,53 @@ static void disconnect_reply(DBusPendingCall *call, void *user_data)
 
 done:
        dbus_message_unref(reply);
-
        dbus_pending_call_unref(call);
+       data->disconnect_call = NULL;
 }
 
 static int disconnect_provider(struct connection_data *data)
 {
-       DBusPendingCall *call;
+       bool sent;
        DBusMessage *message;
 
        DBG("data %p path %s", data, data->path);
 
+       if (data->disconnect_call) {
+               DBG("already disconnecting");
+               return -EINVAL;
+       }
+
        message = dbus_message_new_method_call(VPN_SERVICE, data->path,
                                        VPN_CONNECTION_INTERFACE,
                                        VPN_DISCONNECT);
        if (!message)
                return -ENOMEM;
 
-       if (!dbus_connection_send_with_reply(connection, message,
-                                               &call, DBUS_TIMEOUT)) {
+       sent = dbus_connection_send_with_reply(connection, message,
+                                       &data->disconnect_call, DBUS_TIMEOUT);
+       dbus_message_unref(message);
+
+       if (!sent || !data->disconnect_call) {
                connman_error("Unable to call %s.%s()",
                        VPN_CONNECTION_INTERFACE, VPN_DISCONNECT);
-               dbus_message_unref(message);
                return -EINVAL;
        }
 
-       if (!call) {
-               dbus_message_unref(message);
-               return -EINVAL;
-       }
+       dbus_pending_call_set_notify(data->disconnect_call, disconnect_reply,
+                                                               data, NULL);
 
-       dbus_pending_call_set_notify(call, disconnect_reply, NULL, NULL);
-
-       dbus_message_unref(message);
+       g_free(data->service_ident);
+       data->service_ident = NULL;
 
        connman_provider_set_state(data->provider,
                                        CONNMAN_PROVIDER_STATE_DISCONNECT);
-       /*
-        * We return 0 here instead of -EINPROGRESS because
-        * __connman_service_disconnect() needs to return something
-        * to gdbus so that gdbus will not call Disconnect() more
-        * than once. This way we do not need to pass the dbus reply
-        * message around the code.
-        */
-       return 0;
+       return -EINPROGRESS;
 }
 
 static int provider_disconnect(struct connman_provider *provider)
 {
+       int err = 0;
+
        struct connection_data *data;
 
        DBG("provider %p", provider);
@@ -946,11 +1030,18 @@ static int provider_disconnect(struct connman_provider *provider)
        if (!data)
                return -EINVAL;
 
-       if (g_str_equal(data->state, "ready") ||
-                       g_str_equal(data->state, "configuration"))
-               return disconnect_provider(data);
+       if (provider_is_connected(data))
+               err = disconnect_provider(data);
 
-       return 0;
+       if (data->call) {
+               dbus_pending_call_cancel(data->call);
+               dbus_pending_call_unref(data->call);
+               data->call = NULL;
+       }
+
+       data->connect_pending = false;
+
+       return err;
 }
 
 static void configuration_create_reply(DBusPendingCall *call, void *user_data)
@@ -1078,7 +1169,7 @@ static struct vpn_route *parse_user_route(const char *user_route)
                                char *ptr;
                                long int value = strtol(netmask, &ptr, 10);
                                if (ptr != netmask && *ptr == '\0' &&
-                                                               value <= 32)
+                                               value && value <= 32)
                                        prefix_len = value;
                        }
 
@@ -1455,17 +1546,11 @@ static void destroy_provider(struct connection_data *data)
 {
        DBG("data %p", data);
 
-       if (g_str_equal(data->state, "ready") ||
-                       g_str_equal(data->state, "configuration"))
+       if (provider_is_connected(data))
                connman_provider_disconnect(data->provider);
 
-       if (data->call)
-               dbus_pending_call_cancel(data->call);
-
        connman_provider_set_data(data->provider, NULL);
-
        connman_provider_remove(data->provider);
-
        data->provider = NULL;
 }
 
@@ -1478,13 +1563,24 @@ static void connection_destroy(gpointer hash_data)
        if (data->provider)
                destroy_provider(data);
 
+       if (data->call) {
+               dbus_pending_call_cancel(data->call);
+               dbus_pending_call_unref(data->call);
+       }
+
+       if (data->disconnect_call) {
+               dbus_pending_call_cancel(data->disconnect_call);
+               dbus_pending_call_unref(data->disconnect_call);
+       }
+
+       g_free(data->service_ident);
        g_free(data->path);
        g_free(data->ident);
        g_free(data->state);
        g_free(data->type);
        g_free(data->name);
        g_free(data->host);
-       g_free(data->host_ip);
+       g_strfreev(data->host_ip);
        g_free(data->domain);
        g_hash_table_destroy(data->server_routes);
        g_hash_table_destroy(data->user_routes);
@@ -1792,6 +1888,121 @@ static gboolean property_changed(DBusConnection *conn,
        return TRUE;
 }
 
+static int vpn_find_online_transport_cb(struct connman_service *service,
+                                                       void *user_data)
+{
+       if (connman_service_get_type(service) != CONNMAN_SERVICE_TYPE_VPN) {
+               switch (connman_service_get_state(service)) {
+               case CONNMAN_SERVICE_STATE_ONLINE:
+                       *((struct connman_service**)user_data) = service;
+                       return 1;
+               default:
+                       break;
+               }
+       }
+
+       return 0;
+}
+
+static struct connman_service *vpn_find_online_transport()
+{
+       struct connman_service *service = NULL;
+
+       connman_service_iterate_services(vpn_find_online_transport_cb,
+                                                               &service);
+       return service;
+}
+
+static bool vpn_is_valid_transport(struct connman_service *transport)
+{
+       if (transport) {
+               struct connman_service *online;
+
+               switch (connman_service_get_state(transport)) {
+               case CONNMAN_SERVICE_STATE_READY:
+                       online = vpn_find_online_transport();
+
+                       /* Stay connected if there are no online services */
+                       if (!online)
+                               return true;
+
+                       DBG("%s is ready, %s is online, disconnecting",
+                               connman_service_get_identifier(transport),
+                               connman_service_get_identifier(online));
+                       break;
+
+               case CONNMAN_SERVICE_STATE_ONLINE:
+                       online = vpn_find_online_transport();
+
+                       /* Check if our transport is still the default */
+                       if (online == transport)
+                               return true;
+
+                       DBG("%s is replaced by %s as default, disconnecting",
+                               connman_service_get_identifier(transport),
+                               connman_service_get_identifier(online));
+                       break;
+
+               default:
+                       break;
+               }
+       } else {
+               DBG("transport gone");
+       }
+
+       return false;
+}
+
+static void vpn_disconnect_check_provider(struct connection_data *data)
+{
+       if (provider_is_connected(data)) {
+               /* With NULL service ident NULL is returned immediately */
+               struct connman_service *service =
+                       connman_service_lookup_from_identifier
+                                               (data->service_ident);
+
+               if (!vpn_is_valid_transport(service)) {
+                       connman_provider_disconnect(data->provider);
+               }
+       }
+}
+
+static void vpn_disconnect_check()
+{
+       GHashTableIter iter;
+       gpointer value;
+
+       DBG("");
+       g_hash_table_iter_init(&iter, vpn_connections);
+       while (g_hash_table_iter_next(&iter, NULL, &value))
+               vpn_disconnect_check_provider(value);
+}
+
+static void vpn_service_add(struct connman_service *service, const char *name)
+{
+       vpn_disconnect_check();
+}
+
+static void vpn_service_list_changed(struct connman_service *service)
+{
+       vpn_disconnect_check();
+}
+
+static void vpn_service_state_changed(struct connman_service *service,
+                                       enum connman_service_state state)
+{
+       vpn_disconnect_check();
+}
+
+static const struct connman_notifier vpn_notifier = {
+       .name                   = "vpn",
+       .priority               = CONNMAN_NOTIFIER_PRIORITY_DEFAULT,
+       .default_changed        = vpn_service_list_changed,
+       .service_add            = vpn_service_add,
+       .service_remove         = vpn_service_list_changed,
+       .service_state_changed  = vpn_service_state_changed
+};
+
 static int vpn_init(void)
 {
        int err;
@@ -1832,6 +2043,7 @@ static int vpn_init(void)
                vpnd_created(connection, &provider_driver);
        }
 
+       connman_notifier_register(&vpn_notifier);
        return err;
 
 remove:
@@ -1852,6 +2064,7 @@ static void vpn_exit(void)
        g_dbus_remove_watch(connection, removed_watch);
        g_dbus_remove_watch(connection, property_watch);
 
+       connman_notifier_unregister(&vpn_notifier);
        connman_provider_driver_unregister(&provider_driver);
 
        if (vpn_connections)