/* NetworkManager -- Network link manager * * Dan Williams * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * (C) Copyright 2005 Red Hat, Inc. */ #include #include #include #include #include #include #include #include "nm-utils.h" #include "nm-device.h" #include "NetworkManagerDbus.h" #include "NetworkManagerDbusUtils.h" #include "NetworkManagerPolicy.h" #include "NetworkManagerUtils.h" #include "nm-dbus-device.h" #include "nm-device-802-3-ethernet.h" #include "nm-device-802-11-wireless.h" static DBusMessage *nm_dbus_device_get_name (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { const char *iface; iface = nm_device_get_iface (dev); dbus_message_append_args (reply, DBUS_TYPE_STRING, &iface, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_type (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { dbus_int32_t type; type = nm_device_get_device_type (dev); dbus_message_append_args (reply, DBUS_TYPE_INT32, &type, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_hal_udi (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { const char *udi; udi = nm_device_get_udi (dev); dbus_message_append_args (reply, DBUS_TYPE_STRING, &udi, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_ip4_address (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { dbus_uint32_t address; address = nm_device_get_ip4_address (dev); dbus_message_append_args (reply, DBUS_TYPE_UINT32, &address, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_hw_address (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { struct ether_addr addr; char char_addr[20]; char * ptr = &char_addr[0]; memset (&addr, 0, sizeof (struct ether_addr)); if (nm_device_is_802_3_ethernet (dev)) nm_device_802_3_ethernet_get_address (NM_DEVICE_802_3_ETHERNET (dev), &addr); else if (nm_device_is_802_11_wireless (dev)) nm_device_802_11_wireless_get_address (NM_DEVICE_802_11_WIRELESS (dev), &addr); memset (char_addr, 0, 20); iw_ether_ntop (&addr, char_addr); dbus_message_append_args (reply, DBUS_TYPE_STRING, &ptr, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_mode (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if (!nm_device_is_802_11_wireless (dev)) { reply = nm_dbus_create_error_message (message, NM_DBUS_INTERFACE, "DeviceNotWireless", "Wired devices cannot see wireless networks."); } else if ((reply = dbus_message_new_method_return (message))) { dbus_int32_t mode = (dbus_int32_t) nm_device_802_11_wireless_get_mode (NM_DEVICE_802_11_WIRELESS (dev)); dbus_message_append_args (reply, DBUS_TYPE_INT32, &mode, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_link_active (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { dbus_bool_t is_active; is_active = nm_device_has_active_link (dev); dbus_message_append_args (reply, DBUS_TYPE_BOOLEAN, &is_active, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_get_active_network (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; gboolean success = FALSE; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); /* Only wireless devices have an active network */ dev = data->dev; if (!nm_device_is_802_11_wireless (dev)) { reply = nm_dbus_create_error_message (message, NM_DBUS_INTERFACE, "DeviceNotWireless", "Wired devices cannot have active networks."); } else if ((reply = dbus_message_new_method_return (message))) { NMActRequest * req = nm_device_get_act_request (dev); NMAccessPoint * ap; if (req && (ap = nm_act_request_get_ap (req))) { NMAccessPoint *tmp_ap; char * object_path = NULL; tmp_ap = nm_device_802_11_wireless_ap_list_get_ap_by_essid (NM_DEVICE_802_11_WIRELESS (dev), nm_ap_get_essid (ap)); if (tmp_ap && (object_path = nm_dbus_get_object_path_for_network (dev, tmp_ap))) { dbus_message_append_args (reply, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID); g_free (object_path); success = TRUE; } } if (!success) { dbus_message_unref (reply); reply = nm_dbus_create_error_message (message, NM_DBUS_INTERFACE, "NoActiveNetwork", "The device is not associated with any networks at this time."); } } return reply; } static DBusMessage *nm_dbus_device_get_networks (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); /* Only wireless devices have networks */ dev = data->dev; if (!nm_device_is_802_11_wireless (dev)) { reply = nm_dbus_create_error_message (message, NM_DBUS_INTERFACE, "DeviceNotWireless", "Wired devices cannot see wireless networks."); } else if ((reply = dbus_message_new_method_return (message))) { DBusMessageIter iter; DBusMessageIter iter_array; NMAccessPoint *ap = NULL; gboolean success = FALSE; NMAccessPointList *ap_list; NMAPListIter *list_iter; char *escaped_object_path; dbus_message_iter_init_append (reply, &iter); dbus_message_iter_open_container (&iter, DBUS_TYPE_ARRAY, DBUS_TYPE_OBJECT_PATH_AS_STRING, &iter_array); if ((ap_list = nm_device_802_11_wireless_ap_list_get (NM_DEVICE_802_11_WIRELESS (dev)))) { if ((list_iter = nm_ap_list_iter_new (ap_list))) { while ((ap = nm_ap_list_iter_next (list_iter))) { if (nm_ap_get_essid (ap)) { escaped_object_path = nm_dbus_get_object_path_for_network (dev, ap); dbus_message_iter_append_basic (&iter_array, DBUS_TYPE_OBJECT_PATH, &escaped_object_path); g_free (escaped_object_path); success = TRUE; } } nm_ap_list_iter_free (list_iter); } } dbus_message_iter_close_container (&iter, &iter_array); if (!success) { dbus_message_unref (reply); reply = nm_dbus_create_error_message (message, NM_DBUS_INTERFACE, "NoNetworks", "The device cannot see any wireless networks."); } } return reply; } static DBusMessage *nm_dbus_device_get_capabilities (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { dbus_uint32_t capabilities = nm_device_get_capabilities (dev); dbus_message_append_args (reply, DBUS_TYPE_UINT32, &capabilities, DBUS_TYPE_INVALID); } return reply; } static DBusMessage *nm_dbus_device_set_link_active (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); /* Can only set link status for active devices */ dev = data->dev; if (!nm_device_is_test_device (dev)) { reply = nm_dbus_create_error_message (message, NM_DBUS_INTERFACE, "NotTestDevice", "Only test devices can have their link status set manually."); } else if ((reply = dbus_message_new_method_return (message))) { DBusError error; gboolean link; dbus_error_init (&error); if (dbus_message_get_args (message, &error, DBUS_TYPE_BOOLEAN, &link, DBUS_TYPE_INVALID)) { nm_device_set_active_link (dev, link); nm_policy_schedule_device_change_check (data->data); } } return reply; } static DBusMessage *nm_dbus_device_get_properties (DBusConnection *connection, DBusMessage *message, NMDbusCBData *data) { DBusMessage *reply = NULL; NMDevice *dev; g_return_val_if_fail (data && data->data && data->dev && connection && message, NULL); dev = data->dev; if ((reply = dbus_message_new_method_return (message))) { char * op = nm_dbus_get_object_path_for_device (dev); const char * iface = nm_device_get_iface (dev); dbus_uint32_t type = (dbus_uint32_t) nm_device_get_device_type (dev); const char * udi = nm_device_get_udi (dev); gchar * ip4_address; gchar * broadcast; gchar * subnetmask; gchar * route; gchar * primary_dns; gchar * secondary_dns; struct ether_addr hw_addr; char hw_addr_buf[20]; char * hw_addr_buf_ptr = &hw_addr_buf[0]; dbus_int32_t mode = -1; dbus_int32_t strength = -1; char * active_network_path = NULL; dbus_bool_t link_active = (dbus_bool_t) nm_device_has_active_link (dev); dbus_uint32_t capabilities = (dbus_uint32_t) nm_device_get_capabilities (dev); dbus_uint32_t type_capabilities = (dbus_uint32_t) nm_device_get_type_capabilities (dev); char ** networks = NULL; int num_networks = 0; dbus_bool_t active = nm_device_get_act_request (dev) ? TRUE : FALSE; NMActStage act_stage = active ? nm_act_request_get_stage (nm_device_get_act_request (dev)) : NM_ACT_STAGE_UNKNOWN; NMIP4Config * ip4config; guint32 broadcast_addr = 0; guint32 subnetmask_addr = 0; guint32 route_addr = 0; guint32 primary_dns_addr = 0; guint32 secondary_dns_addr = 0; memset (hw_addr_buf, 0, 20); if (nm_device_is_802_3_ethernet (dev)) nm_device_802_3_ethernet_get_address (NM_DEVICE_802_3_ETHERNET (dev), &hw_addr); else if (nm_device_is_802_11_wireless (dev)) nm_device_802_11_wireless_get_address (NM_DEVICE_802_11_WIRELESS (dev), &hw_addr); iw_ether_ntop (&hw_addr, hw_addr_buf); ip4config = nm_device_get_ip4_config (dev); if (ip4config) { guint32 nr_nameservers; broadcast_addr = nm_ip4_config_get_broadcast (ip4config); subnetmask_addr = nm_ip4_config_get_netmask (ip4config); route_addr = nm_ip4_config_get_gateway (ip4config); nr_nameservers = nm_ip4_config_get_num_nameservers (ip4config); if (nr_nameservers > 1) secondary_dns_addr = nm_ip4_config_get_nameserver (ip4config, 1); if (nr_nameservers > 0) primary_dns_addr = nm_ip4_config_get_nameserver (ip4config, 0); } ip4_address = nm_utils_inet_ip4_address_as_string (nm_device_get_ip4_address (dev)); broadcast = nm_utils_inet_ip4_address_as_string (broadcast_addr); subnetmask = nm_utils_inet_ip4_address_as_string (subnetmask_addr); route = nm_utils_inet_ip4_address_as_string (route_addr); primary_dns = nm_utils_inet_ip4_address_as_string (primary_dns_addr); secondary_dns = nm_utils_inet_ip4_address_as_string (secondary_dns_addr); if (nm_device_is_802_11_wireless (dev)) { NMDevice80211Wireless * wdev = NM_DEVICE_802_11_WIRELESS (dev); NMActRequest * req = nm_device_get_act_request (dev); NMAccessPoint * ap; NMAccessPointList * ap_list; NMAPListIter * iter; strength = (dbus_int32_t) nm_device_802_11_wireless_get_signal_strength (wdev); mode = (dbus_int32_t) nm_device_802_11_wireless_get_mode (wdev); if (req && (ap = nm_act_request_get_ap (req))) { NMAccessPoint *tmp_ap; if ((tmp_ap = nm_device_802_11_wireless_ap_list_get_ap_by_essid (wdev, nm_ap_get_essid (ap)))) active_network_path = nm_dbus_get_object_path_for_network (dev, tmp_ap); } ap_list = nm_device_802_11_wireless_ap_list_get (wdev); if (ap_list && (num_networks = nm_ap_list_size (ap_list))) { if ((iter = nm_ap_list_iter_new (ap_list))) { int i = 0; NMAccessPoint * ap; networks = g_malloc0 (sizeof (char *) * (num_networks + 1)); while ((ap = nm_ap_list_iter_next (iter))) { char *ap_op = nm_dbus_get_object_path_for_network (dev, ap); if (ap_op) networks[i++] = ap_op; } num_networks = i; /* # actually added to array, since we can have NULL essid access points */ nm_ap_list_iter_free (iter); } } } if (!active_network_path) active_network_path = g_strdup (""); dbus_message_append_args (reply, DBUS_TYPE_OBJECT_PATH, &op, DBUS_TYPE_STRING, &iface, DBUS_TYPE_UINT32, &type, DBUS_TYPE_STRING, &udi, DBUS_TYPE_BOOLEAN,&active, DBUS_TYPE_UINT32, &act_stage, DBUS_TYPE_STRING, &ip4_address, DBUS_TYPE_STRING, &subnetmask, DBUS_TYPE_STRING, &broadcast, DBUS_TYPE_STRING, &hw_addr_buf_ptr, DBUS_TYPE_STRING, &route, DBUS_TYPE_STRING, &primary_dns, DBUS_TYPE_STRING, &secondary_dns, DBUS_TYPE_INT32, &mode, DBUS_TYPE_INT32, &strength, DBUS_TYPE_BOOLEAN,&link_active, DBUS_TYPE_UINT32, &capabilities, DBUS_TYPE_UINT32, &type_capabilities, DBUS_TYPE_STRING, &active_network_path, DBUS_TYPE_ARRAY, DBUS_TYPE_STRING, &networks, num_networks, DBUS_TYPE_INVALID); g_free (op); g_free (active_network_path); g_strfreev (networks); g_free (route); g_free (ip4_address); g_free (broadcast); g_free (subnetmask); } return reply; } /* * nm_dbus_device_methods_setup * * Register handlers for dbus methods on the org.freedesktop.NetworkManager.Devices object. * */ NMDbusMethodList *nm_dbus_device_methods_setup (void) { NMDbusMethodList *list = nm_dbus_method_list_new (NULL); nm_dbus_method_list_add_method (list, "getProperties", nm_dbus_device_get_properties); nm_dbus_method_list_add_method (list, "getName", nm_dbus_device_get_name); nm_dbus_method_list_add_method (list, "getType", nm_dbus_device_get_type); nm_dbus_method_list_add_method (list, "getHalUdi", nm_dbus_device_get_hal_udi); nm_dbus_method_list_add_method (list, "getIP4Address", nm_dbus_device_get_ip4_address); nm_dbus_method_list_add_method (list, "getHWAddress", nm_dbus_device_get_hw_address); nm_dbus_method_list_add_method (list, "getMode", nm_dbus_device_get_mode); nm_dbus_method_list_add_method (list, "getActiveNetwork", nm_dbus_device_get_active_network); nm_dbus_method_list_add_method (list, "getNetworks", nm_dbus_device_get_networks); nm_dbus_method_list_add_method (list, "getLinkActive", nm_dbus_device_get_link_active); nm_dbus_method_list_add_method (list, "setLinkActive", nm_dbus_device_set_link_active); nm_dbus_method_list_add_method (list, "getCapabilities", nm_dbus_device_get_capabilities); return (list); }