diff options
author | jorgelo <jorgelo@chromium.org> | 2014-10-20 15:42:36 -0700 |
---|---|---|
committer | Commit bot <commit-bot@chromium.org> | 2014-10-20 22:43:47 +0000 |
commit | 2aa4e40e4bac2c708c0e91c9a7c03fb9539323fd (patch) | |
tree | ace1b85d15e3a367700ea6360d749a57f0e08438 | |
parent | f937604c661f75a423b6e1c0a285eff9a8556577 (diff) | |
download | chromium_src-2aa4e40e4bac2c708c0e91c9a7c03fb9539323fd.zip chromium_src-2aa4e40e4bac2c708c0e91c9a7c03fb9539323fd.tar.gz chromium_src-2aa4e40e4bac2c708c0e91c9a7c03fb9539323fd.tar.bz2 |
Implement permission_broker for serial devices.
This change makes Chrome on Chrome OS ask permission_broker for access to serial devices.
BUG=173548
TEST=Chrome can access serial devices on Linux, chromeos=1, full Chrome OS.
Review URL: https://codereview.chromium.org/646053002
Cr-Commit-Position: refs/heads/master@{#300364}
-rw-r--r-- | device/serial/serial_io_handler.cc | 42 | ||||
-rw-r--r-- | device/serial/serial_io_handler.h | 19 | ||||
-rw-r--r-- | device/serial/serial_io_handler_posix.cc | 50 | ||||
-rw-r--r-- | device/serial/serial_io_handler_posix.h | 7 | ||||
-rw-r--r-- | device/serial/serial_io_handler_win.cc | 11 | ||||
-rw-r--r-- | device/serial/serial_io_handler_win.h | 3 | ||||
-rw-r--r-- | device/serial/serial_service_impl.cc | 14 | ||||
-rw-r--r-- | device/serial/serial_service_impl.h | 2 | ||||
-rw-r--r-- | device/serial/serial_service_unittest.cc | 2 | ||||
-rw-r--r-- | device/serial/test_serial_io_handler.cc | 2 | ||||
-rw-r--r-- | extensions/browser/api/serial/serial_connection.cc | 4 |
11 files changed, 126 insertions, 30 deletions
diff --git a/device/serial/serial_io_handler.cc b/device/serial/serial_io_handler.cc index 341df29..a9cb52f 100644 --- a/device/serial/serial_io_handler.cc +++ b/device/serial/serial_io_handler.cc @@ -12,8 +12,10 @@ namespace device { SerialIoHandler::SerialIoHandler( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop) - : file_thread_message_loop_(file_thread_message_loop) { + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop) + : file_thread_message_loop_(file_thread_message_loop), + ui_thread_message_loop_(ui_thread_message_loop) { } SerialIoHandler::~SerialIoHandler() { @@ -27,12 +29,36 @@ void SerialIoHandler::Open(const std::string& port, DCHECK(open_complete_.is_null()); open_complete_ = callback; DCHECK(file_thread_message_loop_.get()); - file_thread_message_loop_->PostTask( - FROM_HERE, - base::Bind(&SerialIoHandler::StartOpen, - this, - port, - base::MessageLoopProxy::current())); + DCHECK(ui_thread_message_loop_.get()); + RequestAccess(port, file_thread_message_loop_, ui_thread_message_loop_); +} + +void SerialIoHandler::RequestAccess( + const std::string& port, + scoped_refptr<base::MessageLoopProxy> file_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop) { + OnRequestAccessComplete(port, true /* success */); +} + +void SerialIoHandler::OnRequestAccessComplete(const std::string& port, + bool success) { + DCHECK(CalledOnValidThread()); + if (success) { + DCHECK(file_thread_message_loop_.get()); + file_thread_message_loop_->PostTask( + FROM_HERE, + base::Bind(&SerialIoHandler::StartOpen, + this, + port, + base::MessageLoopProxy::current())); + return; + } else { + DCHECK(!open_complete_.is_null()); + OpenCompleteCallback callback = open_complete_; + open_complete_.Reset(); + callback.Run(false); + return; + } } void SerialIoHandler::StartOpen( diff --git a/device/serial/serial_io_handler.h b/device/serial/serial_io_handler.h index 5b2d059..46e0ed0 100644 --- a/device/serial/serial_io_handler.h +++ b/device/serial/serial_io_handler.h @@ -24,7 +24,8 @@ class SerialIoHandler : public base::NonThreadSafe, public: // Constructs an instance of some platform-specific subclass. static scoped_refptr<SerialIoHandler> Create( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop); + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop); typedef base::Callback<void(bool success)> OpenCompleteCallback; @@ -32,6 +33,9 @@ class SerialIoHandler : public base::NonThreadSafe, virtual void Open(const std::string& port, const OpenCompleteCallback& callback); + // Signals that the access request for |port| is complete. + void OnRequestAccessComplete(const std::string& port, bool success); + // Performs an async Read operation. Behavior is undefined if this is called // while a Read is already pending. Otherwise, the Done or DoneWithError // method on |buffer| will eventually be called with a result. @@ -79,7 +83,8 @@ class SerialIoHandler : public base::NonThreadSafe, protected: explicit SerialIoHandler( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop); + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop); virtual ~SerialIoHandler(); // Performs a platform-specific read operation. This must guarantee that @@ -92,7 +97,7 @@ class SerialIoHandler : public base::NonThreadSafe, // Performs a platform-specific write operation. This must guarantee that // WriteCompleted is called when the underlying async operation is completed // or the SerialIoHandler instance will leak. - // NOTE: Implementations of Writempl should never call WriteCompleted + // NOTE: Implementations of WriteImpl should never call WriteCompleted // directly. Use QueueWriteCompleted instead to avoid reentrancy. virtual void WriteImpl() = 0; @@ -102,6 +107,12 @@ class SerialIoHandler : public base::NonThreadSafe, // Platform-specific write cancelation. virtual void CancelWriteImpl() = 0; + // Requests access to the underlying serial device, if needed. + virtual void RequestAccess( + const std::string& port, + scoped_refptr<base::MessageLoopProxy> file_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop); + // Performs platform-specific, one-time port configuration on open. virtual bool PostOpen(); @@ -187,6 +198,8 @@ class SerialIoHandler : public base::NonThreadSafe, OpenCompleteCallback open_complete_; scoped_refptr<base::MessageLoopProxy> file_thread_message_loop_; + // On Chrome OS, PermissionBrokerClient should be called on the UI thread. + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop_; DISALLOW_COPY_AND_ASSIGN(SerialIoHandler); }; diff --git a/device/serial/serial_io_handler_posix.cc b/device/serial/serial_io_handler_posix.cc index a215661..020ee4c 100644 --- a/device/serial/serial_io_handler_posix.cc +++ b/device/serial/serial_io_handler_posix.cc @@ -11,7 +11,13 @@ #if defined(OS_LINUX) #include <linux/serial.h> -#endif +#if defined(OS_CHROMEOS) +#include "base/bind.h" +#include "base/sys_info.h" +#include "chromeos/dbus/dbus_thread_manager.h" +#include "chromeos/dbus/permission_broker_client.h" +#endif // defined(OS_CHROMEOS) +#endif // defined(OS_LINUX) namespace { @@ -123,8 +129,41 @@ namespace device { // static scoped_refptr<SerialIoHandler> SerialIoHandler::Create( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop) { - return new SerialIoHandlerPosix(file_thread_message_loop); + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop) { + return new SerialIoHandlerPosix(file_thread_message_loop, + ui_thread_message_loop); +} + +void SerialIoHandlerPosix::RequestAccess( + const std::string& port, + scoped_refptr<base::MessageLoopProxy> file_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop) { +#if defined(OS_LINUX) && defined(OS_CHROMEOS) + if (base::SysInfo::IsRunningOnChromeOS()) { + chromeos::PermissionBrokerClient* client = + chromeos::DBusThreadManager::Get()->GetPermissionBrokerClient(); + if (!client) { + DVLOG(1) << "Could not get permission_broker client."; + OnRequestAccessComplete(port, false /* failure */); + return; + } + // PermissionBrokerClient should be called on the UI thread. + ui_message_loop->PostTask( + FROM_HERE, + base::Bind( + &chromeos::PermissionBrokerClient::RequestPathAccess, + base::Unretained(client), + port, + -1, + base::Bind(&SerialIoHandler::OnRequestAccessComplete, this, port))); + } else { + OnRequestAccessComplete(port, true /* success */); + return; + } +#else + OnRequestAccessComplete(port, true /* success */); +#endif // defined(OS_LINUX) && defined(OS_CHROMEOS) } void SerialIoHandlerPosix::ReadImpl() { @@ -158,8 +197,9 @@ void SerialIoHandlerPosix::CancelWriteImpl() { } SerialIoHandlerPosix::SerialIoHandlerPosix( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop) - : SerialIoHandler(file_thread_message_loop), + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop) + : SerialIoHandler(file_thread_message_loop, ui_thread_message_loop), is_watching_reads_(false), is_watching_writes_(false) { } diff --git a/device/serial/serial_io_handler_posix.h b/device/serial/serial_io_handler_posix.h index ef4bd37..7a6d97c 100644 --- a/device/serial/serial_io_handler_posix.h +++ b/device/serial/serial_io_handler_posix.h @@ -24,13 +24,18 @@ class SerialIoHandlerPosix : public SerialIoHandler, const serial::HostControlSignals& control_signals) override; virtual bool ConfigurePort(const serial::ConnectionOptions& options) override; virtual serial::ConnectionInfoPtr GetPortInfo() const override; + virtual void RequestAccess( + const std::string& port, + scoped_refptr<base::MessageLoopProxy> file_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop) override; virtual bool PostOpen() override; private: friend class SerialIoHandler; SerialIoHandlerPosix( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop); + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop); virtual ~SerialIoHandlerPosix(); // base::MessageLoopForIO::Watcher implementation. diff --git a/device/serial/serial_io_handler_win.cc b/device/serial/serial_io_handler_win.cc index ac669ad..17d7db1 100644 --- a/device/serial/serial_io_handler_win.cc +++ b/device/serial/serial_io_handler_win.cc @@ -134,8 +134,10 @@ serial::StopBits StopBitsConstantToEnum(int stop_bits) { // static scoped_refptr<SerialIoHandler> SerialIoHandler::Create( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop) { - return new SerialIoHandlerWin(file_thread_message_loop); + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop) { + return new SerialIoHandlerWin(file_thread_message_loop, + ui_thread_message_loop); } bool SerialIoHandlerWin::PostOpen() { @@ -239,8 +241,9 @@ void SerialIoHandlerWin::CancelWriteImpl() { } SerialIoHandlerWin::SerialIoHandlerWin( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop) - : SerialIoHandler(file_thread_message_loop), + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop) + : SerialIoHandler(file_thread_message_loop, ui_thread_message_loop), event_mask_(0), is_comm_pending_(false) { } diff --git a/device/serial/serial_io_handler_win.h b/device/serial/serial_io_handler_win.h index ed45f5b..6ffaea3 100644 --- a/device/serial/serial_io_handler_win.h +++ b/device/serial/serial_io_handler_win.h @@ -31,7 +31,8 @@ class SerialIoHandlerWin : public SerialIoHandler, friend class SerialIoHandler; explicit SerialIoHandlerWin( - scoped_refptr<base::MessageLoopProxy> file_thread_message_loop); + scoped_refptr<base::MessageLoopProxy> file_thread_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_thread_message_loop); virtual ~SerialIoHandlerWin(); // base::MessageLoopForIO::IOHandler implementation. diff --git a/device/serial/serial_service_impl.cc b/device/serial/serial_service_impl.cc index 0f87695..fceb448 100644 --- a/device/serial/serial_service_impl.cc +++ b/device/serial/serial_service_impl.cc @@ -28,10 +28,12 @@ SerialServiceImpl::~SerialServiceImpl() { // static void SerialServiceImpl::Create( scoped_refptr<base::MessageLoopProxy> io_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop, mojo::InterfaceRequest<serial::SerialService> request) { mojo::BindToRequest(new SerialServiceImpl(new SerialConnectionFactory( base::Bind(SerialIoHandler::Create, - base::MessageLoopProxy::current()), + base::MessageLoopProxy::current(), + ui_message_loop), io_message_loop)), &request); } @@ -40,11 +42,13 @@ void SerialServiceImpl::Create( void SerialServiceImpl::CreateOnMessageLoop( scoped_refptr<base::MessageLoopProxy> message_loop, scoped_refptr<base::MessageLoopProxy> io_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop, mojo::InterfaceRequest<serial::SerialService> request) { - message_loop->PostTask( - FROM_HERE, - base::Bind( - &SerialServiceImpl::Create, io_message_loop, base::Passed(&request))); + message_loop->PostTask(FROM_HERE, + base::Bind(&SerialServiceImpl::Create, + io_message_loop, + ui_message_loop, + base::Passed(&request))); } void SerialServiceImpl::GetDevices( diff --git a/device/serial/serial_service_impl.h b/device/serial/serial_service_impl.h index b1c6612..796a1d4 100644 --- a/device/serial/serial_service_impl.h +++ b/device/serial/serial_service_impl.h @@ -24,10 +24,12 @@ class SerialServiceImpl : public mojo::InterfaceImpl<serial::SerialService> { virtual ~SerialServiceImpl(); static void Create(scoped_refptr<base::MessageLoopProxy> io_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop, mojo::InterfaceRequest<serial::SerialService> request); static void CreateOnMessageLoop( scoped_refptr<base::MessageLoopProxy> message_loop, scoped_refptr<base::MessageLoopProxy> io_message_loop, + scoped_refptr<base::MessageLoopProxy> ui_message_loop, mojo::InterfaceRequest<serial::SerialService> request); // mojo::InterfaceImpl<SerialService> overrides. diff --git a/device/serial/serial_service_unittest.cc b/device/serial/serial_service_unittest.cc index 20ebe3a..24558ae 100644 --- a/device/serial/serial_service_unittest.cc +++ b/device/serial/serial_service_unittest.cc @@ -113,7 +113,7 @@ class SerialServiceTest : public testing::Test, public mojo::ErrorHandler { TEST_F(SerialServiceTest, GetDevices) { mojo::InterfacePtr<serial::SerialService> service; - SerialServiceImpl::Create(NULL, mojo::GetProxy(&service)); + SerialServiceImpl::Create(NULL, NULL, mojo::GetProxy(&service)); service.set_error_handler(this); mojo::Array<serial::DeviceInfoPtr> result; service->GetDevices( diff --git a/device/serial/test_serial_io_handler.cc b/device/serial/test_serial_io_handler.cc index f4cc3bd..8ec5648 100644 --- a/device/serial/test_serial_io_handler.cc +++ b/device/serial/test_serial_io_handler.cc @@ -12,7 +12,7 @@ namespace device { TestSerialIoHandler::TestSerialIoHandler() - : SerialIoHandler(NULL), + : SerialIoHandler(NULL, NULL), opened_(false), dtr_(false), rts_(false), diff --git a/extensions/browser/api/serial/serial_connection.cc b/extensions/browser/api/serial/serial_connection.cc index 131aa61..f1963e6 100644 --- a/extensions/browser/api/serial/serial_connection.cc +++ b/extensions/browser/api/serial/serial_connection.cc @@ -204,7 +204,9 @@ SerialConnection::SerialConnection(const std::string& port, paused_(false), io_handler_(device::SerialIoHandler::Create( content::BrowserThread::GetMessageLoopProxyForThread( - content::BrowserThread::FILE))) { + content::BrowserThread::FILE), + content::BrowserThread::GetMessageLoopProxyForThread( + content::BrowserThread::UI))) { DCHECK_CURRENTLY_ON(BrowserThread::IO); } |