Fix some case-insensitive cases for StartsWith.
[chromium-blink-merge.git] / components / storage_monitor / portable_device_watcher_win.cc
blob29b77ffb69feaa48ef4bebef6c1bc29e2271d240
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.
4 //
5 // Any tasks that communicates with the portable device may take >100ms to
6 // complete. Those tasks should be run on an blocking thread instead of the
7 // UI thread.
9 #include "components/storage_monitor/portable_device_watcher_win.h"
11 #include <dbt.h>
12 #include <portabledevice.h>
14 #include "base/files/file_path.h"
15 #include "base/logging.h"
16 #include "base/stl_util.h"
17 #include "base/strings/string_util.h"
18 #include "base/strings/utf_string_conversions.h"
19 #include "base/threading/sequenced_worker_pool.h"
20 #include "base/win/scoped_co_mem.h"
21 #include "base/win/scoped_comptr.h"
22 #include "base/win/scoped_propvariant.h"
23 #include "components/storage_monitor/media_storage_util.h"
24 #include "components/storage_monitor/removable_device_constants.h"
25 #include "components/storage_monitor/storage_info.h"
26 #include "content/public/browser/browser_thread.h"
28 namespace storage_monitor {
30 namespace {
32 // Name of the client application that communicates with the MTP device.
33 const base::char16 kClientName[] = L"Chromium";
35 // Name of the sequenced task runner.
36 const char kMediaTaskRunnerName[] = "media-task-runner";
38 // Returns true if |data| represents a class of portable devices.
39 bool IsPortableDeviceStructure(LPARAM data) {
40 DEV_BROADCAST_HDR* broadcast_hdr =
41 reinterpret_cast<DEV_BROADCAST_HDR*>(data);
42 if (!broadcast_hdr ||
43 (broadcast_hdr->dbch_devicetype != DBT_DEVTYP_DEVICEINTERFACE)) {
44 return false;
47 GUID guidDevInterface = GUID_NULL;
48 if (FAILED(CLSIDFromString(kWPDDevInterfaceGUID, &guidDevInterface)))
49 return false;
50 DEV_BROADCAST_DEVICEINTERFACE* dev_interface =
51 reinterpret_cast<DEV_BROADCAST_DEVICEINTERFACE*>(data);
52 return (IsEqualGUID(dev_interface->dbcc_classguid, guidDevInterface) != 0);
55 // Returns the portable device plug and play device ID string.
56 base::string16 GetPnpDeviceId(LPARAM data) {
57 DEV_BROADCAST_DEVICEINTERFACE* dev_interface =
58 reinterpret_cast<DEV_BROADCAST_DEVICEINTERFACE*>(data);
59 if (!dev_interface)
60 return base::string16();
61 base::string16 device_id(dev_interface->dbcc_name);
62 DCHECK(base::IsStringASCII(device_id));
63 return base::StringToLowerASCII(device_id);
66 // Gets the friendly name of the device specified by the |pnp_device_id|. On
67 // success, returns true and fills in |name|.
68 bool GetFriendlyName(const base::string16& pnp_device_id,
69 IPortableDeviceManager* device_manager,
70 base::string16* name) {
71 DCHECK(device_manager);
72 DCHECK(name);
73 DWORD name_len = 0;
74 HRESULT hr = device_manager->GetDeviceFriendlyName(pnp_device_id.c_str(),
75 NULL, &name_len);
76 if (FAILED(hr))
77 return false;
79 hr = device_manager->GetDeviceFriendlyName(
80 pnp_device_id.c_str(), WriteInto(name, name_len), &name_len);
81 return (SUCCEEDED(hr) && !name->empty());
84 // Gets the manufacturer name of the device specified by the |pnp_device_id|.
85 // On success, returns true and fills in |name|.
86 bool GetManufacturerName(const base::string16& pnp_device_id,
87 IPortableDeviceManager* device_manager,
88 base::string16* name) {
89 DCHECK(device_manager);
90 DCHECK(name);
91 DWORD name_len = 0;
92 HRESULT hr = device_manager->GetDeviceManufacturer(pnp_device_id.c_str(),
93 NULL, &name_len);
94 if (FAILED(hr))
95 return false;
97 hr = device_manager->GetDeviceManufacturer(pnp_device_id.c_str(),
98 WriteInto(name, name_len),
99 &name_len);
100 return (SUCCEEDED(hr) && !name->empty());
103 // Gets the description of the device specified by the |pnp_device_id|. On
104 // success, returns true and fills in |description|.
105 bool GetDeviceDescription(const base::string16& pnp_device_id,
106 IPortableDeviceManager* device_manager,
107 base::string16* description) {
108 DCHECK(device_manager);
109 DCHECK(description);
110 DWORD desc_len = 0;
111 HRESULT hr = device_manager->GetDeviceDescription(pnp_device_id.c_str(), NULL,
112 &desc_len);
113 if (FAILED(hr))
114 return false;
116 hr = device_manager->GetDeviceDescription(pnp_device_id.c_str(),
117 WriteInto(description, desc_len),
118 &desc_len);
119 return (SUCCEEDED(hr) && !description->empty());
122 // On success, returns true and updates |client_info| with a reference to an
123 // IPortableDeviceValues interface that holds information about the
124 // application that communicates with the device.
125 bool GetClientInformation(
126 base::win::ScopedComPtr<IPortableDeviceValues>* client_info) {
127 HRESULT hr = client_info->CreateInstance(__uuidof(PortableDeviceValues),
128 NULL, CLSCTX_INPROC_SERVER);
129 if (FAILED(hr)) {
130 DPLOG(ERROR) << "Failed to create an instance of IPortableDeviceValues";
131 return false;
134 // Attempt to set client details.
135 (*client_info)->SetStringValue(WPD_CLIENT_NAME, kClientName);
136 (*client_info)->SetUnsignedIntegerValue(WPD_CLIENT_MAJOR_VERSION, 0);
137 (*client_info)->SetUnsignedIntegerValue(WPD_CLIENT_MINOR_VERSION, 0);
138 (*client_info)->SetUnsignedIntegerValue(WPD_CLIENT_REVISION, 0);
139 (*client_info)->SetUnsignedIntegerValue(
140 WPD_CLIENT_SECURITY_QUALITY_OF_SERVICE, SECURITY_IMPERSONATION);
141 (*client_info)->SetUnsignedIntegerValue(WPD_CLIENT_DESIRED_ACCESS,
142 GENERIC_READ);
143 return true;
146 // Opens the device for communication. |pnp_device_id| specifies the plug and
147 // play device ID string. On success, returns true and updates |device| with a
148 // reference to the portable device interface.
149 bool SetUp(const base::string16& pnp_device_id,
150 base::win::ScopedComPtr<IPortableDevice>* device) {
151 base::win::ScopedComPtr<IPortableDeviceValues> client_info;
152 if (!GetClientInformation(&client_info))
153 return false;
155 HRESULT hr = device->CreateInstance(__uuidof(PortableDevice), NULL,
156 CLSCTX_INPROC_SERVER);
157 if (FAILED(hr)) {
158 DPLOG(ERROR) << "Failed to create an instance of IPortableDevice";
159 return false;
162 hr = (*device)->Open(pnp_device_id.c_str(), client_info.get());
163 if (SUCCEEDED(hr))
164 return true;
166 if (hr == E_ACCESSDENIED)
167 DPLOG(ERROR) << "Access denied to open the device";
168 return false;
171 // Returns the unique id property key of the object specified by the
172 // |object_id|.
173 REFPROPERTYKEY GetUniqueIdPropertyKey(const base::string16& object_id) {
174 return (object_id == WPD_DEVICE_OBJECT_ID) ?
175 WPD_DEVICE_SERIAL_NUMBER : WPD_OBJECT_PERSISTENT_UNIQUE_ID;
178 // On success, returns true and populates |properties_to_read| with the
179 // property key of the object specified by the |object_id|.
180 bool PopulatePropertyKeyCollection(
181 const base::string16& object_id,
182 base::win::ScopedComPtr<IPortableDeviceKeyCollection>* properties_to_read) {
183 HRESULT hr = properties_to_read->CreateInstance(
184 __uuidof(PortableDeviceKeyCollection), NULL, CLSCTX_INPROC_SERVER);
185 if (FAILED(hr)) {
186 DPLOG(ERROR) << "Failed to create IPortableDeviceKeyCollection instance";
187 return false;
189 REFPROPERTYKEY key = GetUniqueIdPropertyKey(object_id);
190 hr = (*properties_to_read)->Add(key);
191 return SUCCEEDED(hr);
194 // Wrapper function to get content property string value.
195 bool GetStringPropertyValue(IPortableDeviceValues* properties_values,
196 REFPROPERTYKEY key,
197 base::string16* value) {
198 DCHECK(properties_values);
199 DCHECK(value);
200 base::win::ScopedCoMem<base::char16> buffer;
201 HRESULT hr = properties_values->GetStringValue(key, &buffer);
202 if (FAILED(hr))
203 return false;
204 *value = static_cast<const base::char16*>(buffer);
205 return true;
208 // Constructs a unique identifier for the object specified by the |object_id|.
209 // On success, returns true and fills in |unique_id|.
210 bool GetObjectUniqueId(IPortableDevice* device,
211 const base::string16& object_id,
212 base::string16* unique_id) {
213 DCHECK(device);
214 DCHECK(unique_id);
215 base::win::ScopedComPtr<IPortableDeviceContent> content;
216 HRESULT hr = device->Content(content.Receive());
217 if (FAILED(hr)) {
218 DPLOG(ERROR) << "Failed to get IPortableDeviceContent interface";
219 return false;
222 base::win::ScopedComPtr<IPortableDeviceProperties> properties;
223 hr = content->Properties(properties.Receive());
224 if (FAILED(hr)) {
225 DPLOG(ERROR) << "Failed to get IPortableDeviceProperties interface";
226 return false;
229 base::win::ScopedComPtr<IPortableDeviceKeyCollection> properties_to_read;
230 if (!PopulatePropertyKeyCollection(object_id, &properties_to_read))
231 return false;
233 base::win::ScopedComPtr<IPortableDeviceValues> properties_values;
234 if (FAILED(properties->GetValues(object_id.c_str(),
235 properties_to_read.get(),
236 properties_values.Receive()))) {
237 return false;
240 REFPROPERTYKEY key = GetUniqueIdPropertyKey(object_id);
241 return GetStringPropertyValue(properties_values.get(), key, unique_id);
244 // Constructs the device storage unique identifier using |device_serial_num| and
245 // |storage_id|. On success, returns true and fills in |device_storage_id|.
246 bool ConstructDeviceStorageUniqueId(const base::string16& device_serial_num,
247 const base::string16& storage_id,
248 std::string* device_storage_id) {
249 if (device_serial_num.empty() && storage_id.empty())
250 return false;
252 DCHECK(device_storage_id);
253 *device_storage_id = StorageInfo::MakeDeviceId(
254 StorageInfo::MTP_OR_PTP,
255 base::UTF16ToUTF8(storage_id + L':' + device_serial_num));
256 return true;
259 // Gets a list of removable storage object identifiers present in |device|.
260 // On success, returns true and fills in |storage_object_ids|.
261 bool GetRemovableStorageObjectIds(
262 IPortableDevice* device,
263 PortableDeviceWatcherWin::StorageObjectIDs* storage_object_ids) {
264 DCHECK(device);
265 DCHECK(storage_object_ids);
266 base::win::ScopedComPtr<IPortableDeviceCapabilities> capabilities;
267 HRESULT hr = device->Capabilities(capabilities.Receive());
268 if (FAILED(hr)) {
269 DPLOG(ERROR) << "Failed to get IPortableDeviceCapabilities interface";
270 return false;
273 base::win::ScopedComPtr<IPortableDevicePropVariantCollection> storage_ids;
274 hr = capabilities->GetFunctionalObjects(WPD_FUNCTIONAL_CATEGORY_STORAGE,
275 storage_ids.Receive());
276 if (FAILED(hr)) {
277 DPLOG(ERROR) << "Failed to get IPortableDevicePropVariantCollection";
278 return false;
281 DWORD num_storage_obj_ids = 0;
282 hr = storage_ids->GetCount(&num_storage_obj_ids);
283 if (FAILED(hr))
284 return false;
286 for (DWORD index = 0; index < num_storage_obj_ids; ++index) {
287 base::win::ScopedPropVariant object_id;
288 hr = storage_ids->GetAt(index, object_id.Receive());
289 if (SUCCEEDED(hr) && object_id.get().vt == VT_LPWSTR &&
290 object_id.get().pwszVal != NULL) {
291 storage_object_ids->push_back(object_id.get().pwszVal);
294 return true;
297 // Returns true if the portable device belongs to a mass storage class.
298 // |pnp_device_id| specifies the plug and play device id.
299 // |device_name| specifies the name of the device.
300 bool IsMassStoragePortableDevice(const base::string16& pnp_device_id,
301 const base::string16& device_name) {
302 // Based on testing, if the pnp device id starts with "\\?\wpdbusenumroot#",
303 // then the attached device belongs to a mass storage class.
304 if (base::StartsWith(pnp_device_id, L"\\\\?\\wpdbusenumroot#",
305 base::CompareCase::INSENSITIVE_ASCII))
306 return true;
308 // If the device is a volume mounted device, |device_name| will be
309 // the volume name.
310 return ((device_name.length() >= 2) && (device_name[1] == L':') &&
311 (((device_name[0] >= L'A') && (device_name[0] <= L'Z')) ||
312 ((device_name[0] >= L'a') && (device_name[0] <= L'z'))));
315 // Returns the name of the device specified by |pnp_device_id|.
316 base::string16 GetDeviceNameOnBlockingThread(
317 IPortableDeviceManager* portable_device_manager,
318 const base::string16& pnp_device_id) {
319 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
320 DCHECK(portable_device_manager);
321 base::string16 name;
322 GetFriendlyName(pnp_device_id, portable_device_manager, &name) ||
323 GetDeviceDescription(pnp_device_id, portable_device_manager, &name) ||
324 GetManufacturerName(pnp_device_id, portable_device_manager, &name);
325 return name;
328 // Access the device and gets the device storage details. On success, returns
329 // true and populates |storage_objects| with device storage details.
330 bool GetDeviceStorageObjectsOnBlockingThread(
331 const base::string16& pnp_device_id,
332 PortableDeviceWatcherWin::StorageObjects* storage_objects) {
333 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
334 DCHECK(storage_objects);
335 base::win::ScopedComPtr<IPortableDevice> device;
336 if (!SetUp(pnp_device_id, &device))
337 return false;
339 base::string16 device_serial_num;
340 if (!GetObjectUniqueId(device.get(), WPD_DEVICE_OBJECT_ID,
341 &device_serial_num)) {
342 return false;
345 PortableDeviceWatcherWin::StorageObjectIDs storage_obj_ids;
346 if (!GetRemovableStorageObjectIds(device.get(), &storage_obj_ids))
347 return false;
348 for (PortableDeviceWatcherWin::StorageObjectIDs::const_iterator id_iter =
349 storage_obj_ids.begin(); id_iter != storage_obj_ids.end(); ++id_iter) {
350 base::string16 storage_persistent_id;
351 if (!GetObjectUniqueId(device.get(), *id_iter, &storage_persistent_id))
352 continue;
354 std::string device_storage_id;
355 if (ConstructDeviceStorageUniqueId(device_serial_num, storage_persistent_id,
356 &device_storage_id)) {
357 storage_objects->push_back(PortableDeviceWatcherWin::DeviceStorageObject(
358 *id_iter, device_storage_id));
361 return true;
364 // Accesses the device and gets the device details (name, storage info, etc).
365 // On success returns true and fills in |device_details|. On failure, returns
366 // false. |pnp_device_id| specifies the plug and play device ID string.
367 bool GetDeviceInfoOnBlockingThread(
368 IPortableDeviceManager* portable_device_manager,
369 const base::string16& pnp_device_id,
370 PortableDeviceWatcherWin::DeviceDetails* device_details) {
371 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
372 DCHECK(portable_device_manager);
373 DCHECK(device_details);
374 DCHECK(!pnp_device_id.empty());
375 device_details->name = GetDeviceNameOnBlockingThread(portable_device_manager,
376 pnp_device_id);
377 if (IsMassStoragePortableDevice(pnp_device_id, device_details->name))
378 return false;
380 device_details->location = pnp_device_id;
381 PortableDeviceWatcherWin::StorageObjects storage_objects;
382 return GetDeviceStorageObjectsOnBlockingThread(
383 pnp_device_id, &device_details->storage_objects);
386 // Wrapper function to get an instance of portable device manager. On success,
387 // returns true and fills in |portable_device_mgr|. On failure, returns false.
388 bool GetPortableDeviceManager(
389 base::win::ScopedComPtr<IPortableDeviceManager>* portable_device_mgr) {
390 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
391 HRESULT hr = portable_device_mgr->CreateInstance(
392 __uuidof(PortableDeviceManager), NULL, CLSCTX_INPROC_SERVER);
393 if (SUCCEEDED(hr))
394 return true;
396 // Either there is no portable device support (Windows XP with old versions of
397 // Media Player) or the thread does not have COM initialized.
398 DCHECK_NE(CO_E_NOTINITIALIZED, hr);
399 return false;
402 // Enumerates the attached portable devices. On success, returns true and fills
403 // in |devices| with the attached portable device details. On failure, returns
404 // false.
405 bool EnumerateAttachedDevicesOnBlockingThread(
406 PortableDeviceWatcherWin::Devices* devices) {
407 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
408 DCHECK(devices);
409 base::win::ScopedComPtr<IPortableDeviceManager> portable_device_mgr;
410 if (!GetPortableDeviceManager(&portable_device_mgr))
411 return false;
413 // Get the total number of devices found on the system.
414 DWORD pnp_device_count = 0;
415 HRESULT hr = portable_device_mgr->GetDevices(NULL, &pnp_device_count);
416 if (FAILED(hr))
417 return false;
419 scoped_ptr<base::char16*[]> pnp_device_ids(
420 new base::char16*[pnp_device_count]);
421 hr = portable_device_mgr->GetDevices(pnp_device_ids.get(), &pnp_device_count);
422 if (FAILED(hr))
423 return false;
425 for (DWORD index = 0; index < pnp_device_count; ++index) {
426 PortableDeviceWatcherWin::DeviceDetails device_details;
427 if (GetDeviceInfoOnBlockingThread(portable_device_mgr.get(),
428 pnp_device_ids[index], &device_details))
429 devices->push_back(device_details);
430 CoTaskMemFree(pnp_device_ids[index]);
432 return !devices->empty();
435 // Handles the device attach event message on a media task runner.
436 // |pnp_device_id| specifies the attached plug and play device ID string. On
437 // success, returns true and populates |device_details| with device information.
438 // On failure, returns false.
439 bool HandleDeviceAttachedEventOnBlockingThread(
440 const base::string16& pnp_device_id,
441 PortableDeviceWatcherWin::DeviceDetails* device_details) {
442 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
443 DCHECK(device_details);
444 base::win::ScopedComPtr<IPortableDeviceManager> portable_device_mgr;
445 if (!GetPortableDeviceManager(&portable_device_mgr))
446 return false;
447 // Sometimes, portable device manager doesn't have the new device details.
448 // Refresh the manager device list to update its details.
449 portable_device_mgr->RefreshDeviceList();
450 return GetDeviceInfoOnBlockingThread(portable_device_mgr.get(), pnp_device_id,
451 device_details);
454 // Registers |hwnd| to receive portable device notification details. On success,
455 // returns the device notifications handle else returns NULL.
456 HDEVNOTIFY RegisterPortableDeviceNotification(HWND hwnd) {
457 GUID dev_interface_guid = GUID_NULL;
458 HRESULT hr = CLSIDFromString(kWPDDevInterfaceGUID, &dev_interface_guid);
459 if (FAILED(hr))
460 return NULL;
461 DEV_BROADCAST_DEVICEINTERFACE db = {
462 sizeof(DEV_BROADCAST_DEVICEINTERFACE),
463 DBT_DEVTYP_DEVICEINTERFACE,
465 dev_interface_guid
467 return RegisterDeviceNotification(hwnd, &db, DEVICE_NOTIFY_WINDOW_HANDLE);
470 } // namespace
473 // PortableDeviceWatcherWin ---------------------------------------------------
475 PortableDeviceWatcherWin::DeviceStorageObject::DeviceStorageObject(
476 const base::string16& temporary_id,
477 const std::string& persistent_id)
478 : object_temporary_id(temporary_id),
479 object_persistent_id(persistent_id) {
482 PortableDeviceWatcherWin::DeviceDetails::DeviceDetails() {
485 PortableDeviceWatcherWin::DeviceDetails::~DeviceDetails() {
488 PortableDeviceWatcherWin::PortableDeviceWatcherWin()
489 : notifications_(NULL),
490 storage_notifications_(NULL),
491 weak_ptr_factory_(this) {
494 PortableDeviceWatcherWin::~PortableDeviceWatcherWin() {
495 UnregisterDeviceNotification(notifications_);
498 void PortableDeviceWatcherWin::Init(HWND hwnd) {
499 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
500 notifications_ = RegisterPortableDeviceNotification(hwnd);
501 base::SequencedWorkerPool* pool = content::BrowserThread::GetBlockingPool();
502 media_task_runner_ = pool->GetSequencedTaskRunnerWithShutdownBehavior(
503 pool->GetNamedSequenceToken(kMediaTaskRunnerName),
504 base::SequencedWorkerPool::CONTINUE_ON_SHUTDOWN);
505 EnumerateAttachedDevices();
508 void PortableDeviceWatcherWin::OnWindowMessage(UINT event_type, LPARAM data) {
509 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
510 if (!IsPortableDeviceStructure(data))
511 return;
513 base::string16 device_id = GetPnpDeviceId(data);
514 if (event_type == DBT_DEVICEARRIVAL)
515 HandleDeviceAttachEvent(device_id);
516 else if (event_type == DBT_DEVICEREMOVECOMPLETE)
517 HandleDeviceDetachEvent(device_id);
520 bool PortableDeviceWatcherWin::GetMTPStorageInfoFromDeviceId(
521 const std::string& storage_device_id,
522 base::string16* device_location,
523 base::string16* storage_object_id) const {
524 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
525 DCHECK(device_location);
526 DCHECK(storage_object_id);
527 MTPStorageMap::const_iterator storage_map_iter =
528 storage_map_.find(storage_device_id);
529 if (storage_map_iter == storage_map_.end())
530 return false;
532 MTPDeviceMap::const_iterator device_iter =
533 device_map_.find(storage_map_iter->second.location());
534 if (device_iter == device_map_.end())
535 return false;
536 const StorageObjects& storage_objects = device_iter->second;
537 for (StorageObjects::const_iterator storage_object_iter =
538 storage_objects.begin(); storage_object_iter != storage_objects.end();
539 ++storage_object_iter) {
540 if (storage_device_id == storage_object_iter->object_persistent_id) {
541 *device_location = storage_map_iter->second.location();
542 *storage_object_id = storage_object_iter->object_temporary_id;
543 return true;
546 return false;
549 // static
550 base::string16 PortableDeviceWatcherWin::GetStoragePathFromStorageId(
551 const std::string& storage_unique_id) {
552 // Construct a dummy device path using the storage name. This is only used
553 // for registering the device media file system.
554 DCHECK(!storage_unique_id.empty());
555 return base::UTF8ToUTF16("\\\\" + storage_unique_id);
558 void PortableDeviceWatcherWin::SetNotifications(
559 StorageMonitor::Receiver* notifications) {
560 storage_notifications_ = notifications;
563 void PortableDeviceWatcherWin::EjectDevice(
564 const std::string& device_id,
565 base::Callback<void(StorageMonitor::EjectStatus)> callback) {
566 // MTP devices on Windows don't have a detach API needed -- signal
567 // the object as if the device is gone and tell the caller it is OK
568 // to remove.
569 base::string16 device_location; // The device_map_ key.
570 base::string16 storage_object_id;
571 if (!GetMTPStorageInfoFromDeviceId(device_id,
572 &device_location, &storage_object_id)) {
573 callback.Run(StorageMonitor::EJECT_NO_SUCH_DEVICE);
574 return;
576 HandleDeviceDetachEvent(device_location);
578 callback.Run(StorageMonitor::EJECT_OK);
581 void PortableDeviceWatcherWin::EnumerateAttachedDevices() {
582 DCHECK(media_task_runner_.get());
583 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
584 Devices* devices = new Devices;
585 base::PostTaskAndReplyWithResult(
586 media_task_runner_.get(), FROM_HERE,
587 base::Bind(&EnumerateAttachedDevicesOnBlockingThread, devices),
588 base::Bind(&PortableDeviceWatcherWin::OnDidEnumerateAttachedDevices,
589 weak_ptr_factory_.GetWeakPtr(), base::Owned(devices)));
592 void PortableDeviceWatcherWin::OnDidEnumerateAttachedDevices(
593 const Devices* devices, const bool result) {
594 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
595 DCHECK(devices);
596 if (!result)
597 return;
598 for (Devices::const_iterator device_iter = devices->begin();
599 device_iter != devices->end(); ++device_iter) {
600 OnDidHandleDeviceAttachEvent(&(*device_iter), result);
604 void PortableDeviceWatcherWin::HandleDeviceAttachEvent(
605 const base::string16& pnp_device_id) {
606 DCHECK(media_task_runner_.get());
607 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
608 DeviceDetails* device_details = new DeviceDetails;
609 base::PostTaskAndReplyWithResult(
610 media_task_runner_.get(), FROM_HERE,
611 base::Bind(&HandleDeviceAttachedEventOnBlockingThread, pnp_device_id,
612 device_details),
613 base::Bind(&PortableDeviceWatcherWin::OnDidHandleDeviceAttachEvent,
614 weak_ptr_factory_.GetWeakPtr(), base::Owned(device_details)));
617 void PortableDeviceWatcherWin::OnDidHandleDeviceAttachEvent(
618 const DeviceDetails* device_details, const bool result) {
619 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
620 DCHECK(device_details);
621 if (!result)
622 return;
624 const StorageObjects& storage_objects = device_details->storage_objects;
625 const base::string16& name = device_details->name;
626 const base::string16& location = device_details->location;
627 DCHECK(!ContainsKey(device_map_, location));
628 for (StorageObjects::const_iterator storage_iter = storage_objects.begin();
629 storage_iter != storage_objects.end(); ++storage_iter) {
630 const std::string& storage_id = storage_iter->object_persistent_id;
631 DCHECK(!ContainsKey(storage_map_, storage_id));
633 // Keep track of storage id and storage name to see how often we receive
634 // empty values.
635 MediaStorageUtil::RecordDeviceInfoHistogram(false, storage_id, name);
636 if (storage_id.empty() || name.empty())
637 return;
639 // Device can have several data partitions. Therefore, add the
640 // partition identifier to the model name. E.g.: "Nexus 7 (s10001)"
641 base::string16 model_name(name + L" (" +
642 storage_iter->object_temporary_id + L')');
643 StorageInfo info(storage_id, location, base::string16(), base::string16(),
644 model_name, 0);
645 storage_map_[storage_id] = info;
646 if (storage_notifications_) {
647 info.set_location(GetStoragePathFromStorageId(storage_id));
648 storage_notifications_->ProcessAttach(info);
651 device_map_[location] = storage_objects;
654 void PortableDeviceWatcherWin::HandleDeviceDetachEvent(
655 const base::string16& pnp_device_id) {
656 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
657 MTPDeviceMap::iterator device_iter = device_map_.find(pnp_device_id);
658 if (device_iter == device_map_.end())
659 return;
661 const StorageObjects& storage_objects = device_iter->second;
662 for (StorageObjects::const_iterator storage_object_iter =
663 storage_objects.begin(); storage_object_iter != storage_objects.end();
664 ++storage_object_iter) {
665 std::string storage_id = storage_object_iter->object_persistent_id;
666 MTPStorageMap::iterator storage_map_iter = storage_map_.find(storage_id);
667 DCHECK(storage_map_iter != storage_map_.end());
668 if (storage_notifications_) {
669 storage_notifications_->ProcessDetach(
670 storage_map_iter->second.device_id());
672 storage_map_.erase(storage_map_iter);
674 device_map_.erase(device_iter);
677 } // namespace storage_monitor