You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							492 lines
						
					
					
						
							19 KiB
						
					
					
				
			
		
		
	
	
							492 lines
						
					
					
						
							19 KiB
						
					
					
				| #include "selfdrive/ui/qt/network/wifi_manager.h"
 | |
| 
 | |
| #include "selfdrive/ui/ui.h"
 | |
| #include "selfdrive/ui/qt/widgets/prime.h"
 | |
| 
 | |
| #include "common/params.h"
 | |
| #include "common/swaglog.h"
 | |
| #include "selfdrive/ui/qt/util.h"
 | |
| 
 | |
| bool compare_by_strength(const Network &a, const Network &b) {
 | |
|   return std::tuple(a.connected, strengthLevel(a.strength), b.ssid) >
 | |
|          std::tuple(b.connected, strengthLevel(b.strength), a.ssid);
 | |
| }
 | |
| 
 | |
| template <typename T = QDBusMessage, typename... Args>
 | |
| T call(const QString &path, const QString &interface, const QString &method, Args &&...args) {
 | |
|   QDBusInterface nm = QDBusInterface(NM_DBUS_SERVICE, path, interface, QDBusConnection::systemBus());
 | |
|   nm.setTimeout(DBUS_TIMEOUT);
 | |
|   QDBusMessage response = nm.call(method, args...);
 | |
|   if constexpr (std::is_same_v<T, QDBusMessage>) {
 | |
|     return response;
 | |
|   } else if (response.arguments().count() >= 1) {
 | |
|     QVariant vFirst = response.arguments().at(0).value<QDBusVariant>().variant();
 | |
|     if (vFirst.canConvert<T>()) {
 | |
|       return vFirst.value<T>();
 | |
|     }
 | |
|     QDebug critical = qCritical();
 | |
|     critical << "Variant unpacking failure :" << method << ',';
 | |
|     (critical << ... << args);
 | |
|   }
 | |
|   return T();
 | |
| }
 | |
| 
 | |
| template <typename... Args>
 | |
| QDBusPendingCall asyncCall(const QString &path, const QString &interface, const QString &method, Args &&...args) {
 | |
|   QDBusInterface nm = QDBusInterface(NM_DBUS_SERVICE, path, interface, QDBusConnection::systemBus());
 | |
|   return nm.asyncCall(method, args...);
 | |
| }
 | |
| 
 | |
| bool emptyPath(const QString &path) {
 | |
|   return path == "" || path == "/";
 | |
| }
 | |
| 
 | |
| WifiManager::WifiManager(QObject *parent) : QObject(parent) {
 | |
|   qDBusRegisterMetaType<Connection>();
 | |
|   qDBusRegisterMetaType<IpConfig>();
 | |
| 
 | |
|   // Set tethering ssid as "weedle" + first 4 characters of a dongle id
 | |
|   tethering_ssid = "weedle";
 | |
|   if (auto dongle_id = getDongleId()) {
 | |
|     tethering_ssid += "-" + dongle_id->left(4);
 | |
|   }
 | |
| 
 | |
|   adapter = getAdapter();
 | |
|   if (!adapter.isEmpty()) {
 | |
|     setup();
 | |
|   } else {
 | |
|     QDBusConnection::systemBus().connect(NM_DBUS_SERVICE, NM_DBUS_PATH, NM_DBUS_INTERFACE, "DeviceAdded", this, SLOT(deviceAdded(QDBusObjectPath)));
 | |
|   }
 | |
| 
 | |
|   timer.callOnTimeout(this, &WifiManager::requestScan);
 | |
| }
 | |
| 
 | |
| void WifiManager::setup() {
 | |
|   auto bus = QDBusConnection::systemBus();
 | |
|   bus.connect(NM_DBUS_SERVICE, adapter, NM_DBUS_INTERFACE_DEVICE, "StateChanged", this, SLOT(stateChange(unsigned int, unsigned int, unsigned int)));
 | |
|   bus.connect(NM_DBUS_SERVICE, adapter, NM_DBUS_INTERFACE_PROPERTIES, "PropertiesChanged", this, SLOT(propertyChange(QString, QVariantMap, QStringList)));
 | |
| 
 | |
|   bus.connect(NM_DBUS_SERVICE, NM_DBUS_PATH_SETTINGS, NM_DBUS_INTERFACE_SETTINGS, "ConnectionRemoved", this, SLOT(connectionRemoved(QDBusObjectPath)));
 | |
|   bus.connect(NM_DBUS_SERVICE, NM_DBUS_PATH_SETTINGS, NM_DBUS_INTERFACE_SETTINGS, "NewConnection", this, SLOT(newConnection(QDBusObjectPath)));
 | |
| 
 | |
|   raw_adapter_state = call<uint>(adapter, NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_DEVICE, "State");
 | |
|   activeAp = call<QDBusObjectPath>(adapter, NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_DEVICE_WIRELESS, "ActiveAccessPoint").path();
 | |
| 
 | |
|   initConnections();
 | |
|   requestScan();
 | |
| }
 | |
| 
 | |
| void WifiManager::start() {
 | |
|   timer.start(5000);
 | |
|   refreshNetworks();
 | |
| }
 | |
| 
 | |
| void WifiManager::stop() {
 | |
|   timer.stop();
 | |
| }
 | |
| 
 | |
| void WifiManager::refreshNetworks() {
 | |
|   if (adapter.isEmpty() || !timer.isActive()) return;
 | |
| 
 | |
|   QDBusPendingCall pending_call = asyncCall(adapter, NM_DBUS_INTERFACE_DEVICE_WIRELESS, "GetAllAccessPoints");
 | |
|   QDBusPendingCallWatcher *watcher = new QDBusPendingCallWatcher(pending_call);
 | |
|   QObject::connect(watcher, &QDBusPendingCallWatcher::finished, this, &WifiManager::refreshFinished);
 | |
| }
 | |
| 
 | |
| void WifiManager::refreshFinished(QDBusPendingCallWatcher *watcher) {
 | |
|   ipv4_address = getIp4Address();
 | |
|   seenNetworks.clear();
 | |
| 
 | |
|   const QDBusReply<QList<QDBusObjectPath>> wather_reply = *watcher;
 | |
|   for (const QDBusObjectPath &path : wather_reply.value()) {
 | |
|     QDBusReply<QVariantMap> replay = call(path.path(), NM_DBUS_INTERFACE_PROPERTIES, "GetAll", NM_DBUS_INTERFACE_ACCESS_POINT);
 | |
|     auto properties = replay.value();
 | |
| 
 | |
|     const QByteArray ssid = properties["Ssid"].toByteArray();
 | |
|     if (ssid.isEmpty()) continue;
 | |
| 
 | |
|     // May be multiple access points for each SSID.
 | |
|     // Use first for ssid and security type, then update connected status and strength using all
 | |
|     if (!seenNetworks.contains(ssid)) {
 | |
|       seenNetworks[ssid] = {ssid, 0U, ConnectedType::DISCONNECTED, getSecurityType(properties)};
 | |
|     }
 | |
| 
 | |
|     if (path.path() == activeAp) {
 | |
|       seenNetworks[ssid].connected = (ssid == connecting_to_network) ? ConnectedType::CONNECTING : ConnectedType::CONNECTED;
 | |
|     }
 | |
| 
 | |
|     uint32_t strength = properties["Strength"].toUInt();
 | |
|     if (seenNetworks[ssid].strength < strength) {
 | |
|       seenNetworks[ssid].strength = strength;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   emit refreshSignal();
 | |
|   watcher->deleteLater();
 | |
| }
 | |
| 
 | |
| QString WifiManager::getIp4Address() {
 | |
|   if (raw_adapter_state != NM_DEVICE_STATE_ACTIVATED) return "";
 | |
| 
 | |
|   for (const auto &p : getActiveConnections()) {
 | |
|     QString type = call<QString>(p.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_ACTIVE_CONNECTION, "Type");
 | |
|     if (type == "802-11-wireless") {
 | |
|       auto ip4config = call<QDBusObjectPath>(p.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_ACTIVE_CONNECTION, "Ip4Config");
 | |
|       const auto &arr = call<QDBusArgument>(ip4config.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_IP4_CONFIG, "AddressData");
 | |
|       QVariantMap path;
 | |
|       arr.beginArray();
 | |
|       while (!arr.atEnd()) {
 | |
|         arr >> path;
 | |
|         arr.endArray();
 | |
|         return path.value("address").value<QString>();
 | |
|       }
 | |
|       arr.endArray();
 | |
|     }
 | |
|   }
 | |
|   return "";
 | |
| }
 | |
| 
 | |
| SecurityType WifiManager::getSecurityType(const QVariantMap &properties) {
 | |
|   int sflag = properties["Flags"].toUInt();
 | |
|   int wpaflag = properties["WpaFlags"].toUInt();
 | |
|   int rsnflag = properties["RsnFlags"].toUInt();
 | |
|   int wpa_props = wpaflag | rsnflag;
 | |
| 
 | |
|   // obtained by looking at flags of networks in the office as reported by an Android phone
 | |
|   const int supports_wpa = NM_802_11_AP_SEC_PAIR_WEP40 | NM_802_11_AP_SEC_PAIR_WEP104 | NM_802_11_AP_SEC_GROUP_WEP40 | NM_802_11_AP_SEC_GROUP_WEP104 | NM_802_11_AP_SEC_KEY_MGMT_PSK;
 | |
| 
 | |
|   if ((sflag == NM_802_11_AP_FLAGS_NONE) || ((sflag & NM_802_11_AP_FLAGS_WPS) && !(wpa_props & supports_wpa))) {
 | |
|     return SecurityType::OPEN;
 | |
|   } else if ((sflag & NM_802_11_AP_FLAGS_PRIVACY) && (wpa_props & supports_wpa) && !(wpa_props & NM_802_11_AP_SEC_KEY_MGMT_802_1X)) {
 | |
|     return SecurityType::WPA;
 | |
|   } else {
 | |
|     LOGW("Unsupported network! sflag: %d, wpaflag: %d, rsnflag: %d", sflag, wpaflag, rsnflag);
 | |
|     return SecurityType::UNSUPPORTED;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WifiManager::connect(const Network &n, const QString &password, const QString &username) {
 | |
|   setCurrentConnecting(n.ssid);
 | |
|   forgetConnection(n.ssid);  // Clear all connections that may already exist to the network we are connecting
 | |
|   Connection connection;
 | |
|   connection["connection"]["type"] = "802-11-wireless";
 | |
|   connection["connection"]["uuid"] = QUuid::createUuid().toString().remove('{').remove('}');
 | |
|   connection["connection"]["id"] = "openpilot connection " + QString::fromStdString(n.ssid.toStdString());
 | |
|   connection["connection"]["autoconnect-retries"] = 0;
 | |
| 
 | |
|   connection["802-11-wireless"]["ssid"] = n.ssid;
 | |
|   connection["802-11-wireless"]["mode"] = "infrastructure";
 | |
| 
 | |
|   if (n.security_type == SecurityType::WPA) {
 | |
|     connection["802-11-wireless-security"]["key-mgmt"] = "wpa-psk";
 | |
|     connection["802-11-wireless-security"]["auth-alg"] = "open";
 | |
|     connection["802-11-wireless-security"]["psk"] = password;
 | |
|   }
 | |
| 
 | |
|   connection["ipv4"]["method"] = "auto";
 | |
|   connection["ipv4"]["dns-priority"] = 600;
 | |
|   connection["ipv6"]["method"] = "ignore";
 | |
| 
 | |
|   call(NM_DBUS_PATH_SETTINGS, NM_DBUS_INTERFACE_SETTINGS, "AddConnection", QVariant::fromValue(connection));
 | |
| }
 | |
| 
 | |
| void WifiManager::deactivateConnectionBySsid(const QString &ssid) {
 | |
|   for (QDBusObjectPath active_connection : getActiveConnections()) {
 | |
|     auto pth = call<QDBusObjectPath>(active_connection.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_ACTIVE_CONNECTION, "SpecificObject");
 | |
|     if (!emptyPath(pth.path())) {
 | |
|       QString Ssid = get_property(pth.path(), "Ssid");
 | |
|       if (Ssid == ssid) {
 | |
|         deactivateConnection(active_connection);
 | |
|         return;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WifiManager::deactivateConnection(const QDBusObjectPath &path) {
 | |
|   asyncCall(NM_DBUS_PATH, NM_DBUS_INTERFACE, "DeactivateConnection", QVariant::fromValue(path));
 | |
| }
 | |
| 
 | |
| QVector<QDBusObjectPath> WifiManager::getActiveConnections() {
 | |
|   QVector<QDBusObjectPath> conns;
 | |
|   QDBusObjectPath path;
 | |
|   const QDBusArgument &arr = call<QDBusArgument>(NM_DBUS_PATH, NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE, "ActiveConnections");
 | |
|   arr.beginArray();
 | |
|   while (!arr.atEnd()) {
 | |
|     arr >> path;
 | |
|     conns.push_back(path);
 | |
|   }
 | |
|   arr.endArray();
 | |
|   return conns;
 | |
| }
 | |
| 
 | |
| bool WifiManager::isKnownConnection(const QString &ssid) {
 | |
|   return !getConnectionPath(ssid).path().isEmpty();
 | |
| }
 | |
| 
 | |
| void WifiManager::forgetConnection(const QString &ssid) {
 | |
|   const QDBusObjectPath &path = getConnectionPath(ssid);
 | |
|   if (!path.path().isEmpty()) {
 | |
|     call(path.path(), NM_DBUS_INTERFACE_SETTINGS_CONNECTION, "Delete");
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WifiManager::setCurrentConnecting(const QString &ssid) {
 | |
|   connecting_to_network = ssid;
 | |
|   for (auto &network : seenNetworks) {
 | |
|     network.connected = (network.ssid == ssid) ? ConnectedType::CONNECTING : ConnectedType::DISCONNECTED;
 | |
|   }
 | |
|   emit refreshSignal();
 | |
| }
 | |
| 
 | |
| uint WifiManager::getAdapterType(const QDBusObjectPath &path) {
 | |
|   return call<uint>(path.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_DEVICE, "DeviceType");
 | |
| }
 | |
| 
 | |
| void WifiManager::requestScan() {
 | |
|   if (!adapter.isEmpty()) {
 | |
|     asyncCall(adapter, NM_DBUS_INTERFACE_DEVICE_WIRELESS, "RequestScan", QVariantMap());
 | |
|   }
 | |
| }
 | |
| 
 | |
| QByteArray WifiManager::get_property(const QString &network_path , const QString &property) {
 | |
|   return call<QByteArray>(network_path, NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_ACCESS_POINT, property);
 | |
| }
 | |
| 
 | |
| QString WifiManager::getAdapter(const uint adapter_type) {
 | |
|   QDBusReply<QList<QDBusObjectPath>> response = call(NM_DBUS_PATH, NM_DBUS_INTERFACE, "GetDevices");
 | |
|   for (const QDBusObjectPath &path : response.value()) {
 | |
|     if (getAdapterType(path) == adapter_type) {
 | |
|       return path.path();
 | |
|     }
 | |
|   }
 | |
|   return "";
 | |
| }
 | |
| 
 | |
| void WifiManager::stateChange(unsigned int new_state, unsigned int previous_state, unsigned int change_reason) {
 | |
|   raw_adapter_state = new_state;
 | |
|   if (new_state == NM_DEVICE_STATE_NEED_AUTH && change_reason == NM_DEVICE_STATE_REASON_SUPPLICANT_DISCONNECT && !connecting_to_network.isEmpty()) {
 | |
|     forgetConnection(connecting_to_network);
 | |
|     emit wrongPassword(connecting_to_network);
 | |
|   } else if (new_state == NM_DEVICE_STATE_ACTIVATED) {
 | |
|     connecting_to_network = "";
 | |
|     refreshNetworks();
 | |
|   }
 | |
| }
 | |
| 
 | |
| // https://developer.gnome.org/NetworkManager/stable/gdbus-org.freedesktop.NetworkManager.Device.Wireless.html
 | |
| void WifiManager::propertyChange(const QString &interface, const QVariantMap &props, const QStringList &invalidated_props) {
 | |
|   if (interface == NM_DBUS_INTERFACE_DEVICE_WIRELESS && props.contains("LastScan")) {
 | |
|     refreshNetworks();
 | |
|   } else if (interface == NM_DBUS_INTERFACE_DEVICE_WIRELESS && props.contains("ActiveAccessPoint")) {
 | |
|     activeAp = props.value("ActiveAccessPoint").value<QDBusObjectPath>().path();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WifiManager::deviceAdded(const QDBusObjectPath &path) {
 | |
|   if (getAdapterType(path) == NM_DEVICE_TYPE_WIFI && emptyPath(adapter)) {
 | |
|     adapter = path.path();
 | |
|     setup();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WifiManager::connectionRemoved(const QDBusObjectPath &path) {
 | |
|   knownConnections.remove(path);
 | |
| }
 | |
| 
 | |
| void WifiManager::newConnection(const QDBusObjectPath &path) {
 | |
|   Connection settings = getConnectionSettings(path);
 | |
|   if (settings.value("connection").value("type") == "802-11-wireless") {
 | |
|     knownConnections[path] = settings.value("802-11-wireless").value("ssid").toString();
 | |
|     if (knownConnections[path] != tethering_ssid) {
 | |
|       activateWifiConnection(knownConnections[path]);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| QDBusObjectPath WifiManager::getConnectionPath(const QString &ssid) {
 | |
|   return knownConnections.key(ssid);
 | |
| }
 | |
| 
 | |
| Connection WifiManager::getConnectionSettings(const QDBusObjectPath &path) {
 | |
|   return QDBusReply<Connection>(call(path.path(), NM_DBUS_INTERFACE_SETTINGS_CONNECTION, "GetSettings")).value();
 | |
| }
 | |
| 
 | |
| void WifiManager::initConnections() {
 | |
|   const QDBusReply<QList<QDBusObjectPath>> response = call(NM_DBUS_PATH_SETTINGS, NM_DBUS_INTERFACE_SETTINGS, "ListConnections");
 | |
|   for (const QDBusObjectPath &path : response.value()) {
 | |
|     const Connection settings = getConnectionSettings(path);
 | |
|     if (settings.value("connection").value("type") == "802-11-wireless") {
 | |
|       knownConnections[path] = settings.value("802-11-wireless").value("ssid").toString();
 | |
|     } else if (settings.value("connection").value("id") == "lte") {
 | |
|       lteConnectionPath = path;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::optional<QDBusPendingCall> WifiManager::activateWifiConnection(const QString &ssid) {
 | |
|   const QDBusObjectPath &path = getConnectionPath(ssid);
 | |
|   if (!path.path().isEmpty()) {
 | |
|     setCurrentConnecting(ssid);
 | |
|     return asyncCall(NM_DBUS_PATH, NM_DBUS_INTERFACE, "ActivateConnection", QVariant::fromValue(path), QVariant::fromValue(QDBusObjectPath(adapter)), QVariant::fromValue(QDBusObjectPath("/")));
 | |
|   }
 | |
|   return std::nullopt;
 | |
| }
 | |
| 
 | |
| void WifiManager::activateModemConnection(const QDBusObjectPath &path) {
 | |
|   QString modem = getAdapter(NM_DEVICE_TYPE_MODEM);
 | |
|   if (!path.path().isEmpty() && !modem.isEmpty()) {
 | |
|     asyncCall(NM_DBUS_PATH, NM_DBUS_INTERFACE, "ActivateConnection", QVariant::fromValue(path), QVariant::fromValue(QDBusObjectPath(modem)), QVariant::fromValue(QDBusObjectPath("/")));
 | |
|   }
 | |
| }
 | |
| 
 | |
| // function matches tici/hardware.py
 | |
| NetworkType WifiManager::currentNetworkType() {
 | |
|   auto primary_conn = call<QDBusObjectPath>(NM_DBUS_PATH, NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE, "PrimaryConnection");
 | |
|   auto primary_type = call<QString>(primary_conn.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_ACTIVE_CONNECTION, "Type");
 | |
| 
 | |
|   if (primary_type == "802-3-ethernet") {
 | |
|     return NetworkType::ETHERNET;
 | |
|   } else if (primary_type == "802-11-wireless" && !isTetheringEnabled()) {
 | |
|     return NetworkType::WIFI;
 | |
|   } else {
 | |
|     for (const QDBusObjectPath &conn : getActiveConnections()) {
 | |
|       auto type = call<QString>(conn.path(), NM_DBUS_INTERFACE_PROPERTIES, "Get", NM_DBUS_INTERFACE_ACTIVE_CONNECTION, "Type");
 | |
|       if (type == "gsm") {
 | |
|         return NetworkType::CELL;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return NetworkType::NONE;
 | |
| }
 | |
| 
 | |
| void WifiManager::updateGsmSettings(bool roaming, QString apn, bool metered) {
 | |
|   if (!lteConnectionPath.path().isEmpty()) {
 | |
|     bool changes = false;
 | |
|     bool auto_config = apn.isEmpty();
 | |
|     Connection settings = getConnectionSettings(lteConnectionPath);
 | |
|     if (settings.value("gsm").value("auto-config").toBool() != auto_config) {
 | |
|       qWarning() << "Changing gsm.auto-config to" << auto_config;
 | |
|       settings["gsm"]["auto-config"] = auto_config;
 | |
|       changes = true;
 | |
|     }
 | |
| 
 | |
|     if (settings.value("gsm").value("apn").toString() != apn) {
 | |
|       qWarning() << "Changing gsm.apn to" << apn;
 | |
|       settings["gsm"]["apn"] = apn;
 | |
|       changes = true;
 | |
|     }
 | |
| 
 | |
|     if (settings.value("gsm").value("home-only").toBool() == roaming) {
 | |
|       qWarning() << "Changing gsm.home-only to" << !roaming;
 | |
|       settings["gsm"]["home-only"] = !roaming;
 | |
|       changes = true;
 | |
|     }
 | |
| 
 | |
|     int meteredInt = metered ? NM_METERED_UNKNOWN : NM_METERED_NO;
 | |
|     if (settings.value("connection").value("metered").toInt() != meteredInt) {
 | |
|       qWarning() << "Changing connection.metered to" << meteredInt;
 | |
|       settings["connection"]["metered"] = meteredInt;
 | |
|       changes = true;
 | |
|     }
 | |
| 
 | |
|     if (changes) {
 | |
|       call(lteConnectionPath.path(), NM_DBUS_INTERFACE_SETTINGS_CONNECTION, "UpdateUnsaved", QVariant::fromValue(settings));  // update is temporary
 | |
|       deactivateConnection(lteConnectionPath);
 | |
|       activateModemConnection(lteConnectionPath);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Functions for tethering
 | |
| void WifiManager::addTetheringConnection() {
 | |
|   Connection connection;
 | |
|   connection["connection"]["id"] = "Hotspot";
 | |
|   connection["connection"]["uuid"] = QUuid::createUuid().toString().remove('{').remove('}');
 | |
|   connection["connection"]["type"] = "802-11-wireless";
 | |
|   connection["connection"]["interface-name"] = "wlan0";
 | |
|   connection["connection"]["autoconnect"] = false;
 | |
| 
 | |
|   connection["802-11-wireless"]["band"] = "bg";
 | |
|   connection["802-11-wireless"]["mode"] = "ap";
 | |
|   connection["802-11-wireless"]["ssid"] = tethering_ssid.toUtf8();
 | |
| 
 | |
|   connection["802-11-wireless-security"]["group"] = QStringList("ccmp");
 | |
|   connection["802-11-wireless-security"]["key-mgmt"] = "wpa-psk";
 | |
|   connection["802-11-wireless-security"]["pairwise"] = QStringList("ccmp");
 | |
|   connection["802-11-wireless-security"]["proto"] = QStringList("rsn");
 | |
|   connection["802-11-wireless-security"]["psk"] = defaultTetheringPassword;
 | |
| 
 | |
|   connection["ipv4"]["method"] = "shared";
 | |
|   QVariantMap address;
 | |
|   address["address"] = "192.168.43.1";
 | |
|   address["prefix"] = 24u;
 | |
|   connection["ipv4"]["address-data"] = QVariant::fromValue(IpConfig() << address);
 | |
|   connection["ipv4"]["gateway"] = "192.168.43.1";
 | |
|   connection["ipv4"]["route-metric"] = 1100;
 | |
|   connection["ipv6"]["method"] = "ignore";
 | |
| 
 | |
|   call(NM_DBUS_PATH_SETTINGS, NM_DBUS_INTERFACE_SETTINGS, "AddConnection", QVariant::fromValue(connection));
 | |
| }
 | |
| 
 | |
| void WifiManager::tetheringActivated(QDBusPendingCallWatcher *call) {
 | |
|   int prime_type = uiState()->primeType();
 | |
|   int ipv4_forward = (prime_type == PrimeType::NONE || prime_type == PrimeType::LITE);
 | |
| 
 | |
|   if (!ipv4_forward) {
 | |
|     QTimer::singleShot(5000, this, [=] {
 | |
|       qWarning() << "net.ipv4.ip_forward = 0";
 | |
|       std::system("sudo sysctl net.ipv4.ip_forward=0");
 | |
|     });
 | |
|   }
 | |
|   call->deleteLater();
 | |
| }
 | |
| 
 | |
| void WifiManager::setTetheringEnabled(bool enabled) {
 | |
|   if (enabled) {
 | |
|     if (!isKnownConnection(tethering_ssid)) {
 | |
|       addTetheringConnection();
 | |
|     }
 | |
| 
 | |
|     auto pending_call = activateWifiConnection(tethering_ssid);
 | |
| 
 | |
|     if (pending_call) {
 | |
|       QDBusPendingCallWatcher *watcher = new QDBusPendingCallWatcher(*pending_call);
 | |
|       QObject::connect(watcher, &QDBusPendingCallWatcher::finished, this, &WifiManager::tetheringActivated);
 | |
|     }
 | |
| 
 | |
|   } else {
 | |
|     deactivateConnectionBySsid(tethering_ssid);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool WifiManager::isTetheringEnabled() {
 | |
|   if (!emptyPath(activeAp)) {
 | |
|     return get_property(activeAp, "Ssid") == tethering_ssid;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| QString WifiManager::getTetheringPassword() {
 | |
|   if (!isKnownConnection(tethering_ssid)) {
 | |
|     addTetheringConnection();
 | |
|   }
 | |
|   const QDBusObjectPath &path = getConnectionPath(tethering_ssid);
 | |
|   if (!path.path().isEmpty()) {
 | |
|     QDBusReply<QMap<QString, QVariantMap>> response = call(path.path(), NM_DBUS_INTERFACE_SETTINGS_CONNECTION, "GetSecrets", "802-11-wireless-security");
 | |
|     return response.value().value("802-11-wireless-security").value("psk").toString();
 | |
|   }
 | |
|   return "";
 | |
| }
 | |
| 
 | |
| void WifiManager::changeTetheringPassword(const QString &newPassword) {
 | |
|   const QDBusObjectPath &path = getConnectionPath(tethering_ssid);
 | |
|   if (!path.path().isEmpty()) {
 | |
|     Connection settings = getConnectionSettings(path);
 | |
|     settings["802-11-wireless-security"]["psk"] = newPassword;
 | |
|     call(path.path(), NM_DBUS_INTERFACE_SETTINGS_CONNECTION, "Update", QVariant::fromValue(settings));
 | |
|     if (isTetheringEnabled()) {
 | |
|       activateWifiConnection(tethering_ssid);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 |