2018-06-07 22:36:09 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 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 "liblp/builder.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
#include <android-base/unique_fd.h>
|
2018-06-07 22:36:09 +00:00
|
|
|
|
2018-07-12 20:05:46 +00:00
|
|
|
#include "liblp/liblp.h"
|
|
|
|
#include "reader.h"
|
2018-06-07 22:36:09 +00:00
|
|
|
#include "utility.h"
|
|
|
|
|
|
|
|
namespace android {
|
|
|
|
namespace fs_mgr {
|
|
|
|
|
|
|
|
void LinearExtent::AddTo(LpMetadata* out) const {
|
|
|
|
out->extents.push_back(LpMetadataExtent{num_sectors_, LP_TARGET_TYPE_LINEAR, physical_sector_});
|
|
|
|
}
|
|
|
|
|
|
|
|
void ZeroExtent::AddTo(LpMetadata* out) const {
|
|
|
|
out->extents.push_back(LpMetadataExtent{num_sectors_, LP_TARGET_TYPE_ZERO, 0});
|
|
|
|
}
|
|
|
|
|
2018-10-03 20:49:23 +00:00
|
|
|
Partition::Partition(const std::string& name, const std::string& group_name, uint32_t attributes)
|
|
|
|
: name_(name), group_name_(group_name), attributes_(attributes), size_(0) {}
|
2018-06-07 22:36:09 +00:00
|
|
|
|
|
|
|
void Partition::AddExtent(std::unique_ptr<Extent>&& extent) {
|
|
|
|
size_ += extent->num_sectors() * LP_SECTOR_SIZE;
|
2018-08-03 17:56:53 +00:00
|
|
|
|
|
|
|
if (LinearExtent* new_extent = extent->AsLinearExtent()) {
|
|
|
|
if (!extents_.empty() && extents_.back()->AsLinearExtent() &&
|
|
|
|
extents_.back()->AsLinearExtent()->end_sector() == new_extent->physical_sector()) {
|
|
|
|
// If the previous extent can be merged into this new one, do so
|
|
|
|
// to avoid creating unnecessary extents.
|
|
|
|
LinearExtent* prev_extent = extents_.back()->AsLinearExtent();
|
|
|
|
extent = std::make_unique<LinearExtent>(
|
|
|
|
prev_extent->num_sectors() + new_extent->num_sectors(),
|
|
|
|
prev_extent->physical_sector());
|
|
|
|
extents_.pop_back();
|
|
|
|
}
|
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
extents_.push_back(std::move(extent));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Partition::RemoveExtents() {
|
|
|
|
size_ = 0;
|
|
|
|
extents_.clear();
|
|
|
|
}
|
|
|
|
|
2018-07-31 20:34:14 +00:00
|
|
|
void Partition::ShrinkTo(uint64_t aligned_size) {
|
2018-06-07 22:36:09 +00:00
|
|
|
if (aligned_size == 0) {
|
|
|
|
RemoveExtents();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove or shrink extents of any kind until the total partition size is
|
|
|
|
// equal to the requested size.
|
|
|
|
uint64_t sectors_to_remove = (size_ - aligned_size) / LP_SECTOR_SIZE;
|
|
|
|
while (sectors_to_remove) {
|
|
|
|
Extent* extent = extents_.back().get();
|
|
|
|
if (extent->num_sectors() > sectors_to_remove) {
|
|
|
|
size_ -= sectors_to_remove * LP_SECTOR_SIZE;
|
|
|
|
extent->set_num_sectors(extent->num_sectors() - sectors_to_remove);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
size_ -= (extent->num_sectors() * LP_SECTOR_SIZE);
|
|
|
|
sectors_to_remove -= extent->num_sectors();
|
|
|
|
extents_.pop_back();
|
|
|
|
}
|
2018-07-31 20:34:14 +00:00
|
|
|
DCHECK(size_ == aligned_size);
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
|
2018-10-03 01:22:31 +00:00
|
|
|
uint64_t Partition::BytesOnDisk() const {
|
|
|
|
uint64_t sectors = 0;
|
|
|
|
for (const auto& extent : extents_) {
|
|
|
|
if (!extent->AsLinearExtent()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
sectors += extent->num_sectors();
|
|
|
|
}
|
|
|
|
return sectors * LP_SECTOR_SIZE;
|
|
|
|
}
|
|
|
|
|
2018-10-23 01:05:54 +00:00
|
|
|
std::unique_ptr<MetadataBuilder> MetadataBuilder::New(const IPartitionOpener& opener,
|
|
|
|
const std::string& super_partition,
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
uint32_t slot_number) {
|
2018-10-23 01:05:54 +00:00
|
|
|
std::unique_ptr<LpMetadata> metadata = ReadMetadata(opener, super_partition, slot_number);
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
if (!metadata) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
std::unique_ptr<MetadataBuilder> builder = New(*metadata.get());
|
|
|
|
if (!builder) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
BlockDeviceInfo device_info;
|
2018-10-23 01:05:54 +00:00
|
|
|
if (opener.GetInfo(super_partition, &device_info)) {
|
2018-10-11 01:25:07 +00:00
|
|
|
builder->UpdateBlockDeviceInfo(device_info);
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
}
|
|
|
|
return builder;
|
|
|
|
}
|
|
|
|
|
2018-10-23 01:05:54 +00:00
|
|
|
std::unique_ptr<MetadataBuilder> MetadataBuilder::New(const std::string& super_partition,
|
|
|
|
uint32_t slot_number) {
|
|
|
|
return New(PartitionOpener(), super_partition, slot_number);
|
|
|
|
}
|
|
|
|
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
std::unique_ptr<MetadataBuilder> MetadataBuilder::New(const BlockDeviceInfo& device_info,
|
2018-06-07 22:36:09 +00:00
|
|
|
uint32_t metadata_max_size,
|
|
|
|
uint32_t metadata_slot_count) {
|
|
|
|
std::unique_ptr<MetadataBuilder> builder(new MetadataBuilder());
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
if (!builder->Init(device_info, metadata_max_size, metadata_slot_count)) {
|
2018-06-07 22:36:09 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return builder;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<MetadataBuilder> MetadataBuilder::New(const LpMetadata& metadata) {
|
|
|
|
std::unique_ptr<MetadataBuilder> builder(new MetadataBuilder());
|
|
|
|
if (!builder->Init(metadata)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return builder;
|
|
|
|
}
|
|
|
|
|
|
|
|
MetadataBuilder::MetadataBuilder() {
|
|
|
|
memset(&geometry_, 0, sizeof(geometry_));
|
|
|
|
geometry_.magic = LP_METADATA_GEOMETRY_MAGIC;
|
|
|
|
geometry_.struct_size = sizeof(geometry_);
|
|
|
|
|
|
|
|
memset(&header_, 0, sizeof(header_));
|
|
|
|
header_.magic = LP_METADATA_HEADER_MAGIC;
|
|
|
|
header_.major_version = LP_METADATA_MAJOR_VERSION;
|
|
|
|
header_.minor_version = LP_METADATA_MINOR_VERSION;
|
|
|
|
header_.header_size = sizeof(header_);
|
|
|
|
header_.partitions.entry_size = sizeof(LpMetadataPartition);
|
|
|
|
header_.extents.entry_size = sizeof(LpMetadataExtent);
|
2018-10-03 01:22:31 +00:00
|
|
|
header_.groups.entry_size = sizeof(LpMetadataPartitionGroup);
|
2018-10-12 20:51:52 +00:00
|
|
|
header_.block_devices.entry_size = sizeof(LpMetadataBlockDevice);
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool MetadataBuilder::Init(const LpMetadata& metadata) {
|
|
|
|
geometry_ = metadata.geometry;
|
|
|
|
|
2018-10-03 01:22:31 +00:00
|
|
|
for (const auto& group : metadata.groups) {
|
|
|
|
std::string group_name = GetPartitionGroupName(group);
|
|
|
|
if (!AddGroup(group_name, group.maximum_size)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-12 20:51:52 +00:00
|
|
|
for (const auto& block_device : metadata.block_devices) {
|
|
|
|
block_devices_.push_back(block_device);
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
for (const auto& partition : metadata.partitions) {
|
2018-10-03 01:22:31 +00:00
|
|
|
std::string group_name = GetPartitionGroupName(metadata.groups[partition.group_index]);
|
2018-10-03 20:49:23 +00:00
|
|
|
Partition* builder =
|
|
|
|
AddPartition(GetPartitionName(partition), group_name, partition.attributes);
|
2018-06-07 22:36:09 +00:00
|
|
|
if (!builder) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < partition.num_extents; i++) {
|
|
|
|
const LpMetadataExtent& extent = metadata.extents[partition.first_extent_index + i];
|
|
|
|
if (extent.target_type == LP_TARGET_TYPE_LINEAR) {
|
|
|
|
auto copy = std::make_unique<LinearExtent>(extent.num_sectors, extent.target_data);
|
|
|
|
builder->AddExtent(std::move(copy));
|
|
|
|
} else if (extent.target_type == LP_TARGET_TYPE_ZERO) {
|
|
|
|
auto copy = std::make_unique<ZeroExtent>(extent.num_sectors);
|
|
|
|
builder->AddExtent(std::move(copy));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
bool MetadataBuilder::Init(const BlockDeviceInfo& device_info, uint32_t metadata_max_size,
|
2018-06-07 22:36:09 +00:00
|
|
|
uint32_t metadata_slot_count) {
|
|
|
|
if (metadata_max_size < sizeof(LpMetadataHeader)) {
|
|
|
|
LERROR << "Invalid metadata maximum size.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (metadata_slot_count == 0) {
|
|
|
|
LERROR << "Invalid metadata slot count.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Align the metadata size up to the nearest sector.
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
metadata_max_size = AlignTo(metadata_max_size, LP_SECTOR_SIZE);
|
|
|
|
|
|
|
|
// Check that device properties are sane.
|
2018-10-11 01:25:07 +00:00
|
|
|
if (device_info.size % LP_SECTOR_SIZE != 0) {
|
2018-07-14 00:08:08 +00:00
|
|
|
LERROR << "Block device size must be a multiple of 512.";
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-11 01:25:07 +00:00
|
|
|
if (device_info.logical_block_size % LP_SECTOR_SIZE != 0) {
|
2018-08-01 21:14:37 +00:00
|
|
|
LERROR << "Logical block size must be a multiple of 512.";
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-11 01:25:07 +00:00
|
|
|
if (device_info.alignment_offset % LP_SECTOR_SIZE != 0) {
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
LERROR << "Alignment offset is not sector-aligned.";
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-11 01:25:07 +00:00
|
|
|
if (device_info.alignment % LP_SECTOR_SIZE != 0) {
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
LERROR << "Partition alignment is not sector-aligned.";
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-11 01:25:07 +00:00
|
|
|
if (device_info.alignment_offset > device_info.alignment) {
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
LERROR << "Partition alignment offset is greater than its alignment.";
|
|
|
|
return false;
|
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
|
|
|
|
// We reserve a geometry block (4KB) plus space for each copy of the
|
|
|
|
// maximum size of a metadata blob. Then, we double that space since
|
|
|
|
// we store a backup copy of everything.
|
2018-10-12 20:51:52 +00:00
|
|
|
uint64_t total_reserved = GetTotalMetadataSize(metadata_max_size, metadata_slot_count);
|
2018-10-11 01:25:07 +00:00
|
|
|
if (device_info.size < total_reserved) {
|
2018-06-07 22:36:09 +00:00
|
|
|
LERROR << "Attempting to create metadata on a block device that is too small.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
// Compute the first free sector, factoring in alignment.
|
2018-10-12 16:36:06 +00:00
|
|
|
uint64_t free_area_start = total_reserved;
|
|
|
|
if (device_info.alignment || device_info.alignment_offset) {
|
|
|
|
free_area_start =
|
|
|
|
AlignTo(free_area_start, device_info.alignment, device_info.alignment_offset);
|
|
|
|
} else {
|
|
|
|
free_area_start = AlignTo(free_area_start, device_info.logical_block_size);
|
|
|
|
}
|
2018-10-11 01:49:36 +00:00
|
|
|
uint64_t first_sector = free_area_start / LP_SECTOR_SIZE;
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
|
2018-10-11 01:49:36 +00:00
|
|
|
// There must be one logical block of free space remaining (enough for one partition).
|
|
|
|
uint64_t minimum_disk_size = (first_sector * LP_SECTOR_SIZE) + device_info.logical_block_size;
|
|
|
|
if (device_info.size < minimum_disk_size) {
|
|
|
|
LERROR << "Device must be at least " << minimum_disk_size << " bytes, only has "
|
|
|
|
<< device_info.size;
|
2018-08-01 21:14:37 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-12 20:51:52 +00:00
|
|
|
block_devices_.push_back(LpMetadataBlockDevice{
|
|
|
|
first_sector,
|
|
|
|
device_info.alignment,
|
|
|
|
device_info.alignment_offset,
|
|
|
|
device_info.size,
|
|
|
|
"super",
|
|
|
|
});
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
geometry_.metadata_max_size = metadata_max_size;
|
|
|
|
geometry_.metadata_slot_count = metadata_slot_count;
|
2018-08-01 21:14:37 +00:00
|
|
|
geometry_.logical_block_size = device_info.logical_block_size;
|
2018-10-03 01:22:31 +00:00
|
|
|
|
|
|
|
if (!AddGroup("default", 0)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MetadataBuilder::AddGroup(const std::string& group_name, uint64_t maximum_size) {
|
|
|
|
if (FindGroup(group_name)) {
|
|
|
|
LERROR << "Group already exists: " << group_name;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
groups_.push_back(std::make_unique<PartitionGroup>(group_name, maximum_size));
|
2018-06-07 22:36:09 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-10-03 20:49:23 +00:00
|
|
|
Partition* MetadataBuilder::AddPartition(const std::string& name, uint32_t attributes) {
|
|
|
|
return AddPartition(name, "default", attributes);
|
2018-10-03 01:22:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Partition* MetadataBuilder::AddPartition(const std::string& name, const std::string& group_name,
|
2018-10-03 20:49:23 +00:00
|
|
|
uint32_t attributes) {
|
2018-06-07 22:36:09 +00:00
|
|
|
if (name.empty()) {
|
|
|
|
LERROR << "Partition must have a non-empty name.";
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (FindPartition(name)) {
|
|
|
|
LERROR << "Attempting to create duplication partition with name: " << name;
|
|
|
|
return nullptr;
|
|
|
|
}
|
2018-10-03 01:22:31 +00:00
|
|
|
if (!FindGroup(group_name)) {
|
|
|
|
LERROR << "Could not find partition group: " << group_name;
|
|
|
|
return nullptr;
|
|
|
|
}
|
2018-10-03 20:49:23 +00:00
|
|
|
partitions_.push_back(std::make_unique<Partition>(name, group_name, attributes));
|
2018-06-07 22:36:09 +00:00
|
|
|
return partitions_.back().get();
|
|
|
|
}
|
|
|
|
|
|
|
|
Partition* MetadataBuilder::FindPartition(const std::string& name) {
|
|
|
|
for (const auto& partition : partitions_) {
|
|
|
|
if (partition->name() == name) {
|
|
|
|
return partition.get();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-10-18 22:29:01 +00:00
|
|
|
PartitionGroup* MetadataBuilder::FindGroup(const std::string& group_name) {
|
2018-10-03 01:22:31 +00:00
|
|
|
for (const auto& group : groups_) {
|
|
|
|
if (group->name() == group_name) {
|
|
|
|
return group.get();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t MetadataBuilder::TotalSizeOfGroup(PartitionGroup* group) const {
|
|
|
|
uint64_t total = 0;
|
|
|
|
for (const auto& partition : partitions_) {
|
|
|
|
if (partition->group_name() != group->name()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
total += partition->BytesOnDisk();
|
|
|
|
}
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
void MetadataBuilder::RemovePartition(const std::string& name) {
|
|
|
|
for (auto iter = partitions_.begin(); iter != partitions_.end(); iter++) {
|
|
|
|
if ((*iter)->name() == name) {
|
|
|
|
partitions_.erase(iter);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
void MetadataBuilder::ExtentsToFreeList(const std::vector<Interval>& extents,
|
|
|
|
std::vector<Interval>* free_regions) const {
|
|
|
|
// Convert the extent list into a list of gaps between the extents; i.e.,
|
|
|
|
// the list of ranges that are free on the disk.
|
|
|
|
for (size_t i = 1; i < extents.size(); i++) {
|
|
|
|
const Interval& previous = extents[i - 1];
|
|
|
|
const Interval& current = extents[i];
|
2018-10-03 01:22:31 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
uint64_t aligned = AlignSector(previous.end);
|
|
|
|
if (aligned >= current.start) {
|
|
|
|
// There is no gap between these two extents, try the next one.
|
|
|
|
// Note that we check with >= instead of >, since alignment may
|
|
|
|
// bump the ending sector past the beginning of the next extent.
|
|
|
|
continue;
|
2018-10-03 01:22:31 +00:00
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
// The new interval represents the free space starting at the end of
|
|
|
|
// the previous interval, and ending at the start of the next interval.
|
|
|
|
free_regions->emplace_back(aligned, current.start);
|
|
|
|
}
|
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
auto MetadataBuilder::GetFreeRegions() const -> std::vector<Interval> {
|
|
|
|
std::vector<Interval> free_regions;
|
2018-06-07 22:36:09 +00:00
|
|
|
|
2018-07-31 23:43:26 +00:00
|
|
|
// Collect all extents in the partition table, then sort them by starting
|
|
|
|
// sector.
|
|
|
|
std::vector<Interval> extents;
|
2018-06-07 22:36:09 +00:00
|
|
|
for (const auto& partition : partitions_) {
|
|
|
|
for (const auto& extent : partition->extents()) {
|
|
|
|
LinearExtent* linear = extent->AsLinearExtent();
|
|
|
|
if (!linear) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-07-31 23:43:26 +00:00
|
|
|
extents.emplace_back(linear->physical_sector(),
|
|
|
|
linear->physical_sector() + extent->num_sectors());
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
}
|
2018-10-20 00:15:31 +00:00
|
|
|
|
|
|
|
// Add 0-length intervals for the first and last sectors. This will cause
|
2018-10-12 20:51:52 +00:00
|
|
|
// ExtentToFreeList() to treat the space in between as available.
|
|
|
|
uint64_t first_sector = super_device().first_logical_sector;
|
|
|
|
uint64_t last_sector = super_device().size / LP_SECTOR_SIZE;
|
|
|
|
extents.emplace_back(first_sector, first_sector);
|
2018-10-20 00:15:31 +00:00
|
|
|
extents.emplace_back(last_sector, last_sector);
|
|
|
|
|
2018-07-31 23:43:26 +00:00
|
|
|
std::sort(extents.begin(), extents.end());
|
2018-06-07 22:36:09 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
ExtentsToFreeList(extents, &free_regions);
|
|
|
|
return free_regions;
|
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
bool MetadataBuilder::GrowPartition(Partition* partition, uint64_t aligned_size) {
|
|
|
|
PartitionGroup* group = FindGroup(partition->group_name());
|
|
|
|
CHECK(group);
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
// Figure out how much we need to allocate, and whether our group has
|
|
|
|
// enough space remaining.
|
|
|
|
uint64_t space_needed = aligned_size - partition->size();
|
|
|
|
if (group->maximum_size() > 0) {
|
|
|
|
uint64_t group_size = TotalSizeOfGroup(group);
|
|
|
|
if (group_size >= group->maximum_size() ||
|
|
|
|
group->maximum_size() - group_size < space_needed) {
|
|
|
|
LERROR << "Partition " << partition->name() << " is part of group " << group->name()
|
|
|
|
<< " which does not have enough space free (" << space_needed << "requested, "
|
|
|
|
<< group_size << " used out of " << group->maximum_size();
|
|
|
|
return false;
|
|
|
|
}
|
2018-07-31 23:43:26 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
uint64_t sectors_needed = space_needed / LP_SECTOR_SIZE;
|
|
|
|
DCHECK(sectors_needed * LP_SECTOR_SIZE == space_needed);
|
2018-10-11 01:49:36 +00:00
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
std::vector<Interval> free_regions = GetFreeRegions();
|
2018-07-31 23:43:26 +00:00
|
|
|
|
2018-10-11 01:25:07 +00:00
|
|
|
const uint64_t sectors_per_block = geometry_.logical_block_size / LP_SECTOR_SIZE;
|
2018-09-04 19:41:21 +00:00
|
|
|
CHECK_NE(sectors_per_block, 0);
|
2018-08-01 21:14:37 +00:00
|
|
|
CHECK(sectors_needed % sectors_per_block == 0);
|
|
|
|
|
2018-07-31 23:43:26 +00:00
|
|
|
// Find gaps that we can use for new extents. Note we store new extents in a
|
|
|
|
// temporary vector, and only commit them if we are guaranteed enough free
|
|
|
|
// space.
|
|
|
|
std::vector<std::unique_ptr<LinearExtent>> new_extents;
|
2018-08-01 21:14:37 +00:00
|
|
|
for (auto& region : free_regions) {
|
|
|
|
if (region.length() % sectors_per_block != 0) {
|
|
|
|
// This should never happen, because it would imply that we
|
|
|
|
// once allocated an extent that was not a multiple of the
|
|
|
|
// block size. That extent would be rejected by DM_TABLE_LOAD.
|
|
|
|
LERROR << "Region " << region.start << ".." << region.end
|
|
|
|
<< " is not a multiple of the block size, " << sectors_per_block;
|
|
|
|
|
|
|
|
// If for some reason the final region is mis-sized we still want
|
|
|
|
// to be able to grow partitions. So just to be safe, round the
|
|
|
|
// region down to the nearest block.
|
|
|
|
region.end = region.start + (region.length() / sectors_per_block) * sectors_per_block;
|
|
|
|
if (!region.length()) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
|
2018-08-01 21:14:37 +00:00
|
|
|
uint64_t sectors = std::min(sectors_needed, region.length());
|
|
|
|
CHECK(sectors % sectors_per_block == 0);
|
|
|
|
|
|
|
|
auto extent = std::make_unique<LinearExtent>(sectors, region.start);
|
2018-06-07 22:36:09 +00:00
|
|
|
new_extents.push_back(std::move(extent));
|
2018-08-01 21:14:37 +00:00
|
|
|
sectors_needed -= sectors;
|
|
|
|
if (!sectors_needed) {
|
|
|
|
break;
|
|
|
|
}
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
if (sectors_needed) {
|
2018-07-31 23:43:26 +00:00
|
|
|
LERROR << "Not enough free space to expand partition: " << partition->name();
|
|
|
|
return false;
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
|
2018-08-01 21:14:37 +00:00
|
|
|
// Everything succeeded, so commit the new extents.
|
2018-06-07 22:36:09 +00:00
|
|
|
for (auto& extent : new_extents) {
|
|
|
|
partition->AddExtent(std::move(extent));
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-07-31 20:34:14 +00:00
|
|
|
void MetadataBuilder::ShrinkPartition(Partition* partition, uint64_t aligned_size) {
|
|
|
|
partition->ShrinkTo(aligned_size);
|
2018-06-07 22:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<LpMetadata> MetadataBuilder::Export() {
|
|
|
|
std::unique_ptr<LpMetadata> metadata = std::make_unique<LpMetadata>();
|
|
|
|
metadata->header = header_;
|
|
|
|
metadata->geometry = geometry_;
|
|
|
|
|
2018-10-03 01:22:31 +00:00
|
|
|
std::map<std::string, size_t> group_indices;
|
|
|
|
for (const auto& group : groups_) {
|
|
|
|
LpMetadataPartitionGroup out = {};
|
|
|
|
|
|
|
|
if (group->name().size() > sizeof(out.name)) {
|
|
|
|
LERROR << "Partition group name is too long: " << group->name();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
strncpy(out.name, group->name().c_str(), sizeof(out.name));
|
|
|
|
out.maximum_size = group->maximum_size();
|
|
|
|
|
2018-10-16 22:27:56 +00:00
|
|
|
group_indices[group->name()] = metadata->groups.size();
|
2018-10-03 01:22:31 +00:00
|
|
|
metadata->groups.push_back(out);
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
// Flatten the partition and extent structures into an LpMetadata, which
|
|
|
|
// makes it very easy to validate, serialize, or pass on to device-mapper.
|
|
|
|
for (const auto& partition : partitions_) {
|
|
|
|
LpMetadataPartition part;
|
|
|
|
memset(&part, 0, sizeof(part));
|
|
|
|
|
|
|
|
if (partition->name().size() > sizeof(part.name)) {
|
|
|
|
LERROR << "Partition name is too long: " << partition->name();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (partition->attributes() & ~(LP_PARTITION_ATTRIBUTE_MASK)) {
|
|
|
|
LERROR << "Partition " << partition->name() << " has unsupported attribute.";
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(part.name, partition->name().c_str(), sizeof(part.name));
|
|
|
|
part.first_extent_index = static_cast<uint32_t>(metadata->extents.size());
|
|
|
|
part.num_extents = static_cast<uint32_t>(partition->extents().size());
|
|
|
|
part.attributes = partition->attributes();
|
|
|
|
|
2018-10-16 22:27:56 +00:00
|
|
|
auto iter = group_indices.find(partition->group_name());
|
|
|
|
if (iter == group_indices.end()) {
|
|
|
|
LERROR << "Partition " << partition->name() << " is a member of unknown group "
|
|
|
|
<< partition->group_name();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
part.group_index = iter->second;
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
for (const auto& extent : partition->extents()) {
|
|
|
|
extent->AddTo(metadata.get());
|
|
|
|
}
|
|
|
|
metadata->partitions.push_back(part);
|
|
|
|
}
|
|
|
|
|
2018-10-12 20:51:52 +00:00
|
|
|
metadata->block_devices = block_devices_;
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
metadata->header.partitions.num_entries = static_cast<uint32_t>(metadata->partitions.size());
|
|
|
|
metadata->header.extents.num_entries = static_cast<uint32_t>(metadata->extents.size());
|
2018-10-03 01:22:31 +00:00
|
|
|
metadata->header.groups.num_entries = static_cast<uint32_t>(metadata->groups.size());
|
2018-10-12 20:51:52 +00:00
|
|
|
metadata->header.block_devices.num_entries =
|
|
|
|
static_cast<uint32_t>(metadata->block_devices.size());
|
2018-06-07 22:36:09 +00:00
|
|
|
return metadata;
|
|
|
|
}
|
|
|
|
|
2018-06-15 18:13:29 +00:00
|
|
|
uint64_t MetadataBuilder::AllocatableSpace() const {
|
2018-10-12 20:51:52 +00:00
|
|
|
return super_device().size - (super_device().first_logical_sector * LP_SECTOR_SIZE);
|
2018-06-15 18:13:29 +00:00
|
|
|
}
|
|
|
|
|
2018-09-24 15:07:35 +00:00
|
|
|
uint64_t MetadataBuilder::UsedSpace() const {
|
|
|
|
uint64_t size = 0;
|
|
|
|
for (const auto& partition : partitions_) {
|
|
|
|
size += partition->size();
|
|
|
|
}
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2018-10-20 00:15:31 +00:00
|
|
|
uint64_t MetadataBuilder::AlignSector(uint64_t sector) const {
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
// Note: when reading alignment info from the Kernel, we don't assume it
|
|
|
|
// is aligned to the sector size, so we round up to the nearest sector.
|
|
|
|
uint64_t lba = sector * LP_SECTOR_SIZE;
|
2018-10-12 20:51:52 +00:00
|
|
|
uint64_t aligned = AlignTo(lba, super_device().alignment, super_device().alignment_offset);
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
return AlignTo(aligned, LP_SECTOR_SIZE) / LP_SECTOR_SIZE;
|
|
|
|
}
|
|
|
|
|
2018-10-11 01:25:07 +00:00
|
|
|
bool MetadataBuilder::GetBlockDeviceInfo(BlockDeviceInfo* info) const {
|
2018-10-12 20:51:52 +00:00
|
|
|
info->size = super_device().size;
|
|
|
|
info->alignment = super_device().alignment;
|
|
|
|
info->alignment_offset = super_device().alignment_offset;
|
2018-10-11 01:25:07 +00:00
|
|
|
info->logical_block_size = geometry_.logical_block_size;
|
|
|
|
return true;
|
|
|
|
}
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
|
2018-10-11 01:25:07 +00:00
|
|
|
bool MetadataBuilder::UpdateBlockDeviceInfo(const BlockDeviceInfo& device_info) {
|
2018-10-12 20:51:52 +00:00
|
|
|
if (device_info.size != super_device().size) {
|
2018-10-11 01:25:07 +00:00
|
|
|
LERROR << "Device size does not match (got " << device_info.size << ", expected "
|
2018-10-12 20:51:52 +00:00
|
|
|
<< super_device().size << ")";
|
2018-10-11 01:25:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (device_info.logical_block_size != geometry_.logical_block_size) {
|
|
|
|
LERROR << "Device logical block size does not match (got " << device_info.logical_block_size
|
|
|
|
<< ", expected " << geometry_.logical_block_size << ")";
|
|
|
|
return false;
|
|
|
|
}
|
2018-08-01 21:14:37 +00:00
|
|
|
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
// The kernel does not guarantee these values are present, so we only
|
|
|
|
// replace existing values if the new values are non-zero.
|
|
|
|
if (device_info.alignment) {
|
2018-10-12 20:51:52 +00:00
|
|
|
super_device().alignment = device_info.alignment;
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
}
|
|
|
|
if (device_info.alignment_offset) {
|
2018-10-12 20:51:52 +00:00
|
|
|
super_device().alignment_offset = device_info.alignment_offset;
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
}
|
2018-10-11 01:25:07 +00:00
|
|
|
return true;
|
liblp: Implement support for request queue alignment.
Block devices in the Linux kernel have a "minimum I/O request" size. The
minimum size is usually acquired by the block driver and can change
from device to device. When stacking devices (such as with
device-mapper), the kernel goes through great lengths to make sure this
alignment is respected for optimal I/O. In device-mapper's case,
misalignment can lead to kernel warnings and performance issues.
While this is unlikely to matter with a few targets, it could become
problematic on a large number of targets, and so we would prefer to
align all partition extents to the minimum I/O size.
We now support two new properties in the partition table geometry: an
"alignment", which is the minimum I/O size, and an "alignment offset",
which is an offset that when applied to sector 0, causes the sector to
be properly aligned within its parent device (for example, if a
physical partition is misaligned). All partition extents now begin on a
sector that respects this alignment.
One major caveat is that it is difficult for the initial partition table
to have the correct alignment without build system and/or flash tool
support. To accomodate this, all alignment is optional, and the lpmake
tool will support a default alignment of 1MiB as a failsafe.
Bug: 79173901
Test: liblp_test gtest
Change-Id: I5bc41b90aa085f4f30393951af0d2b37c4ac2a72
2018-07-09 19:12:52 +00:00
|
|
|
}
|
|
|
|
|
2018-07-31 20:34:14 +00:00
|
|
|
bool MetadataBuilder::ResizePartition(Partition* partition, uint64_t requested_size) {
|
|
|
|
// Align the space needed up to the nearest sector.
|
2018-10-11 01:25:07 +00:00
|
|
|
uint64_t aligned_size = AlignTo(requested_size, geometry_.logical_block_size);
|
2018-08-03 17:12:16 +00:00
|
|
|
uint64_t old_size = partition->size();
|
2018-07-31 20:34:14 +00:00
|
|
|
|
2018-08-03 17:12:16 +00:00
|
|
|
if (aligned_size > old_size) {
|
|
|
|
if (!GrowPartition(partition, aligned_size)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (aligned_size < partition->size()) {
|
2018-07-31 20:34:14 +00:00
|
|
|
ShrinkPartition(partition, aligned_size);
|
|
|
|
}
|
2018-08-03 17:12:16 +00:00
|
|
|
|
2018-10-03 01:22:31 +00:00
|
|
|
if (partition->size() != old_size) {
|
|
|
|
LINFO << "Partition " << partition->name() << " will resize from " << old_size
|
|
|
|
<< " bytes to " << aligned_size << " bytes";
|
|
|
|
}
|
2018-07-31 20:34:14 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-10-18 22:29:01 +00:00
|
|
|
std::vector<std::string> MetadataBuilder::ListGroups() const {
|
|
|
|
std::vector<std::string> names;
|
|
|
|
for (const auto& group : groups_) {
|
|
|
|
names.emplace_back(group->name());
|
|
|
|
}
|
|
|
|
return names;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MetadataBuilder::RemoveGroupAndPartitions(const std::string& group_name) {
|
|
|
|
if (group_name == "default") {
|
|
|
|
// Cannot remove the default group.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
std::vector<std::string> partition_names;
|
|
|
|
for (const auto& partition : partitions_) {
|
|
|
|
if (partition->group_name() == group_name) {
|
|
|
|
partition_names.emplace_back(partition->name());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& partition_name : partition_names) {
|
|
|
|
RemovePartition(partition_name);
|
|
|
|
}
|
|
|
|
for (auto iter = groups_.begin(); iter != groups_.end(); iter++) {
|
|
|
|
if ((*iter)->name() == group_name) {
|
|
|
|
groups_.erase(iter);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:36:09 +00:00
|
|
|
} // namespace fs_mgr
|
|
|
|
} // namespace android
|