android/system/core/fs_mgr/libdm和dm设备的创建
DmTargetLinear(uint64_t start, uint64_t length, const std::string& block_device,
uint64_t physical_sector)
// 开始,长度,/dev/block/mmcblk,物理扇区起始数
: DmTarget(start, length), block_device_(block_device), physical_sector_(physical_sector) {}
DmTarget(uint64_t start, uint64_t length) : start_(start), length_(length) {}
std::string DmTarget::Serialize() const {
// Create a string containing a dm_target_spec, parameter data, and an
// explicit null terminator.
std::string data(sizeof(dm_target_spec), ‘\0‘);
// dm_target_spec + /dev/block/mmcblk0p05 1234
data += GetParameterString();
data.push_back(‘\0‘);
// The kernel expects each target to be 8-byte aligned.
size_t padding = DM_ALIGN(data.size()) - data.size();
for (size_t i = 0; i < padding; i++) {
data.push_back(‘\0‘);
}
// Finally fill in the dm_target_spec.
struct dm_target_spec* spec = reinterpret_cast<struct dm_target_spec*>(&data[0]);
spec->sector_start = start();
spec->length = size();
snprintf(spec->target_type, sizeof(spec->target_type), "%s", name().c_str());
spec->next = (uint32_t)data.size();
return data;
}
struct dm_target_spec {
__u64 sector_start;
__u64 length;
__s32 status;
__u32 next;
char target_type[DM_MAX_TYPE_NAME];
};
std::string DmTargetLinear::GetParameterString() const {
return block_device_ + " " + std::to_string(physical_sector_);
}
DeviceMapper::DeviceMapper() : fd_(-1) {
// 打开/dev/device-mapper设备
fd_ = TEMP_FAILURE_RETRY(open("/dev/device-mapper", O_RDWR | O_CLOEXEC));
if (fd_ < 0) {
PLOG(ERROR) << "Failed to open device-mapper";
}
}
// system_a,table,path是要传回去的,/sys/devices/virtual/block/dm-0/dm可以看到
bool DeviceMapper::CreateDevice(const std::string& name, const DmTable& table, std::string* path,
const std::chrono::milliseconds& timeout_ms) {
// 1. 生成uuid
std::string uuid = GenerateUuid();
// 2. 通过ioctl函数创建DM_DEV_CREATE设备
if (!CreateDevice(name, uuid)) {
return false;
}
// We use the unique path for testing whether the device is ready. After
// that, it‘s safe to use the dm-N path which is compatible with callers
// that expect it to be formatted as such.
std::string unique_path;
// 1. 加载dm table
// 2. 获取设备文件/dev/block/mapper/by-uuid/uuid
// 3. 获取设备文件/dev/block/dm-0
if (!LoadTableAndActivate(name, table) || !GetDeviceUniquePath(name, &unique_path) ||
!GetDmDevicePathByName(name, path)) {
DeleteDevice(name);
return false;
}
if (timeout_ms <= std::chrono::milliseconds::zero()) {
return true;
}
if (IsRecovery()) {
bool non_ab_device = android::base::GetProperty("ro.build.ab_update", "").empty();
int sdk = android::base::GetIntProperty("ro.build.version.sdk", 0);
if (non_ab_device && sdk && sdk <= 29) {
LOG(INFO) << "Detected ueventd incompatibility, reverting to legacy libdm behavior.";
unique_path = *path;
}
}
if (!WaitForFile(unique_path, timeout_ms)) {
LOG(ERROR) << "Failed waiting for device path: " << unique_path;
DeleteDevice(name);
return false;
}
return true;
}
bool DeviceMapper::CreateDevice(const std::string& name, const std::string& uuid) {
if (name.empty()) {
LOG(ERROR) << "Unnamed device mapper device creation is not supported";
return false;
}
if (name.size() >= DM_NAME_LEN) {
LOG(ERROR) << "[" << name << "] is too long to be device mapper name";
return false;
}
struct dm_ioctl io;
InitIo(&io, name);
if (!uuid.empty()) {
snprintf(io.uuid, sizeof(io.uuid), "%s", uuid.c_str());
}
// 创建dm设备
if (ioctl(fd_, DM_DEV_CREATE, &io)) {
PLOG(ERROR) << "DM_DEV_CREATE failed for [" << name << "]";
return false;
}
// Check to make sure the newly created device doesn‘t already have targets
// added or opened by someone
CHECK(io.target_count == 0) << "Unexpected targets for newly created [" << name << "] device";
CHECK(io.open_count == 0) << "Unexpected opens for newly created [" << name << "] device";
// Creates a new device mapper device with the name passed in
return true;
}
static std::string GenerateUuid() {
uuid_t uuid_bytes;
// 通过/dev/urandom生成随机数
uuid_generate(uuid_bytes);
char uuid_chars[37] = {};
// 将二进制格式转成36-byte字符串的类型
uuid_unparse_lower(uuid_bytes, uuid_chars);
return std::string{uuid_chars};
}
void DeviceMapper::InitIo(struct dm_ioctl* io, const std::string& name) const {
CHECK(io != nullptr) << "nullptr passed to dm_ioctl initialization";
memset(io, 0, sizeof(*io));
io->version[0] = DM_VERSION0;
io->version[1] = DM_VERSION1;
io->version[2] = DM_VERSION2;
io->data_size = sizeof(*io);
io->data_start = 0;
if (!name.empty()) {
snprintf(io->name, sizeof(io->name), "%s", name.c_str());
}
}
// system_a table
bool DeviceMapper::LoadTableAndActivate(const std::string& name, const DmTable& table) {
std::string ioctl_buffer(sizeof(struct dm_ioctl), 0);
// 建成serialize
ioctl_buffer += table.Serialize();
struct dm_ioctl* io = reinterpret_cast<struct dm_ioctl*>(&ioctl_buffer[0]);
InitIo(io, name);
io->data_size = ioctl_buffer.size();
io->data_start = sizeof(struct dm_ioctl);
io->target_count = static_cast<uint32_t>(table.num_targets());
if (table.readonly()) {
io->flags |= DM_READONLY_FLAG;
}// 设成DM_TABLE_LOAD
if (ioctl(fd_, DM_TABLE_LOAD, io)) {
PLOG(ERROR) << "DM_TABLE_LOAD failed";
return false;
}
InitIo(io, name);
// 设成DM_DEV_SUSPEND
if (ioctl(fd_, DM_DEV_SUSPEND, io)) {
PLOG(ERROR) << "DM_TABLE_SUSPEND resume failed";
return false;
}
return true;
}
bool DeviceMapper::GetDeviceUniquePath(const std::string& name, std::string* path) {
struct dm_ioctl io;
InitIo(&io, name);
if (ioctl(fd_, DM_DEV_STATUS, &io) < 0) {
PLOG(ERROR) << "Failed to get device path: " << name;
return false;
}
if (io.uuid[0] == ‘\0‘) {
LOG(ERROR) << "Device does not have a unique path: " << name;
return false;
}
*path = "/dev/block/mapper/by-uuid/"s + io.uuid;
return true;
}
bool DeviceMapper::GetDmDevicePathByName(const std::string& name, std::string* path) {
struct dm_ioctl io;
InitIo(&io, name);
if (ioctl(fd_, DM_DEV_STATUS, &io) < 0) {
PLOG(WARNING) << "DM_DEV_STATUS failed for " << name;
return false;
}
uint32_t dev_num = minor(io.dev);
*path = "/dev/block/dm-" + std::to_string(dev_num);
return true;
}
std::string DmTable::Serialize() const {
if (!valid()) {
return "";
}
std::string table;
for (const auto& target : targets_) {
table += target->Serialize();
}
return table;
}
原文:https://www.cnblogs.com/pyjetson/p/14872499.html