/* * Copyright (C) 2014 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 "verity_tool.h" #include #include #include #include #include #include #include static int make_block_device_writable(const std::string& block_device) { int fd = open(block_device.c_str(), O_RDONLY | O_CLOEXEC); if (fd < 0) { return -errno; } int OFF = 0; int rc = ioctl(fd, BLKROSET, &OFF); if (rc < 0) { rc = -errno; goto out; } rc = 0; out: close(fd); return rc; } /* Turn verity on/off */ bool set_block_device_verity_enabled(const std::string& block_device, bool enable) { int rc = make_block_device_writable(block_device); if (rc) { LOG(ERROR) << "Could not make block device " << block_device << " writable:" << rc; return false; } fec::io fh(block_device, O_RDWR); if (!fh) { PLOG(ERROR) << "Could not open block device " << block_device; return false; } fec_verity_metadata metadata; if (!fh.get_verity_metadata(metadata)) { LOG(ERROR) << "Couldn't find verity metadata!"; return false; } if (!enable && metadata.disabled) { LOG(ERROR) << "Verity already disabled on " << block_device; return false; } if (enable && !metadata.disabled) { LOG(WARNING) << "Verity already enabled on " << block_device; return false; } if (!fh.set_verity_status(enable)) { PLOG(ERROR) << "Could not set verity " << (enable ? "enabled" : "disabled") << " flag on device " << block_device; return false; } LOG(DEBUG) << "Verity " << (enable ? "enabled" : "disabled") << " on " << block_device; return true; } /* Helper function to get A/B suffix, if any. If the device isn't * using A/B the empty string is returned. Otherwise either "_a", * "_b", ... is returned. * * Note that since sometime in O androidboot.slot_suffix is deprecated * and androidboot.slot should be used instead. Since bootloaders may * be out of sync with the OS, we check both and for extra safety * prepend a leading underscore if there isn't one already. */ static std::string get_ab_suffix() { std::string ab_suffix = android::base::GetProperty("ro.boot.slot_suffix", ""); if (ab_suffix.empty()) { ab_suffix = android::base::GetProperty("ro.boot.slot", ""); } if (ab_suffix.size() > 0 && ab_suffix[0] != '_') { ab_suffix = std::string("_") + ab_suffix; } return ab_suffix; } verity_state_t get_verity_state() { verity_state_t rc = VERITY_STATE_NO_DEVICE; std::string ab_suffix = get_ab_suffix(); // Figure out if we're using VB1.0 or VB2.0 (aka AVB) - by // contract, androidboot.vbmeta.digest is set by the bootloader // when using AVB). bool using_avb = !android::base::GetProperty("ro.boot.vbmeta.digest", "").empty(); if (using_avb) { // Yep, the system is using AVB. AvbOps* ops = avb_ops_user_new(); if (ops == nullptr) { LOG(ERROR) << "Error getting AVB ops"; avb_ops_user_free(ops); return VERITY_STATE_UNKNOWN; } bool verity_enabled; if (!avb_user_verity_get(ops, ab_suffix.c_str(), &verity_enabled)) { LOG(ERROR) << "Error getting verity state"; avb_ops_user_free(ops); return VERITY_STATE_UNKNOWN; } rc = verity_enabled ? VERITY_STATE_ENABLED : VERITY_STATE_DISABLED; avb_ops_user_free(ops); } else { // Not using AVB - assume VB1.0. // read all fstab entries at once from all sources struct fstab* fstab = fs_mgr_read_fstab_default(); if (!fstab) { LOG(ERROR) << "Failed to read fstab"; fs_mgr_free_fstab(fstab); return VERITY_STATE_UNKNOWN; } // Loop through entries looking for ones that vold manages. for (int i = 0; i < fstab->num_entries; i++) { if (fs_mgr_is_verified(&fstab->recs[i])) { std::string block_device = fstab->recs[i].blk_device; fec::io fh(block_device, O_RDONLY); if (!fh) { PLOG(ERROR) << "Could not open block device " << block_device; rc = VERITY_STATE_UNKNOWN; break; } fec_verity_metadata metadata; if (!fh.get_verity_metadata(metadata)) { LOG(ERROR) << "Couldn't find verity metadata!"; rc = VERITY_STATE_UNKNOWN; break; } rc = metadata.disabled ? VERITY_STATE_DISABLED : VERITY_STATE_ENABLED; } } fs_mgr_free_fstab(fstab); } return rc; } /* Use AVB to turn verity on/off */ static bool set_avb_verity_enabled_state(AvbOps* ops, bool enable_verity) { std::string ab_suffix = get_ab_suffix(); bool verity_enabled; if (!avb_user_verity_get(ops, ab_suffix.c_str(), &verity_enabled)) { LOG(ERROR) << "Error getting verity state"; return false; } if ((verity_enabled && enable_verity) || (!verity_enabled && !enable_verity)) { LOG(WARNING) << "verity is already " << verity_enabled ? "enabled" : "disabled"; return false; } if (!avb_user_verity_set(ops, ab_suffix.c_str(), enable_verity)) { LOG(ERROR) << "Error setting verity"; return false; } LOG(DEBUG) << "Successfully " << (enable_verity ? "enabled" : "disabled") << " verity"; return true; } bool set_verity_enabled(bool enable) { bool rc = true; // Do not allow changing verity on user builds bool is_user = (android::base::GetProperty("ro.build.type", "") == "user"); if (is_user) { LOG(ERROR) << "Cannot disable verity - USER BUILD"; return false; } // Figure out if we're using VB1.0 or VB2.0 (aka AVB) - by // contract, androidboot.vbmeta.digest is set by the bootloader // when using AVB). bool using_avb = !android::base::GetProperty("ro.boot.vbmeta.digest", "").empty(); // If using AVB, dm-verity is used on any build so we want it to // be possible to disable/enable on any build (except USER). For // VB1.0 dm-verity is only enabled on certain builds. if (using_avb) { // Yep, the system is using AVB. AvbOps* ops = avb_ops_user_new(); if (ops == nullptr) { LOG(ERROR) << "Error getting AVB ops"; return false; } rc = set_avb_verity_enabled_state(ops, enable); avb_ops_user_free(ops); } else { // Not using AVB - assume VB1.0. // read all fstab entries at once from all sources struct fstab* fstab = fs_mgr_read_fstab_default(); if (!fstab) { LOG(ERROR) << "Failed to read fstab"; return false; } // Loop through entries looking for ones that vold manages. for (int i = 0; i < fstab->num_entries; i++) { if (fs_mgr_is_verified(&fstab->recs[i])) { bool result = set_block_device_verity_enabled( fstab->recs[i].blk_device, enable); if (!result) { // Warn, but continue if failure occurred LOG(WARNING) << "Failed to set state " << (enable ? "enabled" : "disabled") << " on " << fstab->recs[i].mount_point; } rc = rc && result; } } } return rc; }