OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/system_monitor/removable_device_notifications_window_wi n.h" | 5 #include "chrome/browser/system_monitor/removable_device_notifications_window_wi n.h" |
6 | 6 |
7 #include <windows.h> | 7 #include <windows.h> |
8 #include <dbt.h> | 8 #include <dbt.h> |
9 #include <fileapi.h> | 9 #include <fileapi.h> |
10 | 10 |
11 #include "base/file_path.h" | 11 #include "base/file_path.h" |
12 #include "base/metrics/histogram.h" | 12 #include "base/i18n/case_conversion.h" |
13 #include "base/string_number_conversions.h" | 13 #include "base/string_number_conversions.h" |
14 #include "base/string_util.h" | 14 #include "base/string_util.h" |
15 #include "base/system_monitor/system_monitor.h" | 15 #include "base/system_monitor/system_monitor.h" |
16 #include "base/utf_string_conversions.h" | 16 #include "base/utf_string_conversions.h" |
17 #include "base/win/wrapped_window_proc.h" | 17 #include "base/win/wrapped_window_proc.h" |
18 #include "chrome/browser/system_monitor/media_device_notifications_utils.h" | 18 #include "chrome/browser/system_monitor/media_device_notifications_utils.h" |
19 #include "chrome/browser/system_monitor/media_storage_util.h" | 19 #include "chrome/browser/system_monitor/media_storage_util.h" |
20 #include "chrome/browser/system_monitor/media_transfer_protocol_device_observer_ win.h" | |
20 #include "content/public/browser/browser_thread.h" | 21 #include "content/public/browser/browser_thread.h" |
21 | 22 |
22 using base::SystemMonitor; | 23 using base::SystemMonitor; |
23 using base::win::WrappedWindowProc; | 24 using base::win::WrappedWindowProc; |
25 using chrome::mtp::MediaTransferProtocolDeviceObserverWin; | |
24 using content::BrowserThread; | 26 using content::BrowserThread; |
25 | 27 |
26 namespace { | 28 namespace { |
27 | 29 |
28 const DWORD kMaxPathBufLen = MAX_PATH + 1; | 30 const DWORD kMaxPathBufLen = MAX_PATH + 1; |
29 | 31 |
30 const char16 kWindowClassName[] = L"Chrome_RemovableDeviceNotificationWindow"; | 32 const char16 kWindowClassName[] = L"Chrome_RemovableDeviceNotificationWindow"; |
31 | 33 |
34 // Windows portable device interface GUID constant. | |
35 const char16 kWPDDevInterfaceGUID[] = L"{6ac27878-a6fa-4155-ba85-f98f491d4f33}"; | |
36 | |
32 static chrome::RemovableDeviceNotificationsWindowWin* | 37 static chrome::RemovableDeviceNotificationsWindowWin* |
33 g_removable_device_notifications_window_win = NULL; | 38 g_removable_device_notifications_window_win = NULL; |
34 | 39 |
35 bool IsRemovable(const char16* mount_point) { | 40 bool IsRemovable(const char16* mount_point) { |
36 if (GetDriveType(mount_point) != DRIVE_REMOVABLE) | 41 if (GetDriveType(mount_point) != DRIVE_REMOVABLE) |
37 return false; | 42 return false; |
38 | 43 |
39 // We don't consider floppy disks as removable, so check for that. | 44 // We don't consider floppy disks as removable, so check for that. |
40 string16 device = mount_point; | 45 string16 device = mount_point; |
41 if (EndsWith(device, L"\\", false)) | 46 if (EndsWith(device, L"\\", false)) |
(...skipping 29 matching lines...) Expand all Loading... | |
71 | 76 |
72 if (name) | 77 if (name) |
73 *name = device_path.LossyDisplayName(); | 78 *name = device_path.LossyDisplayName(); |
74 | 79 |
75 if (removable) | 80 if (removable) |
76 *removable = IsRemovable(mount_point); | 81 *removable = IsRemovable(mount_point); |
77 | 82 |
78 return true; | 83 return true; |
79 } | 84 } |
80 | 85 |
86 // Returns true if |data| represents a logical volume structure. | |
87 bool IsLogicalVolumeStructure(LPARAM data) { | |
88 DEV_BROADCAST_HDR* broadcast_hdr = | |
89 reinterpret_cast<DEV_BROADCAST_HDR*>(data); | |
90 return broadcast_hdr != NULL && | |
91 broadcast_hdr->dbch_devicetype == DBT_DEVTYP_VOLUME; | |
92 } | |
93 | |
94 // Returns true if |data| represents a class of portable devices. | |
95 bool IsPortableDeviceStructure(LPARAM data) { | |
96 DEV_BROADCAST_HDR* broadcast_hdr = | |
97 reinterpret_cast<DEV_BROADCAST_HDR*>(data); | |
98 if (!broadcast_hdr || | |
99 broadcast_hdr->dbch_devicetype != DBT_DEVTYP_DEVICEINTERFACE) { | |
100 return false; | |
101 } | |
102 | |
103 GUID guidDevInterface = GUID_NULL; | |
104 CLSIDFromString(kWPDDevInterfaceGUID, &guidDevInterface); | |
105 DEV_BROADCAST_DEVICEINTERFACE* dev_interface = | |
106 reinterpret_cast<DEV_BROADCAST_DEVICEINTERFACE*>(data); | |
107 return (IsEqualGUID(dev_interface->dbcc_classguid, guidDevInterface) != 0); | |
108 } | |
109 | |
110 // Returns the portable device plug and play device ID string. | |
111 string16 GetPnpDeviceId(LPARAM data) { | |
112 DEV_BROADCAST_DEVICEINTERFACE* dev_interface = | |
113 reinterpret_cast<DEV_BROADCAST_DEVICEINTERFACE*>(data); | |
114 if (!dev_interface) | |
115 return string16(); | |
116 return base::i18n::ToLower(dev_interface->dbcc_name); | |
117 } | |
118 | |
81 std::vector<FilePath> GetAttachedDevices() { | 119 std::vector<FilePath> GetAttachedDevices() { |
82 std::vector<FilePath> result; | 120 std::vector<FilePath> result; |
83 char16 volume_name[kMaxPathBufLen]; | 121 char16 volume_name[kMaxPathBufLen]; |
84 HANDLE find_handle = FindFirstVolume(volume_name, kMaxPathBufLen); | 122 HANDLE find_handle = FindFirstVolume(volume_name, kMaxPathBufLen); |
85 if (find_handle == INVALID_HANDLE_VALUE) | 123 if (find_handle == INVALID_HANDLE_VALUE) |
86 return result; | 124 return result; |
87 | 125 |
88 while (true) { | 126 while (true) { |
89 char16 volume_path[kMaxPathBufLen]; | 127 char16 volume_path[kMaxPathBufLen]; |
90 DWORD return_count; | 128 DWORD return_count; |
(...skipping 104 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
195 GetAttachedDevicesFunc get_attached_devices_func) { | 233 GetAttachedDevicesFunc get_attached_devices_func) { |
196 get_device_info_func_ = get_device_info_func; | 234 get_device_info_func_ = get_device_info_func; |
197 DoInit(get_attached_devices_func); | 235 DoInit(get_attached_devices_func); |
198 } | 236 } |
199 | 237 |
200 void RemovableDeviceNotificationsWindowWin::OnDeviceChange(UINT event_type, | 238 void RemovableDeviceNotificationsWindowWin::OnDeviceChange(UINT event_type, |
201 LPARAM data) { | 239 LPARAM data) { |
202 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 240 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
203 switch (event_type) { | 241 switch (event_type) { |
204 case DBT_DEVICEARRIVAL: { | 242 case DBT_DEVICEARRIVAL: { |
205 DWORD unitmask = GetVolumeBitMaskFromBroadcastHeader(data); | 243 if (IsLogicalVolumeStructure(data)) { |
206 for (int i = 0; unitmask; ++i, unitmask >>= 1) { | 244 DWORD unitmask = GetVolumeBitMaskFromBroadcastHeader(data); |
207 if (!(unitmask & 0x01)) | 245 for (int i = 0; unitmask; ++i, unitmask >>= 1) { |
208 continue; | 246 if (!(unitmask & 0x01)) |
209 AddNewDevice(DriveNumberToFilePath(i)); | 247 continue; |
248 AddNewDevice(DriveNumberToFilePath(i)); | |
249 } | |
250 } else if (IsPortableDeviceStructure(data)) { | |
251 HandleMtpDeviceEventOnUIThread(true, GetPnpDeviceId(data)); | |
210 } | 252 } |
211 break; | 253 break; |
212 } | 254 } |
213 case DBT_DEVICEREMOVECOMPLETE: { | 255 case DBT_DEVICEREMOVECOMPLETE: { |
214 DWORD unitmask = GetVolumeBitMaskFromBroadcastHeader(data); | 256 if (IsLogicalVolumeStructure(data)) { |
215 for (int i = 0; unitmask; ++i, unitmask >>= 1) { | 257 DWORD unitmask = GetVolumeBitMaskFromBroadcastHeader(data); |
216 if (!(unitmask & 0x01)) | 258 for (int i = 0; unitmask; ++i, unitmask >>= 1) { |
217 continue; | 259 if (!(unitmask & 0x01)) |
218 | 260 continue; |
219 FilePath device = DriveNumberToFilePath(i); | 261 HandleMassStorageDeviceDetachEventOnUIThread( |
220 MountPointDeviceIdMap::const_iterator device_info = | 262 DriveNumberToFilePath(i).value()); |
221 device_ids_.find(device); | 263 } |
222 // If the devices isn't type removable (like a CD), it won't be there. | 264 } else if (IsPortableDeviceStructure(data)) { |
223 if (device_info == device_ids_.end()) | 265 HandleMtpDeviceEventOnUIThread(false, GetPnpDeviceId(data)); |
224 continue; | |
225 | |
226 SystemMonitor::Get()->ProcessRemovableStorageDetached( | |
227 device_info->second); | |
228 device_ids_.erase(device_info); | |
229 } | 266 } |
230 break; | 267 break; |
231 } | 268 } |
232 } | 269 } |
233 } | 270 } |
234 | 271 |
235 RemovableDeviceNotificationsWindowWin:: | 272 RemovableDeviceNotificationsWindowWin:: |
236 ~RemovableDeviceNotificationsWindowWin() { | 273 ~RemovableDeviceNotificationsWindowWin() { |
237 if (window_) | 274 if (window_) |
238 DestroyWindow(window_); | 275 DestroyWindow(window_); |
239 | 276 |
240 if (window_class_) | 277 if (window_class_) |
241 UnregisterClass(MAKEINTATOM(window_class_), instance_); | 278 UnregisterClass(MAKEINTATOM(window_class_), instance_); |
242 | 279 |
243 DCHECK_EQ(this, g_removable_device_notifications_window_win); | 280 DCHECK_EQ(this, g_removable_device_notifications_window_win); |
244 g_removable_device_notifications_window_win = NULL; | 281 g_removable_device_notifications_window_win = NULL; |
245 } | 282 } |
246 | 283 |
247 // static | 284 // static |
248 LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProcThunk( | 285 LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProcThunk( |
249 HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) { | 286 HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) { |
287 #if defined(DBT_DEVTYP_DEVICEINTERFACE) && defined(DEVICE_NOTIFY_WINDOW_HANDLE) | |
288 static HDEVNOTIFY hDeviceNotify; | |
289 if (message == WM_CREATE) { | |
vandebo (ex-Chrome)
2012/10/16 17:42:36
content/browser/system_message_window_win.cc just
kmadhusu
2012/10/19 04:01:38
This code is very similar to the SystemMessageWind
| |
290 GUID guidDevInterface = GUID_NULL; | |
291 HRESULT hr = CLSIDFromString(kWPDDevInterfaceGUID, &guidDevInterface); | |
292 if (SUCCEEDED(hr)) { | |
293 DEV_BROADCAST_DEVICEINTERFACE db = {0}; | |
294 db.dbcc_size = sizeof(db); | |
295 db.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE; | |
296 db.dbcc_classguid = guidDevInterface; | |
297 hDeviceNotify = RegisterDeviceNotification(hwnd, &db, | |
298 DEVICE_NOTIFY_WINDOW_HANDLE); | |
299 } | |
300 } else if (message == WM_CLOSE) { | |
301 UnregisterDeviceNotification(hDeviceNotify); | |
302 } | |
303 #endif | |
250 RemovableDeviceNotificationsWindowWin* msg_wnd = | 304 RemovableDeviceNotificationsWindowWin* msg_wnd = |
251 reinterpret_cast<RemovableDeviceNotificationsWindowWin*>( | 305 reinterpret_cast<RemovableDeviceNotificationsWindowWin*>( |
252 GetWindowLongPtr(hwnd, GWLP_USERDATA)); | 306 GetWindowLongPtr(hwnd, GWLP_USERDATA)); |
253 if (msg_wnd) | 307 if (msg_wnd) |
254 return msg_wnd->WndProc(hwnd, message, wparam, lparam); | 308 return msg_wnd->WndProc(hwnd, message, wparam, lparam); |
255 return ::DefWindowProc(hwnd, message, wparam, lparam); | 309 return ::DefWindowProc(hwnd, message, wparam, lparam); |
256 } | 310 } |
257 | 311 |
258 LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProc( | 312 LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProc( |
259 HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) { | 313 HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) { |
(...skipping 64 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
324 const FilePath& device) { | 378 const FilePath& device) { |
325 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); | 379 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); |
326 | 380 |
327 MediaStorageUtil::Type type = | 381 MediaStorageUtil::Type type = |
328 MediaStorageUtil::REMOVABLE_MASS_STORAGE_NO_DCIM; | 382 MediaStorageUtil::REMOVABLE_MASS_STORAGE_NO_DCIM; |
329 if (IsMediaDevice(device.value())) | 383 if (IsMediaDevice(device.value())) |
330 type = MediaStorageUtil::REMOVABLE_MASS_STORAGE_WITH_DCIM; | 384 type = MediaStorageUtil::REMOVABLE_MASS_STORAGE_WITH_DCIM; |
331 std::string device_id = MediaStorageUtil::MakeDeviceId(type, unique_id); | 385 std::string device_id = MediaStorageUtil::MakeDeviceId(type, unique_id); |
332 | 386 |
333 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind( | 387 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind( |
334 &RemovableDeviceNotificationsWindowWin::ProcessDeviceAttachedOnUIThread, | 388 &RemovableDeviceNotificationsWindowWin:: |
335 this, device_id, device_name, device)); | 389 HandleMassStorageDeviceAttachEventOnUIThread, |
390 this, device_id, device_name, device.value())); | |
336 } | 391 } |
337 | 392 |
338 void RemovableDeviceNotificationsWindowWin::ProcessDeviceAttachedOnUIThread( | 393 void RemovableDeviceNotificationsWindowWin:: |
394 HandleMassStorageDeviceAttachEventOnUIThread( | |
339 const std::string& device_id, | 395 const std::string& device_id, |
340 const FilePath::StringType& device_name, | 396 const string16& device_name, |
341 const FilePath& device) { | 397 const string16& device_location) { |
342 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 398 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
343 | 399 |
344 // TODO(kmadhusu) Record device info histogram. | 400 // TODO(kmadhusu) Record device info histogram. |
345 device_ids_[device] = device_id; | 401 device_ids_[device_location] = device_id; |
346 SystemMonitor::Get()->ProcessRemovableStorageAttached(device_id, | 402 SystemMonitor::Get()->ProcessRemovableStorageAttached(device_id, |
347 device_name, | 403 device_name, |
348 device.value()); | 404 device_location); |
405 } | |
406 | |
407 void RemovableDeviceNotificationsWindowWin:: | |
408 HandleMassStorageDeviceDetachEventOnUIThread(const string16& device_location) { | |
409 MountPointDeviceIdMap::const_iterator device_info = | |
410 device_ids_.find(device_location); | |
411 // If the devices isn't type removable (like a CD), it won't be there. | |
412 if (device_info == device_ids_.end()) | |
413 return; | |
414 | |
415 SystemMonitor::Get()->ProcessRemovableStorageDetached(device_info->second); | |
416 device_ids_.erase(device_info); | |
417 } | |
418 | |
419 void RemovableDeviceNotificationsWindowWin:: | |
420 HandleMtpDeviceEventOnUIThread(bool is_attach, const string16& pnp_device_id) { | |
421 if (pnp_device_id.empty()) | |
422 return; | |
423 | |
424 MediaTransferProtocolDeviceObserverWin* mtp_device_observer = | |
425 MediaTransferProtocolDeviceObserverWin::GetInstance(); | |
426 if (mtp_device_observer) { | |
427 mtp_device_observer->HandleMtpDeviceEventOnUIThread(is_attach, | |
428 pnp_device_id); | |
429 } | |
349 } | 430 } |
350 | 431 |
351 } // namespace chrome | 432 } // namespace chrome |
OLD | NEW |