blob: dc21c0a8176eb00d169eaea43f772a2a1792dca9 [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 Stewartd08f4432011-11-04 07:48:20 -070094 // Overrride Load and Save from parent Service class. We will call
95 // the parent method.
96 virtual bool IsLoadableFrom(StoreInterface *storage) const;
97 virtual bool Load(StoreInterface *storage);
98 virtual bool Save(StoreInterface *storage);
Paul Stewart65512e12012-03-26 18:01:08 -070099 virtual bool Unload();
Paul Stewartd08f4432011-11-04 07:48:20 -0700100
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000101 virtual bool HasEndpoints() const { return !endpoints_.empty(); }
Paul Stewarta41e38d2011-11-11 07:47:29 -0800102 virtual bool IsVisible() const;
Paul Stewart6ab23a92011-11-09 17:17:47 -0800103 bool IsSecurityMatch(const std::string &security) const;
Paul Stewartced6a0b2011-11-08 15:32:04 -0800104 bool hidden_ssid() const { return hidden_ssid_; }
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800105 bool ieee80211w_required() const { return ieee80211w_required_; }
Paul Stewartced6a0b2011-11-08 15:32:04 -0800106
Thieu Le48e6d6d2011-12-06 00:40:27 +0000107 virtual void InitializeCustomMetrics() const;
Thieu Leb84ba342012-03-02 15:15:19 -0800108 virtual void SendPostReadyStateMetrics(
109 int64 time_resume_to_ready_milliseconds) const;
Thieu Le48e6d6d2011-12-06 00:40:27 +0000110
Paul Stewart835934a2012-12-06 19:27:09 -0800111 // Clear any cached credentials stored in wpa_supplicant related to |this|.
112 // This will disconnect this service if it is currently connected.
113 void ClearCachedCredentials();
114
Gaurav Shah10109f22011-11-11 20:16:22 -0800115 // Override from parent Service class to correctly update connectability
116 // when the EAP credentials change for 802.1x networks.
Paul Stewartc43cbbe2013-04-11 06:29:30 -0700117 void OnEapCredentialsChanged();
Gaurav Shah10109f22011-11-11 20:16:22 -0800118
Paul Stewart4357f4e2012-04-26 17:39:26 -0700119 // Override from parent Service class to register hidden services once they
120 // have been configured.
121 virtual void OnProfileConfigured();
122
Paul Stewart3c504012013-01-17 17:49:58 -0800123 // Called by WiFiProvider to reset the WiFi device reference on shutdown.
124 virtual void ResetWiFi();
125
Paul Stewart08a54eb2013-03-11 12:07:36 -0700126 // "wpa", "rsn" and "psk" are equivalent from a configuration perspective.
127 // This function maps them all into "psk".
128 static std::string GetSecurityClass(const std::string &security);
129
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000130 protected:
mukesh agrawalbf14e942012-03-02 14:36:34 -0800131 virtual bool IsAutoConnectable(const char **reason) const;
mukesh agrawal43970a22013-02-15 16:00:07 -0800132 virtual void SetEAPKeyManagement(const std::string &key_management);
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000133
mukesh agrawalb54601c2011-06-07 17:39:22 -0700134 private:
Paul Stewartd08f4432011-11-04 07:48:20 -0700135 friend class WiFiServiceSecurityTest;
Paul Stewart0b950442012-09-11 13:10:08 -0700136 friend class WiFiServiceTest; // SetPassphrase
mukesh agrawale1d90e92012-02-15 17:36:08 -0800137 friend class WiFiServiceUpdateFromEndpointsTest; // SignalToStrength
Thieu Lead1ec2c2012-01-05 23:39:48 +0000138 FRIEND_TEST(MetricsTest, WiFiServicePostReady);
Paul Stewart21f40962013-03-01 14:27:28 -0800139 FRIEND_TEST(MetricsTest, WiFiServicePostReadyEAP);
Thieu Lee41a72d2012-02-06 20:46:51 +0000140 FRIEND_TEST(WiFiMainTest, CurrentBSSChangedUpdateServiceEndpoint);
Paul Stewart08a54eb2013-03-11 12:07:36 -0700141 FRIEND_TEST(WiFiProviderTest, OnEndpointAddedWithSecurity); // security_
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000142 FRIEND_TEST(WiFiServiceTest, AutoConnect);
mukesh agrawal8abd2f62012-01-30 14:56:14 -0800143 FRIEND_TEST(WiFiServiceTest, ClearWriteOnlyDerivedProperty); // passphrase_
mukesh agrawal43970a22013-02-15 16:00:07 -0800144 FRIEND_TEST(WiFiServiceTest, ComputeCipher8021x);
Gaurav Shah10109f22011-11-11 20:16:22 -0800145 FRIEND_TEST(WiFiServiceTest, ConnectTask8021x);
Gaurav Shah29d68882012-01-30 19:06:42 -0800146 FRIEND_TEST(WiFiServiceTest, ConnectTaskDynamicWEP);
Gaurav Shahf8721ee2011-11-07 09:12:46 -0800147 FRIEND_TEST(WiFiServiceTest, ConnectTaskPSK);
Gaurav Shah29d68882012-01-30 19:06:42 -0800148 FRIEND_TEST(WiFiServiceTest, ConnectTaskRSN);
Thieu Lef4cbda92011-11-10 23:41:24 +0000149 FRIEND_TEST(WiFiServiceTest, ConnectTaskWEP);
Gaurav Shah29d68882012-01-30 19:06:42 -0800150 FRIEND_TEST(WiFiServiceTest, ConnectTaskWPA);
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800151 FRIEND_TEST(WiFiServiceTest, ConnectTaskWPA80211w);
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000152 FRIEND_TEST(WiFiServiceTest, IsAutoConnectable);
Paul Stewartd08f4432011-11-04 07:48:20 -0700153 FRIEND_TEST(WiFiServiceTest, LoadHidden);
Paul Stewartd8ad3c42012-01-09 12:39:38 -0800154 FRIEND_TEST(WiFiServiceTest, LoadAndUnloadPassphrase);
Paul Stewart6df20bd2013-03-13 19:31:25 -0700155 FRIEND_TEST(WiFiServiceTest, SecurityFromCurrentEndpoint); // GetSecurity
Paul Stewart835934a2012-12-06 19:27:09 -0800156 FRIEND_TEST(WiFiServiceTest, SetPassphraseRemovesCachedCredentials);
mukesh agrawale1d90e92012-02-15 17:36:08 -0800157 FRIEND_TEST(WiFiServiceTest, SignalToStrength); // SignalToStrength
mukesh agrawal43970a22013-02-15 16:00:07 -0800158 FRIEND_TEST(WiFiServiceTest, UpdateSecurity); // SetEAPKeyManagement
Paul Stewartd08f4432011-11-04 07:48:20 -0700159
mukesh agrawalbf14e942012-03-02 14:36:34 -0800160 static const char kAutoConnNoEndpoint[];
Paul Stewart3c504012013-01-17 17:49:58 -0800161 static const char kAnyDeviceAddress[];
mukesh agrawalbf14e942012-03-02 14:36:34 -0800162
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800163 // Override the base clase implementation, because we need to allow
164 // arguments that aren't base class methods.
Paul Stewart6df20bd2013-03-13 19:31:25 -0700165 void HelpRegisterDerivedString(
166 const std::string &name,
167 std::string(WiFiService::*get)(Error *error),
168 void(WiFiService::*set)(const std::string &value, Error *error));
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800169 void HelpRegisterWriteOnlyDerivedString(
Thieu Lef7709452011-11-15 01:13:19 +0000170 const std::string &name,
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800171 void(WiFiService::*set)(const std::string &value, Error *error),
172 void(WiFiService::*clear)(Error *error),
173 const std::string *default_value);
Thieu Lef7709452011-11-15 01:13:19 +0000174
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800175 std::string GetDeviceRpcId(Error *error);
mukesh agrawalf6b32092013-04-10 15:49:55 -0700176
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800177 void ClearPassphrase(Error *error);
mukesh agrawal29c13a12011-11-24 00:09:19 +0000178 void UpdateConnectable();
mukesh agrawale1d90e92012-02-15 17:36:08 -0800179 void UpdateFromEndpoints();
mukesh agrawal43970a22013-02-15 16:00:07 -0800180 void UpdateSecurity();
Chris Masone95207da2011-06-29 16:50:49 -0700181
mukesh agrawal43970a22013-02-15 16:00:07 -0800182 static CryptoAlgorithm ComputeCipher8021x(
183 const std::set<WiFiEndpointConstRefPtr> &endpoints);
Thieu Lef4cbda92011-11-10 23:41:24 +0000184 static void ValidateWEPPassphrase(const std::string &passphrase,
185 Error *error);
186 static void ValidateWPAPassphrase(const std::string &passphrase,
187 Error *error);
188 static void ParseWEPPassphrase(const std::string &passphrase,
189 int *key_index,
190 std::vector<uint8> *password_bytes,
191 Error *error);
mukesh agrawal1a056262011-10-05 14:36:54 -0700192 static bool CheckWEPIsHex(const std::string &passphrase, Error *error);
193 static bool CheckWEPKeyIndex(const std::string &passphrase, Error *error);
194 static bool CheckWEPPrefix(const std::string &passphrase, Error *error);
195
mukesh agrawal8f3f7752012-02-17 19:42:09 -0800196 // Maps a signal value, in dBm, to a "strength" value, from
197 // |Service::kStrengthMin| to |Service:kStrengthMax|.
mukesh agrawale1d90e92012-02-15 17:36:08 -0800198 static uint8 SignalToStrength(int16 signal_dbm);
199
Paul Stewart71a4d3b2013-01-18 18:12:56 -0800200 // Create a default group name for this WiFi service.
201 std::string GetDefaultStorageIdentifier() const;
202
Paul Stewart6df20bd2013-03-13 19:31:25 -0700203 // Return the security of this service. If connected, the security
204 // reported from the currently connected endpoint is returned. Otherwise
205 // the configured security for the service is returned.
206 std::string GetSecurity(Error *error);
207
Paul Stewartd08f4432011-11-04 07:48:20 -0700208 // Profile data for a WPA/RSN service can be stored under a number of
Paul Stewart71a4d3b2013-01-18 18:12:56 -0800209 // different security types. These functions create different storage
210 // property lists based on whether they are saved with their generic
211 // "psk" name or if they use the (legacy) specific "wpa" or "rsn" names.
212 KeyValueStore GetStorageProperties() const;
Paul Stewartd08f4432011-11-04 07:48:20 -0700213
Paul Stewart0b950442012-09-11 13:10:08 -0700214 // Validate then apply a passphrase for this service.
215 void SetPassphrase(const std::string &passphrase, Error *error);
216
Paul Stewart3c504012013-01-17 17:49:58 -0800217 // Select a WiFi device (e.g, for connecting a hidden service with no
218 // endpoints).
219 WiFiRefPtr ChooseDevice();
220
221 void SetWiFi(const WiFiRefPtr &wifi);
222
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700223 // Properties
224 std::string passphrase_;
225 bool need_passphrase_;
226 std::string security_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700227 // TODO(cmasone): see if the below can be pulled from the endpoint associated
228 // with this service instead.
Chris Masone092df3e2011-08-22 09:41:39 -0700229 const std::string mode_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700230 std::string auth_mode_;
231 bool hidden_ssid_;
232 uint16 frequency_;
233 uint16 physical_mode_;
Paul Stewart23b393a2012-09-25 21:21:06 -0700234 // The raw dBm signal strength from the associated endpoint.
235 int16 raw_signal_strength_;
mukesh agrawal32399322011-09-01 10:53:43 -0700236 std::string hex_ssid_;
Paul Stewartd08f4432011-11-04 07:48:20 -0700237 std::string storage_identifier_;
mukesh agrawal923f14f2012-06-04 16:46:08 -0700238 std::string bssid_;
Paul Stewart72b2fdc2012-06-02 08:58:51 -0700239 Stringmap vendor_information_;
Paul Stewartbdbd3c32013-04-17 09:47:21 -0700240 // The country code reported by the current endpoint.
241 std::string country_code_;
mukesh agrawal43970a22013-02-15 16:00:07 -0800242 // If |security_| == kSecurity8021x, the crypto algorithm being used.
243 // (Otherwise, crypto algorithm is implied by |security_|.)
244 CryptoAlgorithm cipher_8021x_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700245
mukesh agrawale1d90e92012-02-15 17:36:08 -0800246 // Track whether or not we've warned about large signal values.
247 // Used to avoid spamming the log.
248 static bool logged_signal_warning;
Chris Masone2b105542011-06-22 10:58:09 -0700249 WiFiRefPtr wifi_;
mukesh agrawal261daca2011-12-02 18:56:56 +0000250 std::set<WiFiEndpointConstRefPtr> endpoints_;
mukesh agrawale1d90e92012-02-15 17:36:08 -0800251 WiFiEndpointConstRefPtr current_endpoint_;
mukesh agrawalb54601c2011-06-07 17:39:22 -0700252 const std::vector<uint8_t> ssid_;
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800253 // Track whether IEEE 802.11w (Protected Management Frame) support is
254 // mandated by one or more endpoints we have seen that provide this service.
255 bool ieee80211w_required_;
Paul Stewartecf4cd12012-04-17 11:08:39 -0700256 NSS *nss_;
Paul Stewart5baebb72013-03-14 11:43:29 -0700257 scoped_ptr<CertificateFile> certificate_file_;
Paul Stewart3c504012013-01-17 17:49:58 -0800258 // Bare pointer is safe because WiFi service instances are owned by
259 // the WiFiProvider and are guaranteed to be deallocated by the time
260 // the WiFiProvider is.
261 WiFiProvider *provider_;
262
mukesh agrawalb54601c2011-06-07 17:39:22 -0700263 DISALLOW_COPY_AND_ASSIGN(WiFiService);
264};
265
266} // namespace shill
267
268#endif // SHILL_WIFI_SERVICE_