blob: 061ffdf99c725e2276597afb14bbdf85e6b07da1 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// StorageMonitorLinux implementation.
#include "components/storage_monitor/storage_monitor_linux.h"
#include <mntent.h>
#include <stdio.h>
#include <list>
#include "base/basictypes.h"
#include "base/bind.h"
#include "base/metrics/histogram.h"
#include "base/process/kill.h"
#include "base/process/launch.h"
#include "base/process/process.h"
#include "base/stl_util.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "components/storage_monitor/media_storage_util.h"
#include "components/storage_monitor/media_transfer_protocol_device_observer_linux.h"
#include "components/storage_monitor/removable_device_constants.h"
#include "components/storage_monitor/storage_info.h"
#include "components/storage_monitor/udev_util_linux.h"
#include "device/media_transfer_protocol/media_transfer_protocol_manager.h"
#include "device/udev_linux/scoped_udev.h"
using content::BrowserThread;
namespace storage_monitor {
typedef MtabWatcherLinux::MountPointDeviceMap MountPointDeviceMap;
namespace {
// udev device property constants.
const char kBlockSubsystemKey[] = "block";
const char kDiskDeviceTypeKey[] = "disk";
const char kFsUUID[] = "ID_FS_UUID";
const char kLabel[] = "ID_FS_LABEL";
const char kModel[] = "ID_MODEL";
const char kModelID[] = "ID_MODEL_ID";
const char kRemovableSysAttr[] = "removable";
const char kSerialShort[] = "ID_SERIAL_SHORT";
const char kSizeSysAttr[] = "size";
const char kVendor[] = "ID_VENDOR";
const char kVendorID[] = "ID_VENDOR_ID";
// Construct a device id using label or manufacturer (vendor and model) details.
std::string MakeDeviceUniqueId(struct udev_device* device) {
std::string uuid = device::UdevDeviceGetPropertyValue(device, kFsUUID);
// Keep track of device uuid, to see how often we receive empty uuid values.
UMA_HISTOGRAM_BOOLEAN(
"RemovableDeviceNotificationsLinux.device_file_system_uuid_available",
!uuid.empty());
if (!uuid.empty())
return kFSUniqueIdPrefix + uuid;
// If one of the vendor, model, serial information is missing, its value
// in the string is empty.
// Format: VendorModelSerial:VendorInfo:ModelInfo:SerialShortInfo
// E.g.: VendorModelSerial:Kn:DataTravel_12.10:8000000000006CB02CDB
std::string vendor = device::UdevDeviceGetPropertyValue(device, kVendorID);
std::string model = device::UdevDeviceGetPropertyValue(device, kModelID);
std::string serial_short =
device::UdevDeviceGetPropertyValue(device, kSerialShort);
if (vendor.empty() && model.empty() && serial_short.empty())
return std::string();
return kVendorModelSerialPrefix + vendor + ":" + model + ":" + serial_short;
}
// Records GetDeviceInfo result on destruction, to see how often we fail to get
// device details.
class ScopedGetDeviceInfoResultRecorder {
public:
ScopedGetDeviceInfoResultRecorder() : result_(false) {}
~ScopedGetDeviceInfoResultRecorder() {
UMA_HISTOGRAM_BOOLEAN("MediaDeviceNotification.UdevRequestSuccess",
result_);
}
void set_result(bool result) {
result_ = result;
}
private:
bool result_;
DISALLOW_COPY_AND_ASSIGN(ScopedGetDeviceInfoResultRecorder);
};
// Returns the storage partition size of the device specified by |device_path|.
// If the requested information is unavailable, returns 0.
uint64 GetDeviceStorageSize(const base::FilePath& device_path,
struct udev_device* device) {
// sysfs provides the device size in units of 512-byte blocks.
const std::string partition_size =
device::UdevDeviceGetSysattrValue(device, kSizeSysAttr);
// Keep track of device size, to see how often this information is
// unavailable.
UMA_HISTOGRAM_BOOLEAN(
"RemovableDeviceNotificationsLinux.device_partition_size_available",
!partition_size.empty());
uint64 total_size_in_bytes = 0;
if (!base::StringToUint64(partition_size, &total_size_in_bytes))
return 0;
return (total_size_in_bytes <= kuint64max / 512) ?
total_size_in_bytes * 512 : 0;
}
// Gets the device information using udev library.
scoped_ptr<StorageInfo> GetDeviceInfo(const base::FilePath& device_path,
const base::FilePath& mount_point) {
DCHECK_CURRENTLY_ON(BrowserThread::FILE);
DCHECK(!device_path.empty());
scoped_ptr<StorageInfo> storage_info;
ScopedGetDeviceInfoResultRecorder results_recorder;
device::ScopedUdevPtr udev_obj(device::udev_new());
if (!udev_obj.get())
return storage_info.Pass();
struct stat device_stat;
if (stat(device_path.value().c_str(), &device_stat) < 0)
return storage_info.Pass();
char device_type;
if (S_ISCHR(device_stat.st_mode))
device_type = 'c';
else if (S_ISBLK(device_stat.st_mode))
device_type = 'b';
else
return storage_info.Pass(); // Not a supported type.
device::ScopedUdevDevicePtr device(
device::udev_device_new_from_devnum(udev_obj.get(), device_type,
device_stat.st_rdev));
if (!device.get())
return storage_info.Pass();
base::string16 volume_label = base::UTF8ToUTF16(
device::UdevDeviceGetPropertyValue(device.get(), kLabel));
base::string16 vendor_name = base::UTF8ToUTF16(
device::UdevDeviceGetPropertyValue(device.get(), kVendor));
base::string16 model_name = base::UTF8ToUTF16(
device::UdevDeviceGetPropertyValue(device.get(), kModel));
std::string unique_id = MakeDeviceUniqueId(device.get());
// Keep track of device info details to see how often we get invalid values.
MediaStorageUtil::RecordDeviceInfoHistogram(true, unique_id, volume_label);
const char* value =
device::udev_device_get_sysattr_value(device.get(), kRemovableSysAttr);
if (!value) {
// |parent_device| is owned by |device| and does not need to be cleaned
// up.
struct udev_device* parent_device =
device::udev_device_get_parent_with_subsystem_devtype(
device.get(),
kBlockSubsystemKey,
kDiskDeviceTypeKey);
value = device::udev_device_get_sysattr_value(parent_device,
kRemovableSysAttr);
}
const bool is_removable = (value && atoi(value) == 1);
StorageInfo::Type type = StorageInfo::FIXED_MASS_STORAGE;
if (is_removable) {
if (MediaStorageUtil::HasDcim(mount_point))
type = StorageInfo::REMOVABLE_MASS_STORAGE_WITH_DCIM;
else
type = StorageInfo::REMOVABLE_MASS_STORAGE_NO_DCIM;
}
results_recorder.set_result(true);
storage_info.reset(new StorageInfo(
StorageInfo::MakeDeviceId(type, unique_id),
mount_point.value(),
volume_label,
vendor_name,
model_name,
GetDeviceStorageSize(device_path, device.get())));
return storage_info.Pass();
}
MtabWatcherLinux* CreateMtabWatcherLinuxOnFileThread(
const base::FilePath& mtab_path,
base::WeakPtr<MtabWatcherLinux::Delegate> delegate) {
DCHECK_CURRENTLY_ON(BrowserThread::FILE);
// Owned by caller.
return new MtabWatcherLinux(mtab_path, delegate);
}
StorageMonitor::EjectStatus EjectPathOnFileThread(
const base::FilePath& path,
const base::FilePath& device) {
DCHECK_CURRENTLY_ON(BrowserThread::FILE);
// Note: Linux LSB says umount should exist in /bin.
static const char kUmountBinary[] = "/bin/umount";
std::vector<std::string> command;
command.push_back(kUmountBinary);
command.push_back(path.value());
base::LaunchOptions options;
base::Process process = base::LaunchProcess(command, options);
if (!process.IsValid())
return StorageMonitor::EJECT_FAILURE;
int exit_code = -1;
if (!process.WaitForExitWithTimeout(base::TimeDelta::FromMilliseconds(3000),
&exit_code)) {
process.Terminate(-1, false);
base::EnsureProcessTerminated(process.Pass());
return StorageMonitor::EJECT_FAILURE;
}
// TODO(gbillock): Make sure this is found in documentation
// somewhere. Experimentally it seems to hold that exit code
// 1 means device is in use.
if (exit_code == 1)
return StorageMonitor::EJECT_IN_USE;
if (exit_code != 0)
return StorageMonitor::EJECT_FAILURE;
return StorageMonitor::EJECT_OK;
}
} // namespace
StorageMonitorLinux::StorageMonitorLinux(const base::FilePath& path)
: mtab_path_(path),
get_device_info_callback_(base::Bind(&GetDeviceInfo)),
weak_ptr_factory_(this) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
}
StorageMonitorLinux::~StorageMonitorLinux() {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
}
void StorageMonitorLinux::Init() {
DCHECK(!mtab_path_.empty());
BrowserThread::PostTaskAndReplyWithResult(
BrowserThread::FILE, FROM_HERE,
base::Bind(&CreateMtabWatcherLinuxOnFileThread,
mtab_path_,
weak_ptr_factory_.GetWeakPtr()),
base::Bind(&StorageMonitorLinux::OnMtabWatcherCreated,
weak_ptr_factory_.GetWeakPtr()));
if (!media_transfer_protocol_manager_) {
media_transfer_protocol_manager_.reset(
device::MediaTransferProtocolManager::Initialize(
BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE)));
}
media_transfer_protocol_device_observer_.reset(
new MediaTransferProtocolDeviceObserverLinux(
receiver(), media_transfer_protocol_manager_.get()));
}
bool StorageMonitorLinux::GetStorageInfoForPath(
const base::FilePath& path,
StorageInfo* device_info) const {
DCHECK(device_info);
DCHECK_CURRENTLY_ON(BrowserThread::UI);
// TODO(thestig) |media_transfer_protocol_device_observer_| should always be
// valid.
if (media_transfer_protocol_device_observer_ &&
media_transfer_protocol_device_observer_->GetStorageInfoForPath(
path, device_info)) {
return true;
}
if (!path.IsAbsolute())
return false;
base::FilePath current = path;
while (!ContainsKey(mount_info_map_, current) && current != current.DirName())
current = current.DirName();
MountMap::const_iterator mount_info = mount_info_map_.find(current);
if (mount_info == mount_info_map_.end())
return false;
*device_info = mount_info->second.storage_info;
return true;
}
device::MediaTransferProtocolManager*
StorageMonitorLinux::media_transfer_protocol_manager() {
return media_transfer_protocol_manager_.get();
}
void StorageMonitorLinux::SetGetDeviceInfoCallbackForTest(
const GetDeviceInfoCallback& get_device_info_callback) {
get_device_info_callback_ = get_device_info_callback;
}
void StorageMonitorLinux::SetMediaTransferProtocolManagerForTest(
device::MediaTransferProtocolManager* test_manager) {
DCHECK(!media_transfer_protocol_manager_);
media_transfer_protocol_manager_.reset(test_manager);
}
void StorageMonitorLinux::EjectDevice(
const std::string& device_id,
base::Callback<void(EjectStatus)> callback) {
StorageInfo::Type type;
if (!StorageInfo::CrackDeviceId(device_id, &type, NULL)) {
callback.Run(EJECT_FAILURE);
return;
}
if (type == StorageInfo::MTP_OR_PTP) {
media_transfer_protocol_device_observer_->EjectDevice(device_id, callback);
return;
}
// Find the mount point for the given device ID.
base::FilePath path;
base::FilePath device;
for (MountMap::iterator mount_info = mount_info_map_.begin();
mount_info != mount_info_map_.end(); ++mount_info) {
if (mount_info->second.storage_info.device_id() == device_id) {
path = mount_info->first;
device = mount_info->second.mount_device;
mount_info_map_.erase(mount_info);
break;
}
}
if (path.empty()) {
callback.Run(EJECT_NO_SUCH_DEVICE);
return;
}
receiver()->ProcessDetach(device_id);
BrowserThread::PostTaskAndReplyWithResult(
BrowserThread::FILE, FROM_HERE,
base::Bind(&EjectPathOnFileThread, path, device),
callback);
}
void StorageMonitorLinux::OnMtabWatcherCreated(MtabWatcherLinux* watcher) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
mtab_watcher_.reset(watcher);
}
void StorageMonitorLinux::UpdateMtab(const MountPointDeviceMap& new_mtab) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
// Check existing mtab entries for unaccounted mount points.
// These mount points must have been removed in the new mtab.
std::list<base::FilePath> mount_points_to_erase;
std::list<base::FilePath> multiple_mounted_devices_needing_reattachment;
for (MountMap::const_iterator old_iter = mount_info_map_.begin();
old_iter != mount_info_map_.end(); ++old_iter) {
const base::FilePath& mount_point = old_iter->first;
const base::FilePath& mount_device = old_iter->second.mount_device;
MountPointDeviceMap::const_iterator new_iter = new_mtab.find(mount_point);
// |mount_point| not in |new_mtab| or |mount_device| is no longer mounted at
// |mount_point|.
if (new_iter == new_mtab.end() || (new_iter->second != mount_device)) {
MountPriorityMap::iterator priority =
mount_priority_map_.find(mount_device);
DCHECK(priority != mount_priority_map_.end());
ReferencedMountPoint::const_iterator has_priority =
priority->second.find(mount_point);
if (StorageInfo::IsRemovableDevice(
old_iter->second.storage_info.device_id())) {
DCHECK(has_priority != priority->second.end());
if (has_priority->second) {
receiver()->ProcessDetach(old_iter->second.storage_info.device_id());
}
if (priority->second.size() > 1)
multiple_mounted_devices_needing_reattachment.push_back(mount_device);
}
priority->second.erase(mount_point);
if (priority->second.empty())
mount_priority_map_.erase(mount_device);
mount_points_to_erase.push_back(mount_point);
}
}
// Erase the |mount_info_map_| entries afterwards. Erasing in the loop above
// using the iterator is slightly more efficient, but more tricky, since
// calling std::map::erase() on an iterator invalidates it.
for (std::list<base::FilePath>::const_iterator it =
mount_points_to_erase.begin();
it != mount_points_to_erase.end();
++it) {
mount_info_map_.erase(*it);
}
// For any multiply mounted device where the mount that we had notified
// got detached, send a notification of attachment for one of the other
// mount points.
for (std::list<base::FilePath>::const_iterator it =
multiple_mounted_devices_needing_reattachment.begin();
it != multiple_mounted_devices_needing_reattachment.end();
++it) {
ReferencedMountPoint::iterator first_mount_point_info =
mount_priority_map_.find(*it)->second.begin();
const base::FilePath& mount_point = first_mount_point_info->first;
first_mount_point_info->second = true;
const StorageInfo& mount_info =
mount_info_map_.find(mount_point)->second.storage_info;
DCHECK(StorageInfo::IsRemovableDevice(mount_info.device_id()));
receiver()->ProcessAttach(mount_info);
}
// Check new mtab entries against existing ones.
for (MountPointDeviceMap::const_iterator new_iter = new_mtab.begin();
new_iter != new_mtab.end(); ++new_iter) {
const base::FilePath& mount_point = new_iter->first;
const base::FilePath& mount_device = new_iter->second;
MountMap::iterator old_iter = mount_info_map_.find(mount_point);
if (old_iter == mount_info_map_.end() ||
old_iter->second.mount_device != mount_device) {
// New mount point found or an existing mount point found with a new
// device.
if (IsDeviceAlreadyMounted(mount_device)) {
HandleDeviceMountedMultipleTimes(mount_device, mount_point);
} else {
BrowserThread::PostTaskAndReplyWithResult(
BrowserThread::FILE, FROM_HERE,
base::Bind(get_device_info_callback_, mount_device, mount_point),
base::Bind(&StorageMonitorLinux::AddNewMount,
weak_ptr_factory_.GetWeakPtr(),
mount_device));
}
}
}
// Note: relies on scheduled tasks on the file thread being sequential. This
// block needs to follow the for loop, so that the DoNothing call on the FILE
// thread happens after the scheduled metadata retrievals, meaning that the
// reply callback will then happen after all the AddNewMount calls.
if (!IsInitialized()) {
BrowserThread::PostTaskAndReply(
BrowserThread::FILE, FROM_HERE,
base::Bind(&base::DoNothing),
base::Bind(&StorageMonitorLinux::MarkInitialized,
weak_ptr_factory_.GetWeakPtr()));
}
}
bool StorageMonitorLinux::IsDeviceAlreadyMounted(
const base::FilePath& mount_device) const {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
return ContainsKey(mount_priority_map_, mount_device);
}
void StorageMonitorLinux::HandleDeviceMountedMultipleTimes(
const base::FilePath& mount_device,
const base::FilePath& mount_point) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
MountPriorityMap::iterator priority = mount_priority_map_.find(mount_device);
DCHECK(priority != mount_priority_map_.end());
const base::FilePath& other_mount_point = priority->second.begin()->first;
priority->second[mount_point] = false;
mount_info_map_[mount_point] =
mount_info_map_.find(other_mount_point)->second;
}
void StorageMonitorLinux::AddNewMount(const base::FilePath& mount_device,
scoped_ptr<StorageInfo> storage_info) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
if (!storage_info)
return;
DCHECK(!storage_info->device_id().empty());
bool removable = StorageInfo::IsRemovableDevice(storage_info->device_id());
const base::FilePath mount_point(storage_info->location());
MountPointInfo mount_point_info;
mount_point_info.mount_device = mount_device;
mount_point_info.storage_info = *storage_info;
mount_info_map_[mount_point] = mount_point_info;
mount_priority_map_[mount_device][mount_point] = removable;
receiver()->ProcessAttach(*storage_info);
}
StorageMonitor* StorageMonitor::CreateInternal() {
const base::FilePath kDefaultMtabPath("/etc/mtab");
return new StorageMonitorLinux(kDefaultMtabPath);
}
} // namespace storage_monitor