// // 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. // #include "update_engine/image_properties.h" #include #include #include #include #include #include #include #include #include "update_engine/common/boot_control_interface.h" #include "update_engine/common/constants.h" #include "update_engine/common/platform_constants.h" #include "update_engine/common/prefs_interface.h" #include "update_engine/common/utils.h" #include "update_engine/system_state.h" using android::base::GetProperty; using std::string; namespace chromeos_update_engine { namespace { // Build time properties name used in Android Things. const char kProductId[] = "product_id"; const char kProductVersion[] = "product_version"; const char kSystemId[] = "system_id"; const char kSystemVersion[] = "system_version"; // The path to the product_components file which stores the version of each // components in OEM partition. const char kProductComponentsPath[] = "/oem/os-release.d/product_components"; // Prefs used to store the powerwash settings. const char kPrefsImgPropPowerwashAllowed[] = "img-prop-powerwash-allowed"; // System properties that identifies the "board". const char kPropProductName[] = "ro.product.name"; const char kPropBuildFingerprint[] = "ro.build.fingerprint"; const char kPropBuildType[] = "ro.build.type"; // Default channel from factory.prop const char kPropDefaultChannel[] = "ro.update.default_channel"; // A prefix added to the path, used for testing. const char* root_prefix = nullptr; string GetStringWithDefault(const brillo::OsReleaseReader& osrelease, const string& key, const string& default_value) { string result; if (osrelease.GetString(key, &result)) return result; LOG(INFO) << "Cannot load ImageProperty " << key << ", using default value " << default_value; return default_value; } // Open misc partition for read or write and output the fd in |out_fd|. bool OpenMisc(bool write, int* out_fd) { string misc_device; int flags = write ? O_WRONLY | O_SYNC : O_RDONLY; if (root_prefix) { // Use a file for unittest and create one if doesn't exist. misc_device = base::FilePath(root_prefix).Append("misc").value(); if (write) flags |= O_CREAT; } else { string err; misc_device = get_bootloader_message_blk_device(&err); if (misc_device.empty()) { LOG(ERROR) << "Unable to get misc block device: " << err; return false; } } int fd = HANDLE_EINTR(open(misc_device.c_str(), flags, 0600)); if (fd < 0) { PLOG(ERROR) << "Opening misc failed"; return false; } *out_fd = fd; return true; } // The offset and size of the channel field in misc partition. constexpr size_t kChannelOffset = BOOTLOADER_MESSAGE_OFFSET_IN_MISC + offsetof(bootloader_message_ab, update_channel); constexpr size_t kChannelSize = sizeof(bootloader_message_ab::update_channel); // Read channel from misc partition to |out_channel|, return false if unable to // read misc or no channel is set in misc. bool ReadChannelFromMisc(string* out_channel) { int fd; TEST_AND_RETURN_FALSE(OpenMisc(false, &fd)); ScopedFdCloser fd_closer(&fd); char channel[kChannelSize] = {0}; ssize_t bytes_read = 0; if (!utils::PReadAll( fd, channel, kChannelSize - 1, kChannelOffset, &bytes_read) || bytes_read != kChannelSize - 1) { PLOG(ERROR) << "Reading update channel from misc failed"; return false; } if (channel[0] == '\0') { LOG(INFO) << "No channel set in misc."; return false; } if (!base::EndsWith(channel, "-channel", base::CompareCase::SENSITIVE)) { LOG(ERROR) << "Channel " << channel << " doesn't end with -channel."; return false; } out_channel->assign(channel); return true; } // Write |in_channel| to misc partition, return false if failed to write. bool WriteChannelToMisc(const string& in_channel) { int fd; TEST_AND_RETURN_FALSE(OpenMisc(true, &fd)); ScopedFdCloser fd_closer(&fd); if (in_channel.size() >= kChannelSize) { LOG(ERROR) << "Channel name is too long: " << in_channel << ", the maximum length is " << kChannelSize - 1; return false; } char channel[kChannelSize] = {0}; memcpy(channel, in_channel.data(), in_channel.size()); if (!utils::PWriteAll(fd, channel, kChannelSize, kChannelOffset)) { PLOG(ERROR) << "Writing update channel to misc failed"; return false; } return true; } string GetTargetChannel() { string channel; if (!ReadChannelFromMisc(&channel)) channel = GetProperty(kPropDefaultChannel, "stable-channel"); return channel; } } // namespace namespace test { void SetImagePropertiesRootPrefix(const char* test_root_prefix) { root_prefix = test_root_prefix; } } // namespace test ImageProperties LoadImageProperties(SystemState* system_state) { ImageProperties result; brillo::OsReleaseReader osrelease; if (root_prefix) osrelease.LoadTestingOnly(base::FilePath(root_prefix)); else osrelease.Load(); result.product_id = GetStringWithDefault(osrelease, kProductId, "invalid-product"); result.system_id = GetStringWithDefault( osrelease, kSystemId, "developer-boards:brillo-starter-board"); // Update the system id to match the prefix of product id for testing. string prefix, not_used, system_id; if (brillo::string_utils::SplitAtFirst( result.product_id, ":", &prefix, ¬_used, false) && brillo::string_utils::SplitAtFirst( result.system_id, ":", ¬_used, &system_id, false)) { result.system_id = prefix + ":" + system_id; } result.canary_product_id = result.product_id; result.version = GetStringWithDefault(osrelease, kProductVersion, "0.0.0.0"); result.system_version = GetStringWithDefault(osrelease, kSystemVersion, "0.0.0.0"); // Can't read it with OsReleaseReader because it has multiple lines. utils::ReadFile(kProductComponentsPath, &result.product_components); result.board = GetProperty(kPropProductName, "brillo"); result.build_fingerprint = GetProperty(kPropBuildFingerprint, "none"); result.build_type = GetProperty(kPropBuildType, ""); // Android doesn't have channel information in system image, we try to read // the channel of current slot from prefs and then fallback to use the // persisted target channel as current channel. string current_channel_key = kPrefsChannelOnSlotPrefix + std::to_string(system_state->boot_control()->GetCurrentSlot()); string current_channel; if (!system_state->prefs()->Exists(current_channel_key) || !system_state->prefs()->GetString(current_channel_key, ¤t_channel)) current_channel = GetTargetChannel(); result.current_channel = current_channel; result.allow_arbitrary_channels = true; // Brillo only supports the official omaha URL. result.omaha_url = constants::kOmahaDefaultProductionURL; return result; } MutableImageProperties LoadMutableImageProperties(SystemState* system_state) { MutableImageProperties result; result.target_channel = GetTargetChannel(); if (!system_state->prefs()->GetBoolean(kPrefsImgPropPowerwashAllowed, &result.is_powerwash_allowed)) { result.is_powerwash_allowed = false; } return result; } bool StoreMutableImageProperties(SystemState* system_state, const MutableImageProperties& properties) { bool ret = true; if (!WriteChannelToMisc(properties.target_channel)) ret = false; if (!system_state->prefs()->SetBoolean(kPrefsImgPropPowerwashAllowed, properties.is_powerwash_allowed)) ret = false; return ret; } void LogImageProperties() { // TODO(*): Implement this. } } // namespace chromeos_update_engine