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, |
| 24 | ArchiveManager* archive_manager, |
| 25 | DiskManager* disk_manager, |
| 26 | FormatManager* format_manager) |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 27 | : DBus::ObjectAdaptor(connection, kCrosDisksServicePath), |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 28 | platform_(platform), |
| 29 | archive_manager_(archive_manager), |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 30 | disk_manager_(disk_manager), |
Ben Chan | 89cf29e | 2011-08-10 13:11:05 -0700 | [diff] [blame] | 31 | format_manager_(format_manager) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 32 | CHECK(platform_) << "Invalid platform object"; |
| 33 | CHECK(archive_manager_) << "Invalid archive manager object"; |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 34 | CHECK(disk_manager_) << "Invalid disk manager object"; |
| 35 | CHECK(format_manager_) << "Invalid format manager object"; |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 36 | |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 37 | // TODO(benchan): Refactor the code so that we don't have to pass |
| 38 | // DiskManager, ArchiveManager, etc to the constructor |
| 39 | // of CrosDisksServer, but instead pass a list of mount |
| 40 | // managers. |
| 41 | mount_managers_.push_back(disk_manager_); |
| 42 | mount_managers_.push_back(archive_manager_); |
| 43 | |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 44 | InitializeProperties(); |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 45 | format_manager_->set_parent(this); |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 46 | } |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 47 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 48 | CrosDisksServer::~CrosDisksServer() { |
| 49 | } |
Ryan Cairns | ea6505f | 2011-04-10 19:54:53 -0700 | [diff] [blame] | 50 | |
| 51 | bool CrosDisksServer::IsAlive(DBus::Error& error) { // NOLINT |
| 52 | return true; |
| 53 | } |
| 54 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 55 | void CrosDisksServer::SignalFormattingFinished(const string& device_path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 56 | int status) { |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 57 | if (status) { |
Szymon Sidor | ea0ecbd3 | 2011-07-19 16:23:10 -0700 | [diff] [blame] | 58 | FormattingFinished(std::string("!") + device_path); |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 59 | LOG(ERROR) << "Could not format device '" << device_path |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 60 | << "'. Formatting process failed with an exit code " << status; |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 61 | } else { |
Szymon Sidor | ea0ecbd3 | 2011-07-19 16:23:10 -0700 | [diff] [blame] | 62 | FormattingFinished(device_path); |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 63 | } |
| 64 | } |
| 65 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 66 | bool CrosDisksServer::FormatDevice(const string& device_path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 67 | const string& filesystem, |
| 68 | DBus::Error &error) { // NOLINT |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 69 | if (!format_manager_->StartFormatting(device_path, filesystem)) { |
| 70 | LOG(ERROR) << "Could not format device " << device_path |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 71 | << " as file system '" << filesystem << "'"; |
Szymon Sidor | 2733b51 | 2011-06-30 18:00:51 -0700 | [diff] [blame] | 72 | return false; |
| 73 | } |
| 74 | return true; |
| 75 | } |
| 76 | |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 77 | void CrosDisksServer::Mount(const string& path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 78 | const string& filesystem_type, |
| 79 | const vector<string>& options, |
| 80 | DBus::Error& error) { // NOLINT |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame^] | 81 | MountErrorType error_type = MOUNT_ERROR_INVALID_PATH; |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 82 | MountSourceType source_type = MOUNT_SOURCE_INVALID; |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 83 | string mount_path; |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 84 | |
| 85 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 86 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 87 | MountManager* manager = *manager_iter; |
| 88 | if (manager->CanMount(path)) { |
| 89 | source_type = manager->GetMountSourceType(); |
| 90 | error_type = manager->Mount(path, filesystem_type, options, &mount_path); |
| 91 | break; |
| 92 | } |
| 93 | } |
| 94 | |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame^] | 95 | if (error_type != MOUNT_ERROR_NONE) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 96 | LOG(ERROR) << "Failed to mount '" << path << "'"; |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 97 | } |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 98 | MountCompleted(error_type, path, source_type, mount_path); |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 99 | } |
| 100 | |
| 101 | void CrosDisksServer::Unmount(const string& path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 102 | const vector<string>& options, |
| 103 | DBus::Error& error) { // NOLINT |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame^] | 104 | MountErrorType error_type = MOUNT_ERROR_INVALID_PATH; |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 105 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 106 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 107 | MountManager* manager = *manager_iter; |
| 108 | if (manager->CanUnmount(path)) { |
| 109 | error_type = manager->Unmount(path, options); |
| 110 | break; |
| 111 | } |
| 112 | } |
| 113 | |
Ben Chan | fcb2fc0 | 2011-11-21 09:44:07 -0800 | [diff] [blame^] | 114 | if (error_type != MOUNT_ERROR_NONE) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 115 | string message = "Failed to unmount '" + path + "'"; |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 116 | error.set(kCrosDisksServiceError, message.c_str()); |
Ben Chan | d3fdc72 | 2011-07-21 18:15:22 -0700 | [diff] [blame] | 117 | } |
| 118 | } |
| 119 | |
Ben Chan | 8f51376 | 2011-11-14 12:44:42 -0800 | [diff] [blame] | 120 | void CrosDisksServer::UnmountAll(DBus::Error& error) { // NOLINT |
| 121 | DoUnmountAll(); |
| 122 | } |
| 123 | |
| 124 | void CrosDisksServer::DoUnmountAll() { |
| 125 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 126 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 127 | MountManager* manager = *manager_iter; |
| 128 | manager->UnmountAll(); |
| 129 | } |
| 130 | } |
| 131 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 132 | vector<string> CrosDisksServer::DoEnumerateDevices( |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 133 | bool auto_mountable_only) const { |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 134 | vector<Disk> disks = disk_manager_->EnumerateDisks(); |
| 135 | vector<string> devices; |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 136 | devices.reserve(disks.size()); |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 137 | for (vector<Disk>::const_iterator disk_iterator = disks.begin(); |
| 138 | disk_iterator != disks.end(); ++disk_iterator) { |
Ben Chan | 1c2d425 | 2011-06-03 13:33:49 -0700 | [diff] [blame] | 139 | if (!auto_mountable_only || disk_iterator->is_auto_mountable()) { |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 140 | devices.push_back(disk_iterator->native_path()); |
| 141 | } |
| 142 | } |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 143 | return devices; |
| 144 | } |
| 145 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 146 | vector<string> CrosDisksServer::EnumerateDevices( |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 147 | DBus::Error& error) { // NOLINT |
| 148 | return DoEnumerateDevices(false); |
| 149 | } |
| 150 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 151 | vector<string> CrosDisksServer::EnumerateAutoMountableDevices( |
Ben Chan | 490319f | 2011-05-06 14:00:42 -0700 | [diff] [blame] | 152 | DBus::Error& error) { // NOLINT |
| 153 | return DoEnumerateDevices(true); |
| 154 | } |
| 155 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 156 | DBusDisk CrosDisksServer::GetDeviceProperties(const string& device_path, |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 157 | DBus::Error& error) { // NOLINT |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 158 | Disk disk; |
| 159 | if (!disk_manager_->GetDiskByDevicePath(device_path, &disk)) { |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 160 | string message = "Could not get the properties of device " + device_path; |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 161 | LOG(ERROR) << message; |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 162 | error.set(kCrosDisksServiceError, message.c_str()); |
Ben Chan | f51ff00 | 2011-04-25 12:41:57 -0700 | [diff] [blame] | 163 | } |
| 164 | return disk.ToDBusFormat(); |
| 165 | } |
| 166 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 167 | void CrosDisksServer::OnSessionStarted(const string& user) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 168 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 169 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 170 | MountManager* manager = *manager_iter; |
| 171 | manager->StartSession(user); |
| 172 | } |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 173 | } |
| 174 | |
Ben Chan | 190d3cf | 2011-07-07 09:38:48 -0700 | [diff] [blame] | 175 | void CrosDisksServer::OnSessionStopped(const string& user) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 176 | for (vector<MountManager*>::iterator manager_iter = mount_managers_.begin(); |
| 177 | manager_iter != mount_managers_.end(); ++manager_iter) { |
| 178 | MountManager* manager = *manager_iter; |
| 179 | manager->StopSession(user); |
| 180 | } |
Ben Chan | 6e72692 | 2011-06-28 15:54:32 -0700 | [diff] [blame] | 181 | } |
| 182 | |
| 183 | void CrosDisksServer::DispatchDeviceEvent(const DeviceEvent& event) { |
| 184 | switch (event.event_type) { |
| 185 | case DeviceEvent::kDeviceAdded: |
| 186 | DeviceAdded(event.device_path); |
| 187 | break; |
| 188 | case DeviceEvent::kDeviceScanned: |
| 189 | DeviceScanned(event.device_path); |
| 190 | break; |
| 191 | case DeviceEvent::kDeviceRemoved: |
| 192 | DeviceRemoved(event.device_path); |
| 193 | break; |
| 194 | case DeviceEvent::kDiskAdded: |
| 195 | DiskAdded(event.device_path); |
| 196 | break; |
| 197 | case DeviceEvent::kDiskAddedAfterRemoved: |
| 198 | DiskRemoved(event.device_path); |
| 199 | DiskAdded(event.device_path); |
| 200 | break; |
| 201 | case DeviceEvent::kDiskChanged: |
| 202 | DiskChanged(event.device_path); |
| 203 | break; |
| 204 | case DeviceEvent::kDiskRemoved: |
| 205 | DiskRemoved(event.device_path); |
| 206 | break; |
| 207 | default: |
| 208 | break; |
| 209 | } |
| 210 | } |
| 211 | |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 212 | void CrosDisksServer::InitializeProperties() { |
| 213 | try { |
| 214 | DBus::Variant value; |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 215 | value.writer().append_bool(platform_->experimental_features_enabled()); |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 216 | CrosDisks_adaptor::set_property(kExperimentalFeaturesEnabled, value); |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 217 | } catch (const DBus::Error& e) { // NOLINT |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 218 | LOG(FATAL) << "Failed to initialize properties: " << e.what(); |
| 219 | } |
| 220 | } |
| 221 | |
| 222 | void CrosDisksServer::on_set_property( |
| 223 | DBus::InterfaceAdaptor& interface, // NOLINT |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 224 | const string& property, const DBus::Variant& value) { |
Ben Chan | 6d0b272 | 2011-11-18 08:24:14 -0800 | [diff] [blame] | 225 | if (property == kExperimentalFeaturesEnabled) { |
Ben Chan | 8dcede8 | 2011-07-25 20:56:13 -0700 | [diff] [blame] | 226 | platform_->set_experimental_features_enabled(value.reader().get_bool()); |
Ben Chan | b092d75 | 2011-07-13 11:44:38 -0700 | [diff] [blame] | 227 | } |
| 228 | } |
| 229 | |
Ben Chan | bdc3974 | 2011-05-11 17:51:26 -0700 | [diff] [blame] | 230 | } // namespace cros_disks |