blob: 4e342c8ec6b68fa80637c24447176305ff06d026 [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
Paul Stewartee6b3d72013-07-12 16:07:51 -070034 // crbug.com/208736
Paul Stewart0756db92012-01-27 08:34:47 -080035 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 agrawale7c7e652013-06-18 17:19:39 -070092 const std::vector<uint16> &frequency_list() const { return frequency_list_; }
mukesh agrawalf6b32092013-04-10 15:49:55 -070093 uint16 physical_mode() const { return physical_mode_; }
mukesh agrawala5dda0e2013-08-16 11:53:10 -070094 uint16 frequency() const { return frequency_; }
mukesh agrawalb54601c2011-06-07 17:39:22 -070095
Paul Stewarte7de2942013-04-25 17:07:31 -070096 // WiFi services can load from profile entries other than their current
97 // storage identifier. Override the methods from the parent Service
98 // class which pertain to whether this service may be loaded from |storage|.
99 virtual std::string GetLoadableStorageIdentifier(
100 const StoreInterface &storage) const;
101 virtual bool IsLoadableFrom(const StoreInterface &storage) const;
102
Paul Stewartd08f4432011-11-04 07:48:20 -0700103 // Overrride Load and Save from parent Service class. We will call
104 // the parent method.
Paul Stewartd08f4432011-11-04 07:48:20 -0700105 virtual bool Load(StoreInterface *storage);
106 virtual bool Save(StoreInterface *storage);
Paul Stewart65512e12012-03-26 18:01:08 -0700107 virtual bool Unload();
Paul Stewartd08f4432011-11-04 07:48:20 -0700108
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000109 virtual bool HasEndpoints() const { return !endpoints_.empty(); }
Paul Stewarta41e38d2011-11-11 07:47:29 -0800110 virtual bool IsVisible() const;
Paul Stewart6ab23a92011-11-09 17:17:47 -0800111 bool IsSecurityMatch(const std::string &security) const;
Paul Stewartbca08f82013-07-09 16:32:37 -0700112
113 // Used by WiFi objects to indicate that the credentials for this network
114 // have been called into question. This method returns true if given this
115 // suspicion, if it is probable that indeed these credentials are likely
116 // to be incorrect. Credentials that have never been used before are
117 // considered suspect by default, while those which have been used
118 // successfully in the past must have this method called a number of times
119 // since the last time ResetSuspectedCredentialsFailures() was called.
120 virtual bool AddSuspectedCredentialFailure();
121 virtual void ResetSuspectedCredentialFailures();
122
Paul Stewartced6a0b2011-11-08 15:32:04 -0800123 bool hidden_ssid() const { return hidden_ssid_; }
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800124 bool ieee80211w_required() const { return ieee80211w_required_; }
Paul Stewartced6a0b2011-11-08 15:32:04 -0800125
mukesh agrawal6cfe53f2013-08-13 13:39:01 -0700126 void InitializeCustomMetrics() const;
Thieu Leb84ba342012-03-02 15:15:19 -0800127 virtual void SendPostReadyStateMetrics(
128 int64 time_resume_to_ready_milliseconds) const;
Thieu Le48e6d6d2011-12-06 00:40:27 +0000129
Paul Stewart835934a2012-12-06 19:27:09 -0800130 // Clear any cached credentials stored in wpa_supplicant related to |this|.
131 // This will disconnect this service if it is currently connected.
132 void ClearCachedCredentials();
133
Gaurav Shah10109f22011-11-11 20:16:22 -0800134 // Override from parent Service class to correctly update connectability
135 // when the EAP credentials change for 802.1x networks.
Paul Stewartc43cbbe2013-04-11 06:29:30 -0700136 void OnEapCredentialsChanged();
Gaurav Shah10109f22011-11-11 20:16:22 -0800137
Paul Stewart4357f4e2012-04-26 17:39:26 -0700138 // Override from parent Service class to register hidden services once they
139 // have been configured.
140 virtual void OnProfileConfigured();
141
Paul Stewart3c504012013-01-17 17:49:58 -0800142 // Called by WiFiProvider to reset the WiFi device reference on shutdown.
143 virtual void ResetWiFi();
144
Paul Stewart08a54eb2013-03-11 12:07:36 -0700145 // "wpa", "rsn" and "psk" are equivalent from a configuration perspective.
146 // This function maps them all into "psk".
147 static std::string GetSecurityClass(const std::string &security);
148
Wade Guthrie9ec08062013-09-25 15:22:24 -0700149 // Signal level in dBm. If no current endpoint, returns
150 // std::numeric_limits<int>::min().
151 int16 SignalLevel() const;
152
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000153 protected:
mukesh agrawalbf14e942012-03-02 14:36:34 -0800154 virtual bool IsAutoConnectable(const char **reason) const;
mukesh agrawal43970a22013-02-15 16:00:07 -0800155 virtual void SetEAPKeyManagement(const std::string &key_management);
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000156
mukesh agrawalb54601c2011-06-07 17:39:22 -0700157 private:
Paul Stewartd08f4432011-11-04 07:48:20 -0700158 friend class WiFiServiceSecurityTest;
Paul Stewart0b950442012-09-11 13:10:08 -0700159 friend class WiFiServiceTest; // SetPassphrase
mukesh agrawale1d90e92012-02-15 17:36:08 -0800160 friend class WiFiServiceUpdateFromEndpointsTest; // SignalToStrength
Thieu Lead1ec2c2012-01-05 23:39:48 +0000161 FRIEND_TEST(MetricsTest, WiFiServicePostReady);
Paul Stewarte4cedde2013-07-17 08:56:44 -0700162 FRIEND_TEST(MetricsTest, WiFiServicePostReadyAdHoc);
Paul Stewart21f40962013-03-01 14:27:28 -0800163 FRIEND_TEST(MetricsTest, WiFiServicePostReadyEAP);
Thieu Lee41a72d2012-02-06 20:46:51 +0000164 FRIEND_TEST(WiFiMainTest, CurrentBSSChangedUpdateServiceEndpoint);
Paul Stewart08a54eb2013-03-11 12:07:36 -0700165 FRIEND_TEST(WiFiProviderTest, OnEndpointAddedWithSecurity); // security_
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000166 FRIEND_TEST(WiFiServiceTest, AutoConnect);
mukesh agrawal8abd2f62012-01-30 14:56:14 -0800167 FRIEND_TEST(WiFiServiceTest, ClearWriteOnlyDerivedProperty); // passphrase_
mukesh agrawal43970a22013-02-15 16:00:07 -0800168 FRIEND_TEST(WiFiServiceTest, ComputeCipher8021x);
Gaurav Shah10109f22011-11-11 20:16:22 -0800169 FRIEND_TEST(WiFiServiceTest, ConnectTask8021x);
Gaurav Shah29d68882012-01-30 19:06:42 -0800170 FRIEND_TEST(WiFiServiceTest, ConnectTaskDynamicWEP);
Gaurav Shahf8721ee2011-11-07 09:12:46 -0800171 FRIEND_TEST(WiFiServiceTest, ConnectTaskPSK);
Gaurav Shah29d68882012-01-30 19:06:42 -0800172 FRIEND_TEST(WiFiServiceTest, ConnectTaskRSN);
Thieu Lef4cbda92011-11-10 23:41:24 +0000173 FRIEND_TEST(WiFiServiceTest, ConnectTaskWEP);
Gaurav Shah29d68882012-01-30 19:06:42 -0800174 FRIEND_TEST(WiFiServiceTest, ConnectTaskWPA);
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800175 FRIEND_TEST(WiFiServiceTest, ConnectTaskWPA80211w);
mukesh agrawal8a3188d2011-12-01 20:56:44 +0000176 FRIEND_TEST(WiFiServiceTest, IsAutoConnectable);
Paul Stewartd08f4432011-11-04 07:48:20 -0700177 FRIEND_TEST(WiFiServiceTest, LoadHidden);
Paul Stewartd8ad3c42012-01-09 12:39:38 -0800178 FRIEND_TEST(WiFiServiceTest, LoadAndUnloadPassphrase);
Paul Stewart6df20bd2013-03-13 19:31:25 -0700179 FRIEND_TEST(WiFiServiceTest, SecurityFromCurrentEndpoint); // GetSecurity
Paul Stewart835934a2012-12-06 19:27:09 -0800180 FRIEND_TEST(WiFiServiceTest, SetPassphraseRemovesCachedCredentials);
mukesh agrawale1d90e92012-02-15 17:36:08 -0800181 FRIEND_TEST(WiFiServiceTest, SignalToStrength); // SignalToStrength
Paul Stewartbca08f82013-07-09 16:32:37 -0700182 FRIEND_TEST(WiFiServiceTest, SuspectedCredentialFailure);
mukesh agrawal43970a22013-02-15 16:00:07 -0800183 FRIEND_TEST(WiFiServiceTest, UpdateSecurity); // SetEAPKeyManagement
Paul Stewartd08f4432011-11-04 07:48:20 -0700184
mukesh agrawalbf14e942012-03-02 14:36:34 -0800185 static const char kAutoConnNoEndpoint[];
Paul Stewart3c504012013-01-17 17:49:58 -0800186 static const char kAnyDeviceAddress[];
Paul Stewartbca08f82013-07-09 16:32:37 -0700187 static const int kSuspectedCredentialFailureThreshold;
mukesh agrawalbf14e942012-03-02 14:36:34 -0800188
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800189 // Override the base clase implementation, because we need to allow
190 // arguments that aren't base class methods.
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700191 void HelpRegisterConstDerivedString(
192 const std::string &name,
193 std::string(WiFiService::*get)(Error *error));
Paul Stewart6df20bd2013-03-13 19:31:25 -0700194 void HelpRegisterDerivedString(
195 const std::string &name,
196 std::string(WiFiService::*get)(Error *error),
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700197 bool(WiFiService::*set)(const std::string &value, Error *error));
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800198 void HelpRegisterWriteOnlyDerivedString(
Thieu Lef7709452011-11-15 01:13:19 +0000199 const std::string &name,
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700200 bool(WiFiService::*set)(const std::string &value, Error *error),
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800201 void(WiFiService::*clear)(Error *error),
202 const std::string *default_value);
Thieu Lef7709452011-11-15 01:13:19 +0000203
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800204 std::string GetDeviceRpcId(Error *error);
mukesh agrawalf6b32092013-04-10 15:49:55 -0700205
mukesh agrawal292dc0f2012-01-26 18:02:46 -0800206 void ClearPassphrase(Error *error);
mukesh agrawal29c13a12011-11-24 00:09:19 +0000207 void UpdateConnectable();
mukesh agrawale1d90e92012-02-15 17:36:08 -0800208 void UpdateFromEndpoints();
mukesh agrawal43970a22013-02-15 16:00:07 -0800209 void UpdateSecurity();
Chris Masone95207da2011-06-29 16:50:49 -0700210
mukesh agrawal43970a22013-02-15 16:00:07 -0800211 static CryptoAlgorithm ComputeCipher8021x(
212 const std::set<WiFiEndpointConstRefPtr> &endpoints);
Thieu Lef4cbda92011-11-10 23:41:24 +0000213 static void ValidateWEPPassphrase(const std::string &passphrase,
214 Error *error);
215 static void ValidateWPAPassphrase(const std::string &passphrase,
216 Error *error);
217 static void ParseWEPPassphrase(const std::string &passphrase,
218 int *key_index,
219 std::vector<uint8> *password_bytes,
220 Error *error);
mukesh agrawal1a056262011-10-05 14:36:54 -0700221 static bool CheckWEPIsHex(const std::string &passphrase, Error *error);
222 static bool CheckWEPKeyIndex(const std::string &passphrase, Error *error);
223 static bool CheckWEPPrefix(const std::string &passphrase, Error *error);
224
mukesh agrawal8f3f7752012-02-17 19:42:09 -0800225 // Maps a signal value, in dBm, to a "strength" value, from
226 // |Service::kStrengthMin| to |Service:kStrengthMax|.
mukesh agrawale1d90e92012-02-15 17:36:08 -0800227 static uint8 SignalToStrength(int16 signal_dbm);
228
Paul Stewart71a4d3b2013-01-18 18:12:56 -0800229 // Create a default group name for this WiFi service.
230 std::string GetDefaultStorageIdentifier() const;
231
Paul Stewart6df20bd2013-03-13 19:31:25 -0700232 // Return the security of this service. If connected, the security
233 // reported from the currently connected endpoint is returned. Otherwise
234 // the configured security for the service is returned.
235 std::string GetSecurity(Error *error);
236
Paul Stewartd08f4432011-11-04 07:48:20 -0700237 // Profile data for a WPA/RSN service can be stored under a number of
Paul Stewart71a4d3b2013-01-18 18:12:56 -0800238 // different security types. These functions create different storage
239 // property lists based on whether they are saved with their generic
240 // "psk" name or if they use the (legacy) specific "wpa" or "rsn" names.
241 KeyValueStore GetStorageProperties() const;
Paul Stewartd08f4432011-11-04 07:48:20 -0700242
Paul Stewart0b950442012-09-11 13:10:08 -0700243 // Validate then apply a passphrase for this service.
mukesh agrawalbebf1b82013-04-23 15:06:33 -0700244 bool SetPassphrase(const std::string &passphrase, Error *error);
Paul Stewart0b950442012-09-11 13:10:08 -0700245
Paul Stewart3c504012013-01-17 17:49:58 -0800246 // Select a WiFi device (e.g, for connecting a hidden service with no
247 // endpoints).
248 WiFiRefPtr ChooseDevice();
249
mukesh agrawalcbfb34e2013-04-17 19:33:25 -0700250 void SetWiFi(const WiFiRefPtr &new_wifi);
Paul Stewart3c504012013-01-17 17:49:58 -0800251
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700252 // Properties
253 std::string passphrase_;
254 bool need_passphrase_;
mukesh agrawalcbfb34e2013-04-17 19:33:25 -0700255 const std::string security_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700256 // TODO(cmasone): see if the below can be pulled from the endpoint associated
257 // with this service instead.
Chris Masone092df3e2011-08-22 09:41:39 -0700258 const std::string mode_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700259 std::string auth_mode_;
260 bool hidden_ssid_;
261 uint16 frequency_;
mukesh agrawale7c7e652013-06-18 17:19:39 -0700262 std::vector<uint16> frequency_list_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700263 uint16 physical_mode_;
Paul Stewart23b393a2012-09-25 21:21:06 -0700264 // The raw dBm signal strength from the associated endpoint.
265 int16 raw_signal_strength_;
mukesh agrawal32399322011-09-01 10:53:43 -0700266 std::string hex_ssid_;
Paul Stewartd08f4432011-11-04 07:48:20 -0700267 std::string storage_identifier_;
mukesh agrawal923f14f2012-06-04 16:46:08 -0700268 std::string bssid_;
Paul Stewart72b2fdc2012-06-02 08:58:51 -0700269 Stringmap vendor_information_;
Paul Stewartbdbd3c32013-04-17 09:47:21 -0700270 // The country code reported by the current endpoint.
271 std::string country_code_;
mukesh agrawal43970a22013-02-15 16:00:07 -0800272 // If |security_| == kSecurity8021x, the crypto algorithm being used.
273 // (Otherwise, crypto algorithm is implied by |security_|.)
274 CryptoAlgorithm cipher_8021x_;
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700275
Paul Stewartbca08f82013-07-09 16:32:37 -0700276 // Track the number of consecutive times our current credentials have
277 // been called into question.
278 int suspected_credential_failures_;
279
mukesh agrawale1d90e92012-02-15 17:36:08 -0800280 // Track whether or not we've warned about large signal values.
281 // Used to avoid spamming the log.
282 static bool logged_signal_warning;
Paul Stewartbca08f82013-07-09 16:32:37 -0700283
Chris Masone2b105542011-06-22 10:58:09 -0700284 WiFiRefPtr wifi_;
mukesh agrawal261daca2011-12-02 18:56:56 +0000285 std::set<WiFiEndpointConstRefPtr> endpoints_;
mukesh agrawale1d90e92012-02-15 17:36:08 -0800286 WiFiEndpointConstRefPtr current_endpoint_;
mukesh agrawalb54601c2011-06-07 17:39:22 -0700287 const std::vector<uint8_t> ssid_;
Paul Stewarta5e7d5f2013-01-09 18:06:15 -0800288 // Track whether IEEE 802.11w (Protected Management Frame) support is
289 // mandated by one or more endpoints we have seen that provide this service.
290 bool ieee80211w_required_;
Paul Stewartecf4cd12012-04-17 11:08:39 -0700291 NSS *nss_;
Paul Stewart5baebb72013-03-14 11:43:29 -0700292 scoped_ptr<CertificateFile> certificate_file_;
Paul Stewart3c504012013-01-17 17:49:58 -0800293 // Bare pointer is safe because WiFi service instances are owned by
294 // the WiFiProvider and are guaranteed to be deallocated by the time
295 // the WiFiProvider is.
296 WiFiProvider *provider_;
297
mukesh agrawalb54601c2011-06-07 17:39:22 -0700298 DISALLOW_COPY_AND_ASSIGN(WiFiService);
299};
300
301} // namespace shill
302
303#endif // SHILL_WIFI_SERVICE_