Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 1 | // Copyright (c) 2011 The Chromium OS 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 | |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 5 | #include "cros-disks/cros-disks-server-impl.h" |
| 6 | |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 7 | #include <base/logging.h> |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 8 | #include <chromeos/dbus/service_constants.h> |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 9 | |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 10 | #include "cros-disks/archive-manager.h" |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 11 | #include "cros-disks/device-event.h" |
Ben Chan | bdc3974 | 2011-05-11 17:51:26 -0700 | [diff] [blame] | 12 | #include "cros-disks/disk.h" |
| 13 | #include "cros-disks/disk-manager.h" |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 14 | #include "cros-disks/format-manager.h" |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 15 | #include "cros-disks/platform.h" |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 16 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 17 | using std::string; |
| 18 | using std::vector; |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 19 | |
Ben Chan | 460439f | 2011-09-13 09:16:28 -0700 | [diff] [blame] | 20 | namespace cros_disks { |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 21 | |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 22 | CrosDisksServer::CrosDisksServer(DBus::Connection& connection, // NOLINT |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 23 | Platform* platform, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 24 | DiskManager* disk_manager, |
| 25 | FormatManager* format_manager) |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 26 | : DBus::ObjectAdaptor(connection, kCrosDisksServicePath), |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 27 | platform_(platform), |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 28 | disk_manager_(disk_manager), |
Ben Chan | 89cf29e | 2011-08-10 13:11:05 -0700 | [diff] [blame] | 29 | format_manager_(format_manager) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 30 | CHECK(platform_) << "Invalid platform object"; |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 31 | CHECK(disk_manager_) << "Invalid disk manager object"; |
| 32 | CHECK(format_manager_) << "Invalid format manager object"; |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 33 | |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 34 | InitializeProperties(); |
Ben Chan | c1e766c | 2011-11-21 12:56:59 -0800 | [diff] [blame] | 35 | format_manager_->set_observer(this); |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 36 | } |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 37 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 38 | CrosDisksServer::~CrosDisksServer() { |
| 39 | } |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 40 | |
Ben Chan | f47fc35 | 2013-05-03 11:28:57 -0700 | [diff] [blame^] | 41 | void CrosDisksServer::RegisterMountManager(MountManager* mount_manager) { |
| 42 | CHECK(mount_manager) << "Invalid mount manager object"; |
| 43 | mount_managers_.push_back(mount_manager); |
| 44 | } |
| 45 | |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 46 | bool CrosDisksServer::IsAlive(DBus::Error& error) { // NOLINT |
| 47 | return true; |
| 48 | } |
| 49 | |
Ben Chan | c1e766c | 2011-11-21 12:56:59 -0800 | [diff] [blame] | 50 | void CrosDisksServer::Format(const string& path, |
| 51 | const string& filesystem_type, |
| 52 | const vector<string>& options, |
| 53 | DBus::Error &error) { // NOLINT |
| 54 | FormatDevice(path, filesystem_type, error); |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 55 | } |
| 56 | |
Ben Chan | c1e766c | 2011-11-21 12:56:59 -0800 | [diff] [blame] | 57 | // TODO(benchan): Remove this method after Chrome switches to use Format(). |
| 58 | bool CrosDisksServer::FormatDevice(const string& path, |
| 59 | const string& filesystem_type, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 60 | DBus::Error &error) { // NOLINT |
Ben Chan | c833a52 | 2012-09-27 23:47:00 -0700 | [diff] [blame] | 61 | FormatErrorType error_type = FORMAT_ERROR_NONE; |
| 62 | Disk disk; |
| 63 | if (!disk_manager_->GetDiskByDevicePath(path, &disk)) { |
| 64 | error_type = FORMAT_ERROR_INVALID_DEVICE_PATH; |
| 65 | } else if (disk.is_on_boot_device()) { |
| 66 | error_type = FORMAT_ERROR_DEVICE_NOT_ALLOWED; |
| 67 | } else { |
Toni Barzic | 031dc71 | 2012-11-20 18:15:46 -0800 | [diff] [blame] | 68 | error_type = |
| 69 | format_manager_->StartFormatting(disk.device_file(), filesystem_type); |
Ben Chan | c833a52 | 2012-09-27 23:47:00 -0700 | [diff] [blame] | 70 | } |
| 71 | |
Ben Chan | c1e766c | 2011-11-21 12:56:59 -0800 | [diff] [blame] | 72 | if (error_type != FORMAT_ERROR_NONE) { |
| 73 | LOG(ERROR) << "Could not format device '" << path |
| 74 | << "' as filesystem '" << filesystem_type << "'"; |
| 75 | FormatCompleted(error_type, path); |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 76 | return false; |
| 77 | } |
| 78 | return true; |
| 79 | } |
| 80 | |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 81 | void CrosDisksServer::Mount(const string& path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 82 | const string& filesystem_type, |
| 83 | const vector<string>& options, |
| 84 | DBus::Error& error) { // NOLINT |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame] | 85 | MountErrorType error_type = MOUNT_ERROR_INVALID_PATH; |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 86 | MountSourceType source_type = MOUNT_SOURCE_INVALID; |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 87 | string mount_path; |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 88 | |
| 89 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 90 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 91 | MountManager* manager = *manager_iter; |
| 92 | if (manager->CanMount(path)) { |
| 93 | source_type = manager->GetMountSourceType(); |
| 94 | error_type = manager->Mount(path, filesystem_type, options, &mount_path); |
| 95 | break; |
| 96 | } |
| 97 | } |
| 98 | |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame] | 99 | if (error_type != MOUNT_ERROR_NONE) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 100 | LOG(ERROR) << "Failed to mount '" << path << "'"; |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 101 | } |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 102 | MountCompleted(error_type, path, source_type, mount_path); |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 103 | } |
| 104 | |
| 105 | void CrosDisksServer::Unmount(const string& path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 106 | const vector<string>& options, |
| 107 | DBus::Error& error) { // NOLINT |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame] | 108 | MountErrorType error_type = MOUNT_ERROR_INVALID_PATH; |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 109 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 110 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 111 | MountManager* manager = *manager_iter; |
| 112 | if (manager->CanUnmount(path)) { |
| 113 | error_type = manager->Unmount(path, options); |
| 114 | break; |
| 115 | } |
| 116 | } |
| 117 | |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame] | 118 | if (error_type != MOUNT_ERROR_NONE) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 119 | string message = "Failed to unmount '" + path + "'"; |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 120 | error.set(kCrosDisksServiceError, message.c_str()); |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 121 | } |
| 122 | } |
| 123 | |
Ben Chan | 8f51376 | 2011-11-14 12:44:42 -0800 | [diff] [blame] | 124 | void CrosDisksServer::UnmountAll(DBus::Error& error) { // NOLINT |
| 125 | DoUnmountAll(); |
| 126 | } |
| 127 | |
| 128 | void CrosDisksServer::DoUnmountAll() { |
| 129 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 130 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 131 | MountManager* manager = *manager_iter; |
| 132 | manager->UnmountAll(); |
| 133 | } |
| 134 | } |
| 135 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 136 | vector<string> CrosDisksServer::DoEnumerateDevices( |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 137 | bool auto_mountable_only) const { |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 138 | vector<Disk> disks = disk_manager_->EnumerateDisks(); |
| 139 | vector<string> devices; |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 140 | devices.reserve(disks.size()); |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 141 | for (vector<Disk>::const_iterator disk_iterator = disks.begin(); |
| 142 | disk_iterator != disks.end(); ++disk_iterator) { |
Ben Chan | 1c2d425 | 2011-06-03 13:33:49 -0700 | [diff] [blame] | 143 | if (!auto_mountable_only || disk_iterator->is_auto_mountable()) { |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 144 | devices.push_back(disk_iterator->native_path()); |
| 145 | } |
| 146 | } |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 147 | return devices; |
| 148 | } |
| 149 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 150 | vector<string> CrosDisksServer::EnumerateDevices( |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 151 | DBus::Error& error) { // NOLINT |
| 152 | return DoEnumerateDevices(false); |
| 153 | } |
| 154 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 155 | vector<string> CrosDisksServer::EnumerateAutoMountableDevices( |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 156 | DBus::Error& error) { // NOLINT |
| 157 | return DoEnumerateDevices(true); |
| 158 | } |
| 159 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 160 | DBusDisk CrosDisksServer::GetDeviceProperties(const string& device_path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 161 | DBus::Error& error) { // NOLINT |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 162 | Disk disk; |
| 163 | if (!disk_manager_->GetDiskByDevicePath(device_path, &disk)) { |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 164 | string message = "Could not get the properties of device " + device_path; |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 165 | LOG(ERROR) << message; |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 166 | error.set(kCrosDisksServiceError, message.c_str()); |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 167 | } |
| 168 | return disk.ToDBusFormat(); |
| 169 | } |
| 170 | |
Ben Chan | c1e766c | 2011-11-21 12:56:59 -0800 | [diff] [blame] | 171 | void CrosDisksServer::OnFormatCompleted(const string& device_path, |
| 172 | FormatErrorType error_type) { |
| 173 | // TODO(benchan): Deprecate the FormattingFinished signal once Chrome |
| 174 | // switches to observe the FormatCompleted signal instead. |
| 175 | if (error_type != FORMAT_ERROR_NONE) { |
| 176 | FormattingFinished("!" + device_path); |
| 177 | LOG(ERROR) << "Failed to format '" << device_path << "'"; |
| 178 | } else { |
| 179 | FormattingFinished(device_path); |
| 180 | } |
| 181 | FormatCompleted(error_type, device_path); |
| 182 | } |
| 183 | |
Ben Chan | 5988f29 | 2012-09-18 08:32:42 -0700 | [diff] [blame] | 184 | void CrosDisksServer::OnScreenIsLocked() { |
| 185 | // no-op |
| 186 | } |
| 187 | |
| 188 | void CrosDisksServer::OnScreenIsUnlocked() { |
| 189 | // no-op |
| 190 | } |
| 191 | |
Ben Chan | b3bf8d1 | 2013-04-23 13:57:55 -0700 | [diff] [blame] | 192 | void CrosDisksServer::OnSessionStarted() { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 193 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 194 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 195 | MountManager* manager = *manager_iter; |
Ben Chan | b3bf8d1 | 2013-04-23 13:57:55 -0700 | [diff] [blame] | 196 | manager->StartSession(); |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 197 | } |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 198 | } |
| 199 | |
Ben Chan | b3bf8d1 | 2013-04-23 13:57:55 -0700 | [diff] [blame] | 200 | void CrosDisksServer::OnSessionStopped() { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 201 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 202 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 203 | MountManager* manager = *manager_iter; |
Ben Chan | b3bf8d1 | 2013-04-23 13:57:55 -0700 | [diff] [blame] | 204 | manager->StopSession(); |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 205 | } |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 206 | } |
| 207 | |
| 208 | void CrosDisksServer::DispatchDeviceEvent(const DeviceEvent& event) { |
| 209 | switch (event.event_type) { |
| 210 | case DeviceEvent::kDeviceAdded: |
| 211 | DeviceAdded(event.device_path); |
| 212 | break; |
| 213 | case DeviceEvent::kDeviceScanned: |
| 214 | DeviceScanned(event.device_path); |
| 215 | break; |
| 216 | case DeviceEvent::kDeviceRemoved: |
| 217 | DeviceRemoved(event.device_path); |
| 218 | break; |
| 219 | case DeviceEvent::kDiskAdded: |
| 220 | DiskAdded(event.device_path); |
| 221 | break; |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 222 | case DeviceEvent::kDiskChanged: |
| 223 | DiskChanged(event.device_path); |
| 224 | break; |
| 225 | case DeviceEvent::kDiskRemoved: |
| 226 | DiskRemoved(event.device_path); |
| 227 | break; |
| 228 | default: |
| 229 | break; |
| 230 | } |
| 231 | } |
| 232 | |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 233 | void CrosDisksServer::InitializeProperties() { |
| 234 | try { |
| 235 | DBus::Variant value; |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 236 | value.writer().append_bool(platform_->experimental_features_enabled()); |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 237 | CrosDisks_adaptor::set_property(kExperimentalFeaturesEnabled, value); |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 238 | } catch (const DBus::Error& e) { // NOLINT |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 239 | LOG(FATAL) << "Failed to initialize properties: " << e.what(); |
| 240 | } |
| 241 | } |
| 242 | |
| 243 | void CrosDisksServer::on_set_property( |
| 244 | DBus::InterfaceAdaptor& interface, // NOLINT |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 245 | const string& property, const DBus::Variant& value) { |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 246 | if (property == kExperimentalFeaturesEnabled) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 247 | platform_->set_experimental_features_enabled(value.reader().get_bool()); |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 248 | } |
| 249 | } |
| 250 | |
Ben Chan | bdc3974 | 2011-05-11 17:51:26 -0700 | [diff] [blame] | 251 | } // namespace cros_disks |