blob: 80844f0c34747e1c7e546f82089095dbe74f459c [file] [log] [blame]
mukesh agrawal8a3188d2011-12-01 20:56:44 +00001// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
mukesh agrawalb54601c2011-06-07 17:39:22 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#ifndef SHILL_WIFI_SERVICE_
6#define SHILL_WIFI_SERVICE_
7
mukesh agrawal261daca2011-12-02 18:56:56 +00008#include <set>
mukesh agrawalb54601c2011-06-07 17:39:22 -07009#include <string>
10#include <vector>
11
Paul Stewart5baebb72013-03-14 11:43:29 -070012#include <base/memory/scoped_ptr.h>
13
Darin Petkov4a09b6b2011-07-19 12:52:06 -070014#include "shill/dbus_bindings/supplicant-interface.h"
Paul Stewart26b327e2011-10-19 11:38:09 -070015#include "shill/event_dispatcher.h"
Paul Stewart71a4d3b2013-01-18 18:12:56 -080016#include "shill/key_value_store.h"
Chris Masone2b105542011-06-22 10:58:09 -070017#include "shill/refptr_types.h"
mukesh agrawalb54601c2011-06-07 17:39:22 -070018#include "shill/service.h"
mukesh agrawalb54601c2011-06-07 17:39:22 -070019
20namespace shill {
21
Paul Stewart5baebb72013-03-14 11:43:29 -070022class CertificateFile;
Chris Masone6791a432011-07-12 13:23:19 -070023class ControlInterface;
24class EventDispatcher;
mukesh agrawal1a056262011-10-05 14:36:54 -070025class Error;
Chris Masone6791a432011-07-12 13:23:19 -070026class Manager;
Thieu Le3426c8f2012-01-11 17:35:11 -080027class Metrics;
Paul Stewartecf4cd12012-04-17 11:08:39 -070028class NSS;
Paul Stewart3c504012013-01-17 17:49:58 -080029class WiFiProvider;
Chris Masone6791a432011-07-12 13:23:19 -070030
mukesh agrawalb54601c2011-06-07 17:39:22 -070031class WiFiService : public Service {
32 public:
Paul Stewart0756db92012-01-27 08:34:47 -080033 // TODO(pstew): Storage constants shouldn't need to be public
34 // crosbug.com/25813
35 static const char kStorageHiddenSSID[];
36 static const char kStorageMode[];
37 static const char kStoragePassphrase[];
38 static const char kStorageSecurity[];
Paul Stewart71a4d3b2013-01-18 18:12:56 -080039 static const char kStorageSecurityClass[];
Paul Stewart0756db92012-01-27 08:34:47 -080040 static const char kStorageSSID[];
41
mukesh agrawalb54601c2011-06-07 17:39:22 -070042 WiFiService(ControlInterface *control_interface,
43 EventDispatcher *dispatcher,
Thieu Le3426c8f2012-01-11 17:35:11 -080044 Metrics *metrics,
Chris Masone6791a432011-07-12 13:23:19 -070045 Manager *manager,
Paul Stewart3c504012013-01-17 17:49:58 -080046 WiFiProvider *provider,
mukesh agrawal7ec71312011-11-10 02:08:26 +000047 const std::vector<uint8_t> &ssid,
Chris Masone092df3e2011-08-22 09:41:39 -070048 const std::string &mode,
Paul Stewartced6a0b2011-11-08 15:32:04 -080049 const std::string &security,
50 bool hidden_ssid);
mukesh agrawalb54601c2011-06-07 17:39:22 -070051 ~WiFiService();
Darin Petkov4d6d9412011-08-24 13:19:54 -070052
53 // Inherited from Service.
mukesh agrawaldc7b8442012-09-27 13:48:14 -070054 virtual void Connect(Error *error, const char *reason);
mukesh agrawal0ed0f2e2011-12-05 20:36:17 +000055 virtual void Disconnect(Error *error);
Paul Stewart81426132012-05-16 10:05:10 -070056 virtual bool Is8021x() const;
57
Paul Stewart3c504012013-01-17 17:49:58 -080058 virtual void AddEndpoint(const WiFiEndpointConstRefPtr &endpoint);
59 virtual void RemoveEndpoint(const WiFiEndpointConstRefPtr &endpoint);
60 virtual int GetEndpointCount() const { return endpoints_.size(); }
mukesh agrawale1d90e92012-02-15 17:36:08 -080061
mukesh agrawalb20776f2012-02-10 16:00:36 -080062 // Called to update the identity of the currently connected endpoint.
mukesh agrawale1d90e92012-02-15 17:36:08 -080063 // To indicate that there is no currently connect endpoint, call with
64 // |endpoint| set to NULL.
Paul Stewart3c504012013-01-17 17:49:58 -080065 virtual void NotifyCurrentEndpoint(const WiFiEndpointConstRefPtr &endpoint);
mukesh agrawalb20776f2012-02-10 16:00:36 -080066 // Called to inform of changes in the properties of an endpoint.
67 // (Not necessarily the currently connected endpoint.)
Paul Stewart3c504012013-01-17 17:49:58 -080068 virtual void NotifyEndpointUpdated(const WiFiEndpointConstRefPtr &endpoint);
mukesh agrawal261daca2011-12-02 18:56:56 +000069
Chris Masone34af2182011-08-22 11:59:36 -070070 // wifi_<MAC>_<BSSID>_<mode_string>_<security_string>
Chris Masone6515aab2011-10-12 16:19:09 -070071 std::string GetStorageIdentifier() const;
Paul Stewarta41e38d2011-11-11 07:47:29 -080072 static bool ParseStorageIdentifier(const std::string &storage_name,
73 std::string *address,
74 std::string *mode,
75 std::string *security);
Chris Masone34af2182011-08-22 11:59:36 -070076
Paul Stewart85aea152013-01-22 09:31:56 -080077 // Iterate over |storage| looking for WiFi servces with "old-style"
78 // properties that don't include explicit type/mode/security, and add
79 // these properties. Returns true if any entries were fixed.
80 static bool FixupServiceEntries(StoreInterface *storage);
81
Paul Stewartd2e1c362013-03-03 19:06:07 -080082 // Validate |mode| against all valid and supported service modes.
83 static bool IsValidMode(const std::string &mode);
84
Paul Stewart3c504012013-01-17 17:49:58 -080085 // Validate |method| against all valid and supported security methods.
86 static bool IsValidSecurityMethod(const std::string &method);
87
Thieu Le48e6d6d2011-12-06 00:40:27 +000088 const std::string &mode() const { return mode_; }
89 const std::string &key_management() const { return GetEAPKeyManagement(); }
90 const std::vector<uint8_t> &ssid() const { return ssid_; }
Christopher Wiley1057cd72013-02-28 15:21:29 -080091 const std::string &bssid() const { return bssid_; }
mukesh agrawalf6b32092013-04-10 15:49:55 -070092 uint16 physical_mode() const { return physical_mode_; }
mukesh agrawalb54601c2011-06-07 17:39:22 -070093
Paul Stewarte7de2942013-04-25 17:07:31 -070094 // WiFi services can load from profile entries other than their current
95 // storage identifier. Override the methods from the parent Service
96 // class which pertain to whether this service may be loaded from |storage|.
97 virtual std::string GetLoadableStorageIdentifier(
98 const StoreInterface &storage) const;
99 virtual bool IsLoadableFrom(const StoreInterface &storage) const;
100
Paul Stewartd08f4432011-11-04 07:48:20 -0700101 // Overrride Load and Save from parent Service class. We will call
102 // the parent method.
Paul Stewartd08f4432011-11-04 07:48:20 -0700103 virtual bool Load(StoreInterface *storage);
104 virtual bool Save(StoreInterface *storage);
Paul Stewart65512e12012-03-26 18:01:08 -0700105 virtual bool Unload();
Paul Stewartd08f4432011-11-04 07:48:20 -0700106
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000107 virtual bool HasEndpoints() const { return !endpoints_.empty(); }
Paul Stewarta41e38d2011-11-11 07:47:29 -0800108 virtual bool IsVisible() const;
Paul Stewart6ab23a92011-11-09 17:17:47 -0800109 bool IsSecurityMatch(const std::string &security) const;
Paul Stewartced6a0b2011-11-08 15:32:04 -0800110 bool hidden_ssid() const { return hidden_ssid_; }
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800111 bool ieee80211w_required() const { return ieee80211w_required_; }
Paul Stewartced6a0b2011-11-08 15:32:04 -0800112
Thieu Le48e6d6d2011-12-06 00:40:27 +0000113 virtual void InitializeCustomMetrics() const;
Thieu Leb84ba342012-03-02 15:15:19 -0800114 virtual void SendPostReadyStateMetrics(
115 int64 time_resume_to_ready_milliseconds) const;
Thieu Le48e6d6d2011-12-06 00:40:27 +0000116
Paul Stewart835934a2012-12-06 19:27:09 -0800117 // Clear any cached credentials stored in wpa_supplicant related to |this|.
118 // This will disconnect this service if it is currently connected.
119 void ClearCachedCredentials();
120
Gaurav Shah10109f22011-11-11 20:16:22 -0800121 // Override from parent Service class to correctly update connectability
122 // when the EAP credentials change for 802.1x networks.
Paul Stewartc43cbbe2013-04-11 06:29:30 -0700123 void OnEapCredentialsChanged();
Gaurav Shah10109f22011-11-11 20:16:22 -0800124
Paul Stewart4357f4e2012-04-26 17:39:26 -0700125 // Override from parent Service class to register hidden services once they
126 // have been configured.
127 virtual void OnProfileConfigured();
128
Paul Stewart3c504012013-01-17 17:49:58 -0800129 // Called by WiFiProvider to reset the WiFi device reference on shutdown.
130 virtual void ResetWiFi();
131
Paul Stewart08a54eb2013-03-11 12:07:36 -0700132 // "wpa", "rsn" and "psk" are equivalent from a configuration perspective.
133 // This function maps them all into "psk".
134 static std::string GetSecurityClass(const std::string &security);
135
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000136 protected:
mukesh agrawalbf14e942012-03-02 14:36:34 -0800137 virtual bool IsAutoConnectable(const char **reason) const;
mukesh agrawal43970a22013-02-15 16:00:07 -0800138 virtual void SetEAPKeyManagement(const std::string &key_management);
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000139
mukesh agrawalb54601c2011-06-07 17:39:22 -0700140 private:
Paul Stewartd08f4432011-11-04 07:48:20 -0700141 friend class WiFiServiceSecurityTest;
Paul Stewart0b950442012-09-11 13:10:08 -0700142 friend class WiFiServiceTest; // SetPassphrase
mukesh agrawale1d90e92012-02-15 17:36:08 -0800143 friend class WiFiServiceUpdateFromEndpointsTest; // SignalToStrength
Thieu Lead1ec2c2012-01-05 23:39:48 +0000144 FRIEND_TEST(MetricsTest, WiFiServicePostReady);
Paul Stewart21f40962013-03-01 14:27:28 -0800145 FRIEND_TEST(MetricsTest, WiFiServicePostReadyEAP);
Thieu Lee41a72d2012-02-06 20:46:51 +0000146 FRIEND_TEST(WiFiMainTest, CurrentBSSChangedUpdateServiceEndpoint);
Paul Stewart08a54eb2013-03-11 12:07:36 -0700147 FRIEND_TEST(WiFiProviderTest, OnEndpointAddedWithSecurity); // security_
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000148 FRIEND_TEST(WiFiServiceTest, AutoConnect);
mukesh agrawal8abd2f62012-01-30 14:56:14 -0800149 FRIEND_TEST(WiFiServiceTest, ClearWriteOnlyDerivedProperty); // passphrase_
mukesh agrawal43970a22013-02-15 16:00:07 -0800150 FRIEND_TEST(WiFiServiceTest, ComputeCipher8021x);
Gaurav Shah10109f22011-11-11 20:16:22 -0800151 FRIEND_TEST(WiFiServiceTest, ConnectTask8021x);
Gaurav Shah29d68882012-01-30 19:06:42 -0800152 FRIEND_TEST(WiFiServiceTest, ConnectTaskDynamicWEP);
Gaurav Shahf8721ee2011-11-07 09:12:46 -0800153 FRIEND_TEST(WiFiServiceTest, ConnectTaskPSK);
Gaurav Shah29d68882012-01-30 19:06:42 -0800154 FRIEND_TEST(WiFiServiceTest, ConnectTaskRSN);
Thieu Lef4cbda92011-11-10 23:41:24 +0000155 FRIEND_TEST(WiFiServiceTest, ConnectTaskWEP);
Gaurav Shah29d68882012-01-30 19:06:42 -0800156 FRIEND_TEST(WiFiServiceTest, ConnectTaskWPA);
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800157 FRIEND_TEST(WiFiServiceTest, ConnectTaskWPA80211w);
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000158 FRIEND_TEST(WiFiServiceTest, IsAutoConnectable);
Paul Stewartd08f4432011-11-04 07:48:20 -0700159 FRIEND_TEST(WiFiServiceTest, LoadHidden);
Paul Stewartd8ad3c42012-01-09 12:39:38 -0800160 FRIEND_TEST(WiFiServiceTest, LoadAndUnloadPassphrase);
Paul Stewart6df20bd2013-03-13 19:31:25 -0700161 FRIEND_TEST(WiFiServiceTest, SecurityFromCurrentEndpoint); // GetSecurity
Paul Stewart835934a2012-12-06 19:27:09 -0800162 FRIEND_TEST(WiFiServiceTest, SetPassphraseRemovesCachedCredentials);
mukesh agrawale1d90e92012-02-15 17:36:08 -0800163 FRIEND_TEST(WiFiServiceTest, SignalToStrength); // SignalToStrength
mukesh agrawal43970a22013-02-15 16:00:07 -0800164 FRIEND_TEST(WiFiServiceTest, UpdateSecurity); // SetEAPKeyManagement
Paul Stewartd08f4432011-11-04 07:48:20 -0700165
mukesh agrawalbf14e942012-03-02 14:36:34 -0800166 static const char kAutoConnNoEndpoint[];
Paul Stewart3c504012013-01-17 17:49:58 -0800167 static const char kAnyDeviceAddress[];
mukesh agrawalbf14e942012-03-02 14:36:34 -0800168
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800169 // Override the base clase implementation, because we need to allow
170 // arguments that aren't base class methods.
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700171 void HelpRegisterConstDerivedString(
172 const std::string &name,
173 std::string(WiFiService::*get)(Error *error));
Paul Stewart6df20bd2013-03-13 19:31:25 -0700174 void HelpRegisterDerivedString(
175 const std::string &name,
176 std::string(WiFiService::*get)(Error *error),
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700177 bool(WiFiService::*set)(const std::string &value, Error *error));
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800178 void HelpRegisterWriteOnlyDerivedString(
Thieu Lef7709452011-11-15 01:13:19 +0000179 const std::string &name,
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700180 bool(WiFiService::*set)(const std::string &value, Error *error),
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800181 void(WiFiService::*clear)(Error *error),
182 const std::string *default_value);
Thieu Lef7709452011-11-15 01:13:19 +0000183
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800184 std::string GetDeviceRpcId(Error *error);
mukesh agrawalf6b32092013-04-10 15:49:55 -0700185
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800186 void ClearPassphrase(Error *error);
mukesh agrawal29c13a12011-11-24 00:09:19 +0000187 void UpdateConnectable();
mukesh agrawale1d90e92012-02-15 17:36:08 -0800188 void UpdateFromEndpoints();
mukesh agrawal43970a22013-02-15 16:00:07 -0800189 void UpdateSecurity();
Chris Masone95207da2011-06-29 16:50:49 -0700190
mukesh agrawal43970a22013-02-15 16:00:07 -0800191 static CryptoAlgorithm ComputeCipher8021x(
192 const std::set<WiFiEndpointConstRefPtr> &endpoints);
Thieu Lef4cbda92011-11-10 23:41:24 +0000193 static void ValidateWEPPassphrase(const std::string &passphrase,
194 Error *error);
195 static void ValidateWPAPassphrase(const std::string &passphrase,
196 Error *error);
197 static void ParseWEPPassphrase(const std::string &passphrase,
198 int *key_index,
199 std::vector<uint8> *password_bytes,
200 Error *error);
mukesh agrawal1a056262011-10-05 14:36:54 -0700201 static bool CheckWEPIsHex(const std::string &passphrase, Error *error);
202 static bool CheckWEPKeyIndex(const std::string &passphrase, Error *error);
203 static bool CheckWEPPrefix(const std::string &passphrase, Error *error);
204
mukesh agrawal8f3f7752012-02-17 19:42:09 -0800205 // Maps a signal value, in dBm, to a "strength" value, from
206 // |Service::kStrengthMin| to |Service:kStrengthMax|.
mukesh agrawale1d90e92012-02-15 17:36:08 -0800207 static uint8 SignalToStrength(int16 signal_dbm);
208
Paul Stewart71a4d3b2013-01-18 18:12:56 -0800209 // Create a default group name for this WiFi service.
210 std::string GetDefaultStorageIdentifier() const;
211
Paul Stewart6df20bd2013-03-13 19:31:25 -0700212 // Return the security of this service. If connected, the security
213 // reported from the currently connected endpoint is returned. Otherwise
214 // the configured security for the service is returned.
215 std::string GetSecurity(Error *error);
216
Paul Stewartd08f4432011-11-04 07:48:20 -0700217 // Profile data for a WPA/RSN service can be stored under a number of
Paul Stewart71a4d3b2013-01-18 18:12:56 -0800218 // different security types. These functions create different storage
219 // property lists based on whether they are saved with their generic
220 // "psk" name or if they use the (legacy) specific "wpa" or "rsn" names.
221 KeyValueStore GetStorageProperties() const;
Paul Stewartd08f4432011-11-04 07:48:20 -0700222
Paul Stewart0b950442012-09-11 13:10:08 -0700223 // Validate then apply a passphrase for this service.
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700224 bool SetPassphrase(const std::string &passphrase, Error *error);
Paul Stewart0b950442012-09-11 13:10:08 -0700225
Paul Stewart3c504012013-01-17 17:49:58 -0800226 // Select a WiFi device (e.g, for connecting a hidden service with no
227 // endpoints).
228 WiFiRefPtr ChooseDevice();
229
mukesh agrawalcbfb34e2013-04-17 19:33:25 -0700230 void SetWiFi(const WiFiRefPtr &new_wifi);
Paul Stewart3c504012013-01-17 17:49:58 -0800231
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700232 // Properties
233 std::string passphrase_;
234 bool need_passphrase_;
mukesh agrawalcbfb34e2013-04-17 19:33:25 -0700235 const std::string security_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700236 // TODO(cmasone): see if the below can be pulled from the endpoint associated
237 // with this service instead.
Chris Masone092df3e2011-08-22 09:41:39 -0700238 const std::string mode_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700239 std::string auth_mode_;
240 bool hidden_ssid_;
241 uint16 frequency_;
242 uint16 physical_mode_;
Paul Stewart23b393a2012-09-25 21:21:06 -0700243 // The raw dBm signal strength from the associated endpoint.
244 int16 raw_signal_strength_;
mukesh agrawal32399322011-09-01 10:53:43 -0700245 std::string hex_ssid_;
Paul Stewartd08f4432011-11-04 07:48:20 -0700246 std::string storage_identifier_;
mukesh agrawal923f14f2012-06-04 16:46:08 -0700247 std::string bssid_;
Paul Stewart72b2fdc2012-06-02 08:58:51 -0700248 Stringmap vendor_information_;
Paul Stewartbdbd3c32013-04-17 09:47:21 -0700249 // The country code reported by the current endpoint.
250 std::string country_code_;
mukesh agrawal43970a22013-02-15 16:00:07 -0800251 // If |security_| == kSecurity8021x, the crypto algorithm being used.
252 // (Otherwise, crypto algorithm is implied by |security_|.)
253 CryptoAlgorithm cipher_8021x_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700254
mukesh agrawale1d90e92012-02-15 17:36:08 -0800255 // Track whether or not we've warned about large signal values.
256 // Used to avoid spamming the log.
257 static bool logged_signal_warning;
Chris Masone2b105542011-06-22 10:58:09 -0700258 WiFiRefPtr wifi_;
mukesh agrawal261daca2011-12-02 18:56:56 +0000259 std::set<WiFiEndpointConstRefPtr> endpoints_;
mukesh agrawale1d90e92012-02-15 17:36:08 -0800260 WiFiEndpointConstRefPtr current_endpoint_;
mukesh agrawalb54601c2011-06-07 17:39:22 -0700261 const std::vector<uint8_t> ssid_;
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800262 // Track whether IEEE 802.11w (Protected Management Frame) support is
263 // mandated by one or more endpoints we have seen that provide this service.
264 bool ieee80211w_required_;
Paul Stewartecf4cd12012-04-17 11:08:39 -0700265 NSS *nss_;
Paul Stewart5baebb72013-03-14 11:43:29 -0700266 scoped_ptr<CertificateFile> certificate_file_;
Paul Stewart3c504012013-01-17 17:49:58 -0800267 // Bare pointer is safe because WiFi service instances are owned by
268 // the WiFiProvider and are guaranteed to be deallocated by the time
269 // the WiFiProvider is.
270 WiFiProvider *provider_;
271
mukesh agrawalb54601c2011-06-07 17:39:22 -0700272 DISALLOW_COPY_AND_ASSIGN(WiFiService);
273};
274
275} // namespace shill
276
277#endif // SHILL_WIFI_SERVICE_