1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "device/test/usb_test_gadget.h"
10 #include "base/command_line.h"
11 #include "base/compiler_specific.h"
12 #include "base/files/file.h"
13 #include "base/files/file_path.h"
14 #include "base/logging.h"
15 #include "base/macros.h"
16 #include "base/memory/ref_counted.h"
17 #include "base/memory/scoped_ptr.h"
18 #include "base/path_service.h"
19 #include "base/process/process_handle.h"
20 #include "base/run_loop.h"
21 #include "base/scoped_observer.h"
22 #include "base/strings/stringprintf.h"
23 #include "base/strings/utf_string_conversions.h"
24 #include "base/thread_task_runner_handle.h"
25 #include "base/time/time.h"
26 #include "device/usb/usb_device.h"
27 #include "device/usb/usb_device_handle.h"
28 #include "device/usb/usb_service.h"
29 #include "net/proxy/proxy_service.h"
30 #include "net/url_request/url_fetcher.h"
31 #include "net/url_request/url_fetcher_delegate.h"
32 #include "net/url_request/url_request_context.h"
33 #include "net/url_request/url_request_context_builder.h"
34 #include "net/url_request/url_request_context_getter.h"
39 class UsbTestGadgetImpl
: public UsbTestGadget
{
42 scoped_refptr
<net::URLRequestContextGetter
> request_context_getter
,
43 UsbService
* usb_service
,
44 scoped_refptr
<UsbDevice
> device
);
45 ~UsbTestGadgetImpl() override
;
47 bool Unclaim() override
;
48 bool Disconnect() override
;
49 bool Reconnect() override
;
50 bool SetType(Type type
) override
;
51 UsbDevice
* GetDevice() const override
;
54 std::string device_address_
;
55 scoped_refptr
<UsbDevice
> device_
;
56 scoped_refptr
<net::URLRequestContextGetter
> request_context_getter_
;
57 UsbService
* usb_service_
;
59 DISALLOW_COPY_AND_ASSIGN(UsbTestGadgetImpl
);
64 static const char kCommandLineSwitch
[] = "enable-gadget-tests";
65 static const int kReenumeratePeriod
= 100; // 0.1 seconds
67 struct UsbTestGadgetConfiguration
{
68 UsbTestGadget::Type type
;
69 const char* http_resource
;
73 static const struct UsbTestGadgetConfiguration kConfigurations
[] = {
74 {UsbTestGadget::DEFAULT
, "/unconfigure", 0x58F0},
75 {UsbTestGadget::KEYBOARD
, "/keyboard/configure", 0x58F1},
76 {UsbTestGadget::MOUSE
, "/mouse/configure", 0x58F2},
77 {UsbTestGadget::HID_ECHO
, "/hid_echo/configure", 0x58F3},
78 {UsbTestGadget::ECHO
, "/echo/configure", 0x58F4},
81 bool ReadFile(const base::FilePath
& file_path
, std::string
* content
) {
82 base::File
file(file_path
, base::File::FLAG_OPEN
| base::File::FLAG_READ
);
83 if (!file
.IsValid()) {
84 LOG(ERROR
) << "Cannot open " << file_path
.MaybeAsASCII() << ": "
85 << base::File::ErrorToString(file
.error_details());
89 STLClearObject(content
);
93 rv
= file
.ReadAtCurrentPos(buf
, sizeof buf
);
95 LOG(ERROR
) << "Cannot read " << file_path
.MaybeAsASCII() << ": "
96 << base::File::ErrorToString(file
.error_details());
99 content
->append(buf
, rv
);
105 bool ReadLocalVersion(std::string
* version
) {
106 base::FilePath file_path
;
107 CHECK(PathService::Get(base::DIR_EXE
, &file_path
));
108 file_path
= file_path
.AppendASCII("usb_gadget.zip.md5");
110 return ReadFile(file_path
, version
);
113 bool ReadLocalPackage(std::string
* package
) {
114 base::FilePath file_path
;
115 CHECK(PathService::Get(base::DIR_EXE
, &file_path
));
116 file_path
= file_path
.AppendASCII("usb_gadget.zip");
118 return ReadFile(file_path
, package
);
121 scoped_ptr
<net::URLFetcher
> CreateURLFetcher(
122 scoped_refptr
<net::URLRequestContextGetter
> request_context_getter
,
124 net::URLFetcher::RequestType request_type
,
125 net::URLFetcherDelegate
* delegate
) {
126 scoped_ptr
<net::URLFetcher
> url_fetcher(
127 net::URLFetcher::Create(url
, request_type
, delegate
));
129 url_fetcher
->SetRequestContext(request_context_getter
.get());
134 class URLRequestContextGetter
: public net::URLRequestContextGetter
{
136 URLRequestContextGetter(
137 scoped_refptr
<base::SingleThreadTaskRunner
> network_task_runner
)
138 : network_task_runner_(network_task_runner
) {}
141 ~URLRequestContextGetter() override
{}
143 // net::URLRequestContextGetter implementation
144 net::URLRequestContext
* GetURLRequestContext() override
{
145 context_builder_
.set_proxy_service(net::ProxyService::CreateDirect());
146 return context_builder_
.Build();
149 scoped_refptr
<base::SingleThreadTaskRunner
> GetNetworkTaskRunner()
151 return network_task_runner_
;
154 net::URLRequestContextBuilder context_builder_
;
155 scoped_refptr
<base::SingleThreadTaskRunner
> network_task_runner_
;
158 class URLFetcherDelegate
: public net::URLFetcherDelegate
{
160 URLFetcherDelegate() {}
161 ~URLFetcherDelegate() override
{}
163 void WaitForCompletion() { run_loop_
.Run(); }
165 void OnURLFetchComplete(const net::URLFetcher
* source
) override
{
170 base::RunLoop run_loop_
;
172 DISALLOW_COPY_AND_ASSIGN(URLFetcherDelegate
);
175 int SimplePOSTRequest(
176 scoped_refptr
<net::URLRequestContextGetter
> request_context_getter
,
178 const std::string
& form_data
) {
179 URLFetcherDelegate delegate
;
180 scoped_ptr
<net::URLFetcher
> url_fetcher
= CreateURLFetcher(
181 request_context_getter
, url
, net::URLFetcher::POST
, &delegate
);
183 url_fetcher
->SetUploadData("application/x-www-form-urlencoded", form_data
);
184 url_fetcher
->Start();
185 delegate
.WaitForCompletion();
187 return url_fetcher
->GetResponseCode();
190 class UsbGadgetFactory
: public UsbService::Observer
,
191 public net::URLFetcherDelegate
{
193 UsbGadgetFactory(scoped_refptr
<base::SingleThreadTaskRunner
> io_task_runner
)
194 : observer_(this), weak_factory_(this) {
195 usb_service_
= UsbService::GetInstance(io_task_runner
);
196 request_context_getter_
= new URLRequestContextGetter(io_task_runner
);
198 static uint32 next_session_id
;
199 base::ProcessId process_id
= base::GetCurrentProcId();
200 session_id_
= base::StringPrintf("%d-%d", process_id
, next_session_id
++);
202 observer_
.Add(usb_service_
);
205 ~UsbGadgetFactory() override
{}
207 scoped_ptr
<UsbTestGadget
> WaitForDevice() {
210 return make_scoped_ptr(
211 new UsbTestGadgetImpl(request_context_getter_
, usb_service_
, device_
));
215 void EnumerateDevices() {
217 usb_service_
->GetDevices(base::Bind(
218 &UsbGadgetFactory::OnDevicesEnumerated
, weak_factory_
.GetWeakPtr()));
222 void OnDevicesEnumerated(
223 const std::vector
<scoped_refptr
<UsbDevice
>>& devices
) {
224 for (const scoped_refptr
<UsbDevice
>& device
: devices
) {
225 OnDeviceAdded(device
);
229 // TODO(reillyg): This timer could be replaced by a way to use long-
230 // polling to wait for claimed devices to become unclaimed.
231 base::MessageLoop::current()->PostDelayedTask(
232 FROM_HERE
, base::Bind(&UsbGadgetFactory::EnumerateDevices
,
233 weak_factory_
.GetWeakPtr()),
234 base::TimeDelta::FromMilliseconds(kReenumeratePeriod
));
238 void OnDeviceAdded(scoped_refptr
<UsbDevice
> device
) override
{
240 // Already trying to claim a device.
244 if (device
->vendor_id() != 0x18D1 || device
->product_id() != 0x58F0 ||
245 device
->serial_number().empty()) {
249 std::string serial_number
= base::UTF16ToUTF8(device
->serial_number());
250 if (serial_number
== serial_number_
) {
251 // We were waiting for the device to reappear after upgrade.
258 serial_number_
= serial_number
;
263 VLOG(1) << "Trying to claim " << serial_number_
<< ".";
265 GURL
url("http://" + serial_number_
+ "/claim");
266 std::string form_data
= base::StringPrintf(
267 "session_id=%s", net::EscapeUrlEncodedData(session_id_
, true).c_str());
268 url_fetcher_
= CreateURLFetcher(request_context_getter_
, url
,
269 net::URLFetcher::POST
, this);
270 url_fetcher_
->SetUploadData("application/x-www-form-urlencoded", form_data
);
271 url_fetcher_
->Start();
275 GURL
url("http://" + serial_number_
+ "/version");
276 url_fetcher_
= CreateURLFetcher(request_context_getter_
, url
,
277 net::URLFetcher::GET
, this);
278 url_fetcher_
->Start();
281 bool Update(const std::string
& version
) {
282 LOG(INFO
) << "Updating " << serial_number_
<< " to " << version
<< "...";
284 GURL
url("http://" + serial_number_
+ "/update");
285 url_fetcher_
= CreateURLFetcher(request_context_getter_
, url
,
286 net::URLFetcher::POST
, this);
287 std::string mime_header
= base::StringPrintf(
289 "Content-Disposition: form-data; name=\"file\"; "
290 "filename=\"usb_gadget-%s.zip\"\r\n"
291 "Content-Type: application/octet-stream\r\n"
294 std::string
mime_footer("\r\n--foo--\r\n");
297 if (!ReadLocalPackage(&package
)) {
301 url_fetcher_
->SetUploadData("multipart/form-data; boundary=foo",
302 mime_header
+ package
+ mime_footer
);
303 url_fetcher_
->Start();
308 void OnURLFetchComplete(const net::URLFetcher
* source
) override
{
309 DCHECK(!serial_number_
.empty());
311 int response_code
= source
->GetResponseCode();
313 // Just completed a /claim request.
314 if (response_code
== 200) {
318 if (response_code
!= 403) {
319 LOG(WARNING
) << "Unexpected HTTP " << response_code
324 } else if (version_
.empty()) {
325 // Just completed a /version request.
326 if (response_code
!= 200) {
327 LOG(WARNING
) << "Unexpected HTTP " << response_code
328 << " from /version.";
333 if (!source
->GetResponseAsString(&version_
)) {
334 LOG(WARNING
) << "Failed to read body from /version.";
339 std::string local_version
;
340 if (!ReadLocalVersion(&local_version
)) {
345 if (version_
== local_version
) {
348 if (!Update(local_version
)) {
353 // Just completed an /update request.
354 if (response_code
!= 200) {
355 LOG(WARNING
) << "Unexpected HTTP " << response_code
<< " from /update.";
360 // Must wait for the device to reconnect.
366 serial_number_
.clear();
370 // Wait a bit and then try again to find an available device.
371 base::MessageLoop::current()->PostDelayedTask(
372 FROM_HERE
, base::Bind(&UsbGadgetFactory::EnumerateDevices
,
373 weak_factory_
.GetWeakPtr()),
374 base::TimeDelta::FromMilliseconds(kReenumeratePeriod
));
377 UsbService
* usb_service_
= nullptr;
378 scoped_refptr
<net::URLRequestContextGetter
> request_context_getter_
;
379 std::string session_id_
;
380 scoped_ptr
<net::URLFetcher
> url_fetcher_
;
381 scoped_refptr
<UsbDevice
> device_
;
382 std::string serial_number_
;
383 bool claimed_
= false;
384 std::string version_
;
385 base::RunLoop run_loop_
;
386 ScopedObserver
<UsbService
, UsbService::Observer
> observer_
;
387 base::WeakPtrFactory
<UsbGadgetFactory
> weak_factory_
;
390 class DeviceAddListener
: public UsbService::Observer
{
392 DeviceAddListener(UsbService
* usb_service
,
393 const std::string
& serial_number
,
395 : usb_service_(usb_service
),
396 serial_number_(serial_number
),
397 product_id_(product_id
),
399 weak_factory_(this) {
400 observer_
.Add(usb_service_
);
402 virtual ~DeviceAddListener() {}
404 scoped_refptr
<UsbDevice
> WaitForAdd() {
405 usb_service_
->GetDevices(base::Bind(&DeviceAddListener::OnDevicesEnumerated
,
406 weak_factory_
.GetWeakPtr()));
412 void OnDevicesEnumerated(
413 const std::vector
<scoped_refptr
<UsbDevice
>>& devices
) {
414 for (const scoped_refptr
<UsbDevice
>& device
: devices
) {
415 OnDeviceAdded(device
);
419 void OnDeviceAdded(scoped_refptr
<UsbDevice
> device
) override
{
420 if (device
->vendor_id() == 0x18D1 && !device
->serial_number().empty()) {
421 const uint16 product_id
= device
->product_id();
422 if (product_id_
== -1) {
424 for (size_t i
= 0; i
< arraysize(kConfigurations
); ++i
) {
425 if (product_id
== kConfigurations
[i
].product_id
) {
434 if (product_id_
!= product_id
) {
439 if (serial_number_
!= base::UTF16ToUTF8(device
->serial_number())) {
448 UsbService
* usb_service_
;
449 const std::string serial_number_
;
450 const int product_id_
;
451 base::RunLoop run_loop_
;
452 scoped_refptr
<UsbDevice
> device_
;
453 ScopedObserver
<UsbService
, UsbService::Observer
> observer_
;
454 base::WeakPtrFactory
<DeviceAddListener
> weak_factory_
;
456 DISALLOW_COPY_AND_ASSIGN(DeviceAddListener
);
459 class DeviceRemoveListener
: public UsbService::Observer
{
461 DeviceRemoveListener(UsbService
* usb_service
, scoped_refptr
<UsbDevice
> device
)
462 : usb_service_(usb_service
),
465 weak_factory_(this) {
466 observer_
.Add(usb_service_
);
468 virtual ~DeviceRemoveListener() {}
470 void WaitForRemove() {
471 usb_service_
->GetDevices(
472 base::Bind(&DeviceRemoveListener::OnDevicesEnumerated
,
473 weak_factory_
.GetWeakPtr()));
478 void OnDevicesEnumerated(
479 const std::vector
<scoped_refptr
<UsbDevice
>>& devices
) {
481 for (const scoped_refptr
<UsbDevice
>& device
: devices
) {
482 if (device_
== device
) {
491 void OnDeviceRemoved(scoped_refptr
<UsbDevice
> device
) override
{
492 if (device_
== device
) {
497 UsbService
* usb_service_
;
498 base::RunLoop run_loop_
;
499 scoped_refptr
<UsbDevice
> device_
;
500 ScopedObserver
<UsbService
, UsbService::Observer
> observer_
;
501 base::WeakPtrFactory
<DeviceRemoveListener
> weak_factory_
;
503 DISALLOW_COPY_AND_ASSIGN(DeviceRemoveListener
);
508 bool UsbTestGadget::IsTestEnabled() {
509 base::CommandLine
* command_line
= base::CommandLine::ForCurrentProcess();
510 return command_line
->HasSwitch(kCommandLineSwitch
);
513 scoped_ptr
<UsbTestGadget
> UsbTestGadget::Claim(
514 scoped_refptr
<base::SingleThreadTaskRunner
> io_task_runner
) {
515 UsbGadgetFactory
gadget_factory(io_task_runner
);
516 return gadget_factory
.WaitForDevice().Pass();
519 UsbTestGadgetImpl::UsbTestGadgetImpl(
520 scoped_refptr
<net::URLRequestContextGetter
> request_context_getter_
,
521 UsbService
* usb_service
,
522 scoped_refptr
<UsbDevice
> device
)
523 : device_address_(base::UTF16ToUTF8(device
->serial_number())),
525 request_context_getter_(request_context_getter_
),
526 usb_service_(usb_service
) {
529 UsbTestGadgetImpl::~UsbTestGadgetImpl() {
530 if (!device_address_
.empty()) {
535 UsbDevice
* UsbTestGadgetImpl::GetDevice() const {
536 return device_
.get();
539 bool UsbTestGadgetImpl::Unclaim() {
540 VLOG(1) << "Releasing the device at " << device_address_
<< ".";
542 GURL
url("http://" + device_address_
+ "/unclaim");
543 int response_code
= SimplePOSTRequest(request_context_getter_
, url
, "");
545 if (response_code
!= 200) {
546 LOG(ERROR
) << "Unexpected HTTP " << response_code
<< " from /unclaim.";
550 device_address_
.clear();
554 bool UsbTestGadgetImpl::SetType(Type type
) {
555 const struct UsbTestGadgetConfiguration
* config
= NULL
;
556 for (size_t i
= 0; i
< arraysize(kConfigurations
); ++i
) {
557 if (kConfigurations
[i
].type
== type
) {
558 config
= &kConfigurations
[i
];
563 GURL
url("http://" + device_address_
+ config
->http_resource
);
564 int response_code
= SimplePOSTRequest(request_context_getter_
, url
, "");
566 if (response_code
!= 200) {
567 LOG(ERROR
) << "Unexpected HTTP " << response_code
568 << " from " << config
->http_resource
<< ".";
572 // Release the old reference to the device and try to open a new one.
573 DeviceAddListener
add_listener(usb_service_
, device_address_
,
575 device_
= add_listener
.WaitForAdd();
576 DCHECK(device_
.get());
580 bool UsbTestGadgetImpl::Disconnect() {
581 GURL
url("http://" + device_address_
+ "/disconnect");
582 int response_code
= SimplePOSTRequest(request_context_getter_
, url
, "");
584 if (response_code
!= 200) {
585 LOG(ERROR
) << "Unexpected HTTP " << response_code
<< " from " << url
<< ".";
589 // Release the old reference to the device and wait until it can't be found.
590 DeviceRemoveListener
remove_listener(usb_service_
, device_
);
591 remove_listener
.WaitForRemove();
596 bool UsbTestGadgetImpl::Reconnect() {
597 GURL
url("http://" + device_address_
+ "/reconnect");
598 int response_code
= SimplePOSTRequest(request_context_getter_
, url
, "");
600 if (response_code
!= 200) {
601 LOG(ERROR
) << "Unexpected HTTP " << response_code
<< " from " << url
<< ".";
605 DeviceAddListener
add_listener(usb_service_
, device_address_
, -1);
606 device_
= add_listener
.WaitForAdd();
607 DCHECK(device_
.get());
611 } // namespace device