• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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
3 // found in the LICENSE file.
4 
5 #include "chrome/browser/storage_monitor/media_storage_util.h"
6 
7 #include <vector>
8 
9 #include "base/callback.h"
10 #include "base/file_util.h"
11 #include "base/logging.h"
12 #include "base/metrics/histogram.h"
13 #include "base/strings/string_util.h"
14 #include "base/strings/utf_string_conversions.h"
15 #include "chrome/browser/storage_monitor/removable_device_constants.h"
16 #include "chrome/browser/storage_monitor/storage_monitor.h"
17 #include "content/public/browser/browser_thread.h"
18 
19 using content::BrowserThread;
20 
21 namespace {
22 
23 // MediaDeviceNotification.DeviceInfo histogram values.
24 enum DeviceInfoHistogramBuckets {
25   MASS_STORAGE_DEVICE_NAME_AND_UUID_AVAILABLE,
26   MASS_STORAGE_DEVICE_UUID_MISSING,
27   MASS_STORAGE_DEVICE_NAME_MISSING,
28   MASS_STORAGE_DEVICE_NAME_AND_UUID_MISSING,
29   MTP_STORAGE_DEVICE_NAME_AND_UUID_AVAILABLE,
30   MTP_STORAGE_DEVICE_UUID_MISSING,
31   MTP_STORAGE_DEVICE_NAME_MISSING,
32   MTP_STORAGE_DEVICE_NAME_AND_UUID_MISSING,
33   DEVICE_INFO_BUCKET_BOUNDARY
34 };
35 
36 #if !defined(OS_WIN)
37 const char kRootPath[] = "/";
38 #endif
39 
40 typedef std::vector<StorageInfo> StorageInfoList;
41 
FindRemovableStorageLocationById(const std::string & device_id)42 base::FilePath::StringType FindRemovableStorageLocationById(
43     const std::string& device_id) {
44   StorageInfoList devices =
45       StorageMonitor::GetInstance()->GetAllAvailableStorages();
46   for (StorageInfoList::const_iterator it = devices.begin();
47        it != devices.end(); ++it) {
48     if (it->device_id() == device_id
49         && StorageInfo::IsRemovableDevice(device_id))
50       return it->location();
51   }
52   return base::FilePath::StringType();
53 }
54 
FilterAttachedDevicesOnFileThread(MediaStorageUtil::DeviceIdSet * devices)55 void FilterAttachedDevicesOnFileThread(MediaStorageUtil::DeviceIdSet* devices) {
56   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
57   MediaStorageUtil::DeviceIdSet missing_devices;
58 
59   for (MediaStorageUtil::DeviceIdSet::const_iterator it = devices->begin();
60        it != devices->end();
61        ++it) {
62     StorageInfo::Type type;
63     std::string unique_id;
64     if (!StorageInfo::CrackDeviceId(*it, &type, &unique_id)) {
65       missing_devices.insert(*it);
66       continue;
67     }
68 
69     if (type == StorageInfo::FIXED_MASS_STORAGE ||
70         type == StorageInfo::ITUNES ||
71         type == StorageInfo::IPHOTO ||
72         type == StorageInfo::PICASA) {
73       if (!base::PathExists(base::FilePath::FromUTF8Unsafe(unique_id)))
74         missing_devices.insert(*it);
75       continue;
76     }
77 
78     if (!MediaStorageUtil::IsRemovableStorageAttached(*it))
79       missing_devices.insert(*it);
80   }
81 
82   for (MediaStorageUtil::DeviceIdSet::const_iterator it =
83            missing_devices.begin();
84        it != missing_devices.end();
85        ++it) {
86     devices->erase(*it);
87   }
88 }
89 
90 }  // namespace
91 
92 // static
HasDcim(const base::FilePath & mount_point)93 bool MediaStorageUtil::HasDcim(const base::FilePath& mount_point) {
94   DCHECK(!content::BrowserThread::CurrentlyOn(content::BrowserThread::UI));
95 
96   base::FilePath::StringType dcim_dir(kDCIMDirectoryName);
97   if (!base::DirectoryExists(mount_point.Append(dcim_dir))) {
98     // Check for lowercase 'dcim' as well.
99     base::FilePath dcim_path_lower(
100         mount_point.Append(StringToLowerASCII(dcim_dir)));
101     if (!base::DirectoryExists(dcim_path_lower))
102       return false;
103   }
104   return true;
105 }
106 
107 // static
CanCreateFileSystem(const std::string & device_id,const base::FilePath & path)108 bool MediaStorageUtil::CanCreateFileSystem(const std::string& device_id,
109                                            const base::FilePath& path) {
110   StorageInfo::Type type;
111   if (!StorageInfo::CrackDeviceId(device_id, &type, NULL))
112     return false;
113 
114   if (type == StorageInfo::MAC_IMAGE_CAPTURE)
115     return true;
116 
117   return path.IsAbsolute() && !path.ReferencesParent();
118 }
119 
120 // static
FilterAttachedDevices(DeviceIdSet * devices,const base::Closure & done)121 void MediaStorageUtil::FilterAttachedDevices(DeviceIdSet* devices,
122                                              const base::Closure& done) {
123   if (BrowserThread::CurrentlyOn(BrowserThread::FILE)) {
124     FilterAttachedDevicesOnFileThread(devices);
125     done.Run();
126     return;
127   }
128   BrowserThread::PostTaskAndReply(BrowserThread::FILE,
129                                   FROM_HERE,
130                                   base::Bind(&FilterAttachedDevicesOnFileThread,
131                                              devices),
132                                   done);
133 }
134 
135 // TODO(kmadhusu) Write unit tests for GetDeviceInfoFromPath().
GetDeviceInfoFromPath(const base::FilePath & path,StorageInfo * device_info,base::FilePath * relative_path)136 bool MediaStorageUtil::GetDeviceInfoFromPath(const base::FilePath& path,
137                                              StorageInfo* device_info,
138                                              base::FilePath* relative_path) {
139   DCHECK(device_info);
140   DCHECK(relative_path);
141 
142   if (!path.IsAbsolute())
143     return false;
144 
145   StorageInfo info;
146   StorageMonitor* monitor = StorageMonitor::GetInstance();
147   bool found_device = monitor->GetStorageInfoForPath(path, &info);
148 
149   if (found_device && StorageInfo::IsRemovableDevice(info.device_id())) {
150     base::FilePath sub_folder_path;
151     base::FilePath device_path(info.location());
152     if (path != device_path) {
153       bool success = device_path.AppendRelativePath(path, &sub_folder_path);
154       DCHECK(success);
155     }
156 
157     *device_info = info;
158     *relative_path = sub_folder_path;
159     return true;
160   }
161 
162   // On Posix systems, there's one root so any absolute path could be valid.
163   // TODO(gbillock): Delete this stanza? Posix systems should have the root
164   // volume information. If not, we should move the below into the
165   // right GetStorageInfoForPath implementations.
166 #if !defined(OS_POSIX)
167   if (!found_device)
168     return false;
169 #endif
170 
171   // Handle non-removable devices. Note: this is just overwriting
172   // good values from StorageMonitor.
173   // TODO(gbillock): Make sure return values from that class are definitive,
174   // and don't do this here.
175   info.set_device_id(
176       StorageInfo::MakeDeviceId(StorageInfo::FIXED_MASS_STORAGE,
177                                 path.AsUTF8Unsafe()));
178   *device_info = info;
179   *relative_path = base::FilePath();
180   return true;
181 }
182 
183 // static
FindDevicePathById(const std::string & device_id)184 base::FilePath MediaStorageUtil::FindDevicePathById(
185     const std::string& device_id) {
186   StorageInfo::Type type;
187   std::string unique_id;
188   if (!StorageInfo::CrackDeviceId(device_id, &type, &unique_id))
189     return base::FilePath();
190 
191   if (type == StorageInfo::FIXED_MASS_STORAGE ||
192       type == StorageInfo::ITUNES ||
193       type == StorageInfo::IPHOTO ||
194       type == StorageInfo::PICASA) {
195     // For this type, the unique_id is the path.
196     return base::FilePath::FromUTF8Unsafe(unique_id);
197   }
198 
199   // For ImageCapture, the synthetic filesystem will be rooted at a fake
200   // top-level directory which is the device_id.
201   if (type == StorageInfo::MAC_IMAGE_CAPTURE) {
202 #if !defined(OS_WIN)
203     return base::FilePath(kRootPath + device_id);
204 #endif
205   }
206 
207   DCHECK(type == StorageInfo::MTP_OR_PTP ||
208          type == StorageInfo::REMOVABLE_MASS_STORAGE_WITH_DCIM ||
209          type == StorageInfo::REMOVABLE_MASS_STORAGE_NO_DCIM);
210   return base::FilePath(FindRemovableStorageLocationById(device_id));
211 }
212 
213 // static
RecordDeviceInfoHistogram(bool mass_storage,const std::string & device_uuid,const base::string16 & device_label)214 void MediaStorageUtil::RecordDeviceInfoHistogram(
215     bool mass_storage,
216     const std::string& device_uuid,
217     const base::string16& device_label) {
218   unsigned int event_number = 0;
219   if (!mass_storage)
220     event_number = 4;
221 
222   if (device_label.empty())
223     event_number += 2;
224 
225   if (device_uuid.empty())
226     event_number += 1;
227   enum DeviceInfoHistogramBuckets event =
228       static_cast<enum DeviceInfoHistogramBuckets>(event_number);
229   if (event >= DEVICE_INFO_BUCKET_BOUNDARY) {
230     NOTREACHED();
231     return;
232   }
233   UMA_HISTOGRAM_ENUMERATION("MediaDeviceNotifications.DeviceInfo", event,
234                             DEVICE_INFO_BUCKET_BOUNDARY);
235 }
236 
IsRemovableStorageAttached(const std::string & id)237 bool MediaStorageUtil::IsRemovableStorageAttached(const std::string& id) {
238   StorageInfoList devices =
239       StorageMonitor::GetInstance()->GetAllAvailableStorages();
240   for (StorageInfoList::const_iterator it = devices.begin();
241        it != devices.end(); ++it) {
242     if (StorageInfo::IsRemovableDevice(id) && it->device_id() == id)
243       return true;
244   }
245   return false;
246 }
247