// // Copyright (C) 2015 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #ifndef UPDATE_ENGINE_CROS_BOOT_CONTROL_CHROMEOS_H_ #define UPDATE_ENGINE_CROS_BOOT_CONTROL_CHROMEOS_H_ #include #include #include #include // for FRIEND_TEST #include "update_engine/common/boot_control_interface.h" #include "update_engine/common/dynamic_partition_control_interface.h" namespace chromeos_update_engine { // The Chrome OS implementation of the BootControlInterface. This interface // assumes the partition names and numbers used in Chrome OS devices. class BootControlChromeOS : public BootControlInterface { public: BootControlChromeOS() = default; ~BootControlChromeOS() = default; // Initialize the BootControl instance loading the constant values. Returns // whether the operation succeeded. In case of failure, normally meaning // some critical failure such as we couldn't determine the slot that we // booted from, the implementation will pretend that there's only one slot and // therefore A/B updates are disabled. bool Init(); // BootControlInterface overrides. unsigned int GetNumSlots() const override; BootControlInterface::Slot GetCurrentSlot() const override; bool GetPartitionDevice(const std::string& partition_name, BootControlInterface::Slot slot, bool not_in_payload, std::string* device, bool* is_dynamic) const override; bool GetPartitionDevice(const std::string& partition_name, BootControlInterface::Slot slot, std::string* device) const override; bool IsSlotBootable(BootControlInterface::Slot slot) const override; bool MarkSlotUnbootable(BootControlInterface::Slot slot) override; bool SetActiveBootSlot(BootControlInterface::Slot slot) override; bool MarkBootSuccessfulAsync(base::Callback callback) override; bool IsSlotMarkedSuccessful(BootControlInterface::Slot slot) const override; DynamicPartitionControlInterface* GetDynamicPartitionControl() override; private: friend class BootControlChromeOSTest; FRIEND_TEST(BootControlChromeOSTest, SysfsBlockDeviceTest); FRIEND_TEST(BootControlChromeOSTest, GetPartitionNumberTest); FRIEND_TEST(BootControlChromeOSTest, ParseDlcPartitionNameTest); // Returns the sysfs block device for a root block device. For example, // SysfsBlockDevice("/dev/sda") returns "/sys/block/sda". Returns an empty // string if the input device is not of the "/dev/xyz" form. static std::string SysfsBlockDevice(const std::string& device); // Returns true if the root |device| (e.g., "/dev/sdb") is known to be // removable, false otherwise. static bool IsRemovableDevice(const std::string& device); // Return the hard-coded partition number used in Chrome OS for the passed // |partition_name| and |slot|. In case of invalid data, returns -1. int GetPartitionNumber(const std::string partition_name, BootControlInterface::Slot slot) const; // Extracts DLC module ID and package ID from partition name. The structure of // the partition name is dlc//. For example: // dlc/fake-dlc/fake-package bool ParseDlcPartitionName(const std::string partition_name, std::string* dlc_id, std::string* dlc_package) const; // Cached values for GetNumSlots() and GetCurrentSlot(). BootControlInterface::Slot num_slots_{1}; BootControlInterface::Slot current_slot_{BootControlInterface::kInvalidSlot}; // The block device of the disk we booted from, without the partition number. std::string boot_disk_name_; std::unique_ptr dynamic_partition_control_; DISALLOW_COPY_AND_ASSIGN(BootControlChromeOS); }; } // namespace chromeos_update_engine #endif // UPDATE_ENGINE_CROS_BOOT_CONTROL_CHROMEOS_H_