| // |
| // 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 <fcntl.h> |
| |
| #include <string> |
| |
| #include <android-base/properties.h> |
| #include <base/logging.h> |
| #include <base/strings/string_util.h> |
| #include <bootloader_message/bootloader_message.h> |
| #include <brillo/osrelease_reader.h> |
| #include <brillo/strings/string_utils.h> |
| |
| #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 |