// Copyright 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // A wifi data provider provides wifi data from the device that is used by a // NetworkLocationProvider to obtain a position fix. We use a singleton // instance of the wifi data provider, which is used by multiple // NetworkLocationProvider objects. // // This file provides WifiDataProvider, which provides static methods to // access the singleton instance. The singleton instance uses a private // implementation to abstract across platforms and also to allow mock providers // to be used for testing. // // This file also provides WifiDataProviderImplBase, a base class which // provides common functionality for the private implementations. #ifndef CONTENT_BROWSER_GEOLOCATION_WIFI_DATA_PROVIDER_H_ #define CONTENT_BROWSER_GEOLOCATION_WIFI_DATA_PROVIDER_H_ #include <set> #include "base/basictypes.h" #include "base/bind.h" #include "base/callback.h" #include "base/memory/ref_counted.h" #include "base/message_loop/message_loop.h" #include "base/strings/string16.h" #include "base/strings/string_util.h" #include "content/browser/geolocation/wifi_data.h" #include "content/common/content_export.h" namespace content { class WifiDataProvider; // See class WifiDataProvider for the public client API. // WifiDataProvider uses containment to hide platform-specific implementation // details from common code. This class provides common functionality for these // contained implementation classes. This is a modified pimpl pattern. class CONTENT_EXPORT WifiDataProviderImplBase : public base::RefCountedThreadSafe<WifiDataProviderImplBase> { public: WifiDataProviderImplBase(); // Tells the provider to start looking for data. Callbacks will start // receiving notifications after this call. virtual void StartDataProvider() = 0; // Tells the provider to stop looking for data. Callbacks will stop // receiving notifications after this call. virtual void StopDataProvider() = 0; // Provides whatever data the provider has, which may be nothing. Return // value indicates whether this is all the data the provider could ever // obtain. virtual bool GetData(WifiData* data) = 0; // Sets the container of this class, which is of type WifiDataProvider. // This is required to pass as a parameter when calling a callback. void SetContainer(WifiDataProvider* container); typedef base::Callback<void(WifiDataProvider*)> WifiDataUpdateCallback; void AddCallback(WifiDataUpdateCallback* callback); bool RemoveCallback(WifiDataUpdateCallback* callback); bool has_callbacks() const; protected: friend class base::RefCountedThreadSafe<WifiDataProviderImplBase>; virtual ~WifiDataProviderImplBase(); typedef std::set<WifiDataUpdateCallback*> CallbackSet; // Runs all callbacks via a posted task, so we can unwind callstack here and // avoid client reentrancy. void RunCallbacks(); bool CalledOnClientThread() const; base::MessageLoop* client_loop() const; private: void DoRunCallbacks(); WifiDataProvider* container_; // Reference to the client's message loop. All callbacks should happen in this // context. base::MessageLoop* client_loop_; CallbackSet callbacks_; DISALLOW_COPY_AND_ASSIGN(WifiDataProviderImplBase); }; // A wifi data provider // // We use a singleton instance of this class which is shared by multiple network // location providers. These location providers access the instance through the // Register and Unregister methods. class CONTENT_EXPORT WifiDataProvider { public: // Sets the factory function which will be used by Register to create the // implementation used by the singleton instance. This factory approach is // used both to abstract accross platform-specific implementations and to // inject mock implementations for testing. typedef WifiDataProviderImplBase* (*ImplFactoryFunction)(void); static void SetFactory(ImplFactoryFunction factory_function_in); // Resets the factory function to the default. static void ResetFactory(); typedef base::Callback<void(WifiDataProvider*)> WifiDataUpdateCallback; // Registers a callback, which will be run whenever new data is available. // Instantiates the singleton if necessary, and always returns it. static WifiDataProvider* Register(WifiDataUpdateCallback* callback); // Removes a callback. If this is the last callback, deletes the singleton // instance. Return value indicates success. static bool Unregister(WifiDataUpdateCallback* callback); // Provides whatever data the provider has, which may be nothing. Return // value indicates whether this is all the data the provider could ever // obtain. bool GetData(WifiData* data); private: // Private constructor and destructor, callers access singleton through // Register and Unregister. WifiDataProvider(); virtual ~WifiDataProvider(); void AddCallback(WifiDataUpdateCallback* callback); bool RemoveCallback(WifiDataUpdateCallback* callback); bool has_callbacks() const; void StartDataProvider(); void StopDataProvider(); static WifiDataProviderImplBase* DefaultFactoryFunction(); // The singleton-like instance of this class. (Not 'true' singleton, as it // may go through multiple create/destroy/create cycles per process instance, // e.g. when under test). static WifiDataProvider* instance_; // The factory function used to create the singleton instance. static ImplFactoryFunction factory_function_; // The internal implementation. scoped_refptr<WifiDataProviderImplBase> impl_; DISALLOW_COPY_AND_ASSIGN(WifiDataProvider); }; } // namespace content #endif // CONTENT_BROWSER_GEOLOCATION_WIFI_DATA_PROVIDER_H_