Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(401)

Unified Diff: chrome/browser/system_monitor/removable_device_notifications_window_win.cc

Issue 10911350: Update Windows System Monitor Removable Device Impl. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Rebase Created 8 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/system_monitor/removable_device_notifications_window_win.cc
diff --git a/chrome/browser/system_monitor/removable_device_notifications_window_win.cc b/chrome/browser/system_monitor/removable_device_notifications_window_win.cc
index e2fca5a33ae4b7d5aced5bf6d2ae4b538cc56995..bb3caad1f175cedb063235a9596cde3c698c79cf 100644
--- a/chrome/browser/system_monitor/removable_device_notifications_window_win.cc
+++ b/chrome/browser/system_monitor/removable_device_notifications_window_win.cc
@@ -6,168 +6,256 @@
#include <windows.h>
#include <dbt.h>
-
-#include <string>
+#include <fileapi.h>
#include "base/file_path.h"
+#include "base/metrics/histogram.h"
#include "base/string_number_conversions.h"
#include "base/system_monitor/system_monitor.h"
+#include "base/utf_string_conversions.h"
#include "base/win/wrapped_window_proc.h"
#include "chrome/browser/system_monitor/media_device_notifications_utils.h"
#include "chrome/browser/system_monitor/media_storage_util.h"
#include "content/public/browser/browser_thread.h"
using base::SystemMonitor;
+using base::win::WrappedWindowProc;
using content::BrowserThread;
namespace {
-const wchar_t WindowClassName[] = L"Chrome_RemovableDeviceNotificationWindow";
+const DWORD kMaxPathBufLen = MAX_PATH + 1;
+
+const char16 kWindowClassName[] = L"Chrome_RemovableDeviceNotificationWindow";
+
+static chrome::RemovableDeviceNotificationsWindowWin*
+ g_removable_device_notifications_window_win = NULL;
+
+// The following msdn blog entry is helpful for understanding disk volumes
+// and how they are treated in Windows:
+// http://blogs.msdn.com/b/adioltean/archive/2005/04/16/408947.aspx
+bool GetDeviceInfo(const FilePath& device_path, string16* device_location,
+ std::string* unique_id, string16* name, bool* removable) {
+ char16 mount_point[kMaxPathBufLen];
+ if (!GetVolumePathName(device_path.value().c_str(), mount_point,
+ kMaxPathBufLen)) {
+ return false;
+ }
+ if (device_location)
+ *device_location = string16(mount_point);
+
+ if (unique_id) {
+ char16 guid[kMaxPathBufLen];
+ if (!GetVolumeNameForVolumeMountPoint(mount_point, guid, kMaxPathBufLen))
+ return false;
+ // In case it has two GUID's (see above mentioned blog), do it again.
+ if (!GetVolumeNameForVolumeMountPoint(guid, guid, kMaxPathBufLen))
+ return false;
+ WideToUTF8(guid, wcslen(guid), unique_id);
+ }
+
+ if (name) {
+ char16 volume_name[kMaxPathBufLen];
+ if (!GetVolumeInformation(mount_point, volume_name, kMaxPathBufLen, NULL,
+ NULL, NULL, NULL, 0)) {
+ return false;
+ }
+ if (wcslen(volume_name) > 0) {
+ *name = string16(volume_name);
+ } else {
+ *name = device_path.LossyDisplayName();
+ }
+ }
+
+ if (removable)
+ *removable = (GetDriveType(mount_point) == DRIVE_REMOVABLE);
-LRESULT GetVolumeName(LPCWSTR drive,
- LPWSTR volume_name,
- unsigned int volume_name_len) {
- return GetVolumeInformation(drive, volume_name, volume_name_len, NULL, NULL,
- NULL, NULL, 0);
+ return true;
+}
+
+std::vector<FilePath> GetAttachedDevices() {
+ std::vector<FilePath> result;
+ char16 volume_name[kMaxPathBufLen];
+ HANDLE find_handle = FindFirstVolume(volume_name, kMaxPathBufLen);
+ if (find_handle == INVALID_HANDLE_VALUE)
+ return result;
+
+ while (true) {
+ char16 volume_path[kMaxPathBufLen];
+ DWORD return_count;
+ if (GetVolumePathNamesForVolumeName(volume_name, volume_path,
+ kMaxPathBufLen, &return_count)) {
+ if (GetDriveType(volume_path) == DRIVE_REMOVABLE)
+ result.push_back(FilePath(volume_path));
+ } else {
+ DPLOG(ERROR);
+ }
+ if (!FindNextVolume(find_handle, volume_name, kMaxPathBufLen)) {
+ if (GetLastError() != ERROR_NO_MORE_FILES)
+ DPLOG(ERROR);
+ break;
+ }
+ }
+
+ FindVolumeClose(find_handle);
+ return result;
}
// Returns 0 if the devicetype is not volume.
-DWORD GetVolumeBitMaskFromBroadcastHeader(DWORD data) {
- PDEV_BROADCAST_HDR dev_broadcast_hdr =
- reinterpret_cast<PDEV_BROADCAST_HDR>(data);
- if (dev_broadcast_hdr->dbch_devicetype == DBT_DEVTYP_VOLUME) {
- PDEV_BROADCAST_VOLUME dev_broadcast_volume =
- reinterpret_cast<PDEV_BROADCAST_VOLUME>(dev_broadcast_hdr);
+uint32 GetVolumeBitMaskFromBroadcastHeader(LPARAM data) {
+ DEV_BROADCAST_VOLUME* dev_broadcast_volume =
+ reinterpret_cast<DEV_BROADCAST_VOLUME*>(data);
+ if (dev_broadcast_volume->dbcv_devicetype == DBT_DEVTYP_VOLUME)
return dev_broadcast_volume->dbcv_unitmask;
- }
return 0;
}
+FilePath DriveNumberToFilePath(int drive_number) {
+ string16 path(L"_:\\");
+ path[0] = L'A' + drive_number;
+ return FilePath(path);
+}
+
} // namespace
namespace chrome {
RemovableDeviceNotificationsWindowWin::RemovableDeviceNotificationsWindowWin()
- : atom_(0),
+ : window_class_(0),
instance_(NULL),
window_(NULL),
- volume_name_func_(&GetVolumeName) {
- Init();
+ get_device_info_func_(&GetDeviceInfo) {
+ DCHECK(!g_removable_device_notifications_window_win);
+ g_removable_device_notifications_window_win = this;
}
-RemovableDeviceNotificationsWindowWin::RemovableDeviceNotificationsWindowWin(
- VolumeNameFunc volume_name_func)
- : atom_(0),
- instance_(NULL),
- window_(NULL),
- volume_name_func_(volume_name_func) {
- Init();
+// static
+RemovableDeviceNotificationsWindowWin*
+RemovableDeviceNotificationsWindowWin::GetInstance() {
+ DCHECK(g_removable_device_notifications_window_win);
+ return g_removable_device_notifications_window_win;
}
void RemovableDeviceNotificationsWindowWin::Init() {
- WNDCLASSEX window_class;
- base::win::InitializeWindowClass(
- WindowClassName,
- &base::win::WrappedWindowProc<
- RemovableDeviceNotificationsWindowWin::WndProcThunk>,
- 0, 0, 0, NULL, NULL, NULL, NULL, NULL,
- &window_class);
- instance_ = window_class.hInstance;
- atom_ = RegisterClassEx(&window_class);
- DCHECK(atom_);
-
- window_ = CreateWindow(MAKEINTATOM(atom_), 0, 0, 0, 0, 0, 0, 0, 0, instance_,
- 0);
- SetWindowLongPtr(window_, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(this));
+ DoInit(&GetAttachedDevices);
}
-RemovableDeviceNotificationsWindowWin::~RemovableDeviceNotificationsWindowWin(
- ) {
- if (window_)
- DestroyWindow(window_);
+bool RemovableDeviceNotificationsWindowWin::GetDeviceInfoForPath(
+ const FilePath& path,
+ base::SystemMonitor::RemovableStorageInfo* device_info) {
+ string16 location;
+ std::string unique_id;
+ string16 name;
+ bool removable;
+ if (!get_device_info_func_(path, &location, &unique_id, &name, &removable))
+ return false;
+
+ // To compute the device id, the device type is needed. For removable
+ // devices, that requires knowing if there's a DCIM directory, which would
+ // require bouncing over to the file thread. Instead, just iterate the
+ // devices in SystemMonitor.
+ std::string device_id;
+ if (removable) {
+ std::vector<SystemMonitor::RemovableStorageInfo> attached_devices =
+ SystemMonitor::Get()->GetAttachedRemovableStorage();
+ bool found = false;
+ for (size_t i = 0; i < attached_devices.size(); i++) {
+ MediaStorageUtil::Type type;
+ std::string id;
+ MediaStorageUtil::CrackDeviceId(attached_devices[i].device_id, &type,
+ &id);
+ if (id == unique_id) {
+ found = true;
+ device_id = attached_devices[i].device_id;
+ break;
+ }
+ }
+ if (!found)
+ return false;
+ } else {
+ device_id = MediaStorageUtil::MakeDeviceId(
+ MediaStorageUtil::FIXED_MASS_STORAGE, unique_id);
+ }
+
+ if (device_info) {
+ device_info->device_id = device_id;
+ device_info->name = name;
+ device_info->location = location;
+ }
+ return true;
+}
- if (atom_)
- UnregisterClass(MAKEINTATOM(atom_), instance_);
+void RemovableDeviceNotificationsWindowWin::InitForTest(
+ GetDeviceInfoFunc get_device_info_func,
+ GetAttachedDevicesFunc get_attached_devices_func) {
+ get_device_info_func_ = get_device_info_func;
+ DoInit(get_attached_devices_func);
}
-LRESULT RemovableDeviceNotificationsWindowWin::OnDeviceChange(UINT event_type,
- DWORD data) {
+void RemovableDeviceNotificationsWindowWin::OnDeviceChange(UINT event_type,
+ LPARAM data) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
switch (event_type) {
case DBT_DEVICEARRIVAL: {
DWORD unitmask = GetVolumeBitMaskFromBroadcastHeader(data);
for (int i = 0; unitmask; ++i, unitmask >>= 1) {
- if (unitmask & 0x01) {
- FilePath::StringType drive(L"_:\\");
- drive[0] = L'A' + i;
- WCHAR volume_name[MAX_PATH + 1];
- if ((*volume_name_func_)(drive.c_str(), volume_name, MAX_PATH + 1)) {
- // TODO(kmadhusu) We need to look up a real device id as well as
- // having a fall back for volume name.
- std::string device_id = MediaStorageUtil::MakeDeviceId(
- MediaStorageUtil::REMOVABLE_MASS_STORAGE_WITH_DCIM,
- base::IntToString(i));
- BrowserThread::PostTask(
- BrowserThread::FILE, FROM_HERE,
- base::Bind(&RemovableDeviceNotificationsWindowWin::
- CheckDeviceTypeOnFileThread, this, device_id,
- FilePath::StringType(volume_name), FilePath(drive)));
- }
- }
+ if (!(unitmask & 0x01))
+ continue;
+ AddNewDevice(DriveNumberToFilePath(i));
}
break;
}
case DBT_DEVICEREMOVECOMPLETE: {
DWORD unitmask = GetVolumeBitMaskFromBroadcastHeader(data);
for (int i = 0; unitmask; ++i, unitmask >>= 1) {
- if (unitmask & 0x01) {
- std::string device_id = MediaStorageUtil::MakeDeviceId(
- MediaStorageUtil::REMOVABLE_MASS_STORAGE_WITH_DCIM,
- base::IntToString(i));
- SystemMonitor::Get()->ProcessRemovableStorageDetached(device_id);
- }
+ if (!(unitmask & 0x01))
+ continue;
+
+ FilePath device = DriveNumberToFilePath(i);
+ MountPointDeviceIdMap::const_iterator device_info =
+ device_ids_.find(device);
+ // If the devices isn't type removable (like a CD), it won't be there.
+ if (device_info == device_ids_.end())
+ continue;
+
+ SystemMonitor::Get()->ProcessRemovableStorageDetached(
+ device_info->second);
+ device_ids_.erase(device_info);
}
break;
}
}
- return TRUE;
}
-void RemovableDeviceNotificationsWindowWin::CheckDeviceTypeOnFileThread(
- const std::string& id,
- const FilePath::StringType& device_name,
- const FilePath& path) {
- DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
- if (!IsMediaDevice(path.value()))
- return;
+RemovableDeviceNotificationsWindowWin::
+ ~RemovableDeviceNotificationsWindowWin() {
+ if (window_)
+ DestroyWindow(window_);
- BrowserThread::PostTask(
- BrowserThread::UI, FROM_HERE,
- base::Bind(
- &RemovableDeviceNotificationsWindowWin::
- ProcessRemovableDeviceAttachedOnUIThread,
- this, id, device_name, path));
-}
+ if (window_class_)
+ UnregisterClass(MAKEINTATOM(window_class_), instance_);
-void
-RemovableDeviceNotificationsWindowWin::ProcessRemovableDeviceAttachedOnUIThread(
- const std::string& id,
- const FilePath::StringType& device_name,
- const FilePath& path) {
- DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
+ DCHECK_EQ(this, g_removable_device_notifications_window_win);
+ g_removable_device_notifications_window_win = NULL;
+}
- // TODO(kmadhusu) Record device info histogram.
- SystemMonitor::Get()->ProcessRemovableStorageAttached(id,
- device_name,
- path.value());
+// static
+LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProcThunk(
+ HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) {
+ RemovableDeviceNotificationsWindowWin* msg_wnd =
+ reinterpret_cast<RemovableDeviceNotificationsWindowWin*>(
+ GetWindowLongPtr(hwnd, GWLP_USERDATA));
+ if (msg_wnd)
+ return msg_wnd->WndProc(hwnd, message, wparam, lparam);
+ return ::DefWindowProc(hwnd, message, wparam, lparam);
}
LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProc(
HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) {
switch (message) {
case WM_DEVICECHANGE:
- return OnDeviceChange(static_cast<UINT>(wparam),
- static_cast<DWORD>(lparam));
+ OnDeviceChange(static_cast<UINT>(wparam), lparam);
+ return TRUE;
default:
break;
}
@@ -175,18 +263,76 @@ LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProc(
return ::DefWindowProc(hwnd, message, wparam, lparam);
}
-// static
-LRESULT CALLBACK RemovableDeviceNotificationsWindowWin::WndProcThunk(
- HWND hwnd,
- UINT message,
- WPARAM wparam,
- LPARAM lparam) {
- RemovableDeviceNotificationsWindowWin* msg_wnd =
- reinterpret_cast<RemovableDeviceNotificationsWindowWin*>(
- GetWindowLongPtr(hwnd, GWLP_USERDATA));
- if (msg_wnd)
- return msg_wnd->WndProc(hwnd, message, wparam, lparam);
- return ::DefWindowProc(hwnd, message, wparam, lparam);
+void RemovableDeviceNotificationsWindowWin::DoInit(
+ GetAttachedDevicesFunc get_attached_devices_func) {
+ WNDCLASSEX window_class;
+ base::win::InitializeWindowClass(
+ kWindowClassName,
+ &WrappedWindowProc<RemovableDeviceNotificationsWindowWin::WndProcThunk>,
+ 0, 0, 0, NULL, NULL, NULL, NULL, NULL,
+ &window_class);
+ instance_ = window_class.hInstance;
+ window_class_ = RegisterClassEx(&window_class);
+ DCHECK(window_class_);
+
+ window_ = CreateWindow(MAKEINTATOM(window_class_), 0, 0, 0, 0, 0, 0, 0, 0,
+ instance_, 0);
+ SetWindowLongPtr(window_, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(this));
+
+ std::vector<FilePath> removable_devices = get_attached_devices_func();
+ for (size_t i = 0; i < removable_devices.size(); i++)
+ AddNewDevice(removable_devices[i]);
+}
+
+void RemovableDeviceNotificationsWindowWin::AddNewDevice(
+ const FilePath& device_path) {
+ std::string unique_id;
+ string16 device_name;
+ bool removable;
+ if (!get_device_info_func_(device_path, NULL, &unique_id, &device_name,
+ &removable)) {
+ return;
+ }
+
+ if (!removable)
+ return;
+
+ BrowserThread::PostTask(
+ BrowserThread::FILE,
+ FROM_HERE,
+ base::Bind(
+ &RemovableDeviceNotificationsWindowWin::CheckDeviceTypeOnFileThread,
+ this, unique_id, device_name, device_path));
+}
+
+void RemovableDeviceNotificationsWindowWin::CheckDeviceTypeOnFileThread(
+ const std::string& unique_id,
+ const FilePath::StringType& device_name,
+ const FilePath& device) {
+ DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
+
+ MediaStorageUtil::Type type =
+ MediaStorageUtil::REMOVABLE_MASS_STORAGE_NO_DCIM;
+ if (IsMediaDevice(device.value()))
+ type = MediaStorageUtil::REMOVABLE_MASS_STORAGE_WITH_DCIM;
+ std::string device_id = MediaStorageUtil::MakeDeviceId(type, unique_id);
+
+ BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind(
+ &RemovableDeviceNotificationsWindowWin::ProcessDeviceAttachedOnUIThread,
+ this, device_id, device_name, device));
+}
+
+void RemovableDeviceNotificationsWindowWin::ProcessDeviceAttachedOnUIThread(
+ const std::string& device_id,
+ const FilePath::StringType& device_name,
+ const FilePath& device) {
+ DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
+
+ // TODO(kmadhusu) Record device info histogram.
+ device_ids_[device] = device_id;
+ SystemMonitor::Get()->ProcessRemovableStorageAttached(device_id,
+ device_name,
+ device.value());
}
} // namespace chrome

Powered by Google App Engine
This is Rietveld 408576698