mirror of
https://github.com/topjohnwu/Magisk.git
synced 2025-01-12 11:33:37 +00:00
Reorganize code
This commit is contained in:
parent
21ddb26db8
commit
7b5ff99cd1
@ -1,7 +1,8 @@
|
|||||||
|
use std::cmp::min;
|
||||||
use std::ffi::CStr;
|
use std::ffi::CStr;
|
||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::io::BufRead;
|
use std::io::{BufRead, Write};
|
||||||
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd, RawFd};
|
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd, RawFd};
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
@ -140,3 +141,19 @@ pub fn read_lines<P: AsRef<Path>>(path: P) -> io::Result<io::Lines<io::BufReader
|
|||||||
let file = File::open(path)?;
|
let file = File::open(path)?;
|
||||||
Ok(io::BufReader::new(file).lines())
|
Ok(io::BufReader::new(file).lines())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait WriteExt {
|
||||||
|
fn write_zeros(&mut self, len: usize) -> io::Result<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Write> WriteExt for T {
|
||||||
|
fn write_zeros(&mut self, mut len: usize) -> io::Result<()> {
|
||||||
|
let mut buf = [0 as u8; 4096];
|
||||||
|
while len > 0 {
|
||||||
|
let l = min(buf.len(), len);
|
||||||
|
self.write_all(&mut buf[..l])?;
|
||||||
|
len -= l;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,23 +1,14 @@
|
|||||||
#![feature(format_args_nl)]
|
#![feature(format_args_nl)]
|
||||||
|
|
||||||
|
pub use base;
|
||||||
|
pub use payload::*;
|
||||||
|
|
||||||
|
mod payload;
|
||||||
mod update_metadata;
|
mod update_metadata;
|
||||||
|
|
||||||
use crate::update_metadata::install_operation::Type;
|
|
||||||
use crate::update_metadata::DeltaArchiveManifest;
|
|
||||||
pub use base;
|
|
||||||
use base::error;
|
|
||||||
use base::libc::c_char;
|
|
||||||
use byteorder::{BigEndian, ReadBytesExt};
|
|
||||||
use protobuf::{EnumFull, Message};
|
|
||||||
use std::ffi::CStr;
|
|
||||||
use std::fs::File;
|
|
||||||
use std::io::{BufReader, ErrorKind, Read, Seek, SeekFrom};
|
|
||||||
use std::io::{Error as IOError, Write};
|
|
||||||
use std::os::unix::io::AsRawFd;
|
|
||||||
|
|
||||||
#[cxx::bridge(namespace = "rust")]
|
#[cxx::bridge(namespace = "rust")]
|
||||||
mod ffi {
|
pub mod ffi {
|
||||||
unsafe extern "C++" {
|
extern "C++" {
|
||||||
pub unsafe fn decompress(in_: *const u8, in_size: u64, fd: i32) -> bool;
|
pub unsafe fn decompress(in_: *const u8, in_size: u64, fd: i32) -> bool;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -28,170 +19,3 @@ mod ffi {
|
|||||||
) -> bool;
|
) -> bool;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[macro_export]
|
|
||||||
macro_rules! data_err {
|
|
||||||
($fmt:expr) => { IOError::new(ErrorKind::InvalidData, format!($fmt)) };
|
|
||||||
($fmt:expr, $($args:tt)*) => { IOError::new(ErrorKind::InvalidData, format!($fmt, $($args)*)) };
|
|
||||||
}
|
|
||||||
|
|
||||||
static PAYLOAD_MAGIC: &str = "CrAU";
|
|
||||||
|
|
||||||
pub fn do_extract_boot_from_payload(in_path: &str, out_path: &str) -> Result<(), IOError> {
|
|
||||||
// let mut file = File::create(out_path).unwrap();
|
|
||||||
let in_file = File::open(in_path)?;
|
|
||||||
let mut reader = BufReader::new(in_file);
|
|
||||||
|
|
||||||
let buf = &mut [0u8; 4];
|
|
||||||
reader.read_exact(buf)?;
|
|
||||||
|
|
||||||
if buf != PAYLOAD_MAGIC.as_bytes() {
|
|
||||||
return Err(data_err!("invalid payload magic"));
|
|
||||||
}
|
|
||||||
|
|
||||||
let version = reader.read_u64::<BigEndian>()?;
|
|
||||||
|
|
||||||
if version != 2 {
|
|
||||||
return Err(data_err!("unsupported version {}", version));
|
|
||||||
}
|
|
||||||
let manifest_len = reader.read_u64::<BigEndian>()?;
|
|
||||||
|
|
||||||
if manifest_len == 0 {
|
|
||||||
return Err(data_err!("manifest length is zero"));
|
|
||||||
}
|
|
||||||
|
|
||||||
let manifest_sig_len = reader.read_u32::<BigEndian>()?;
|
|
||||||
|
|
||||||
if manifest_sig_len == 0 {
|
|
||||||
return Err(data_err!("manifest signature length is zero"));
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut buf = vec![0; manifest_len as usize];
|
|
||||||
|
|
||||||
reader.read_exact(&mut buf)?;
|
|
||||||
|
|
||||||
let manifest = DeltaArchiveManifest::parse_from_bytes(&buf)?;
|
|
||||||
|
|
||||||
if !manifest.has_minor_version() || manifest.minor_version() != 0 {
|
|
||||||
return Err(data_err!(
|
|
||||||
"delta payloads are not supported, please use a full payload file"
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
if !manifest.has_block_size() {
|
|
||||||
return Err(data_err!("block size not found"));
|
|
||||||
}
|
|
||||||
|
|
||||||
let boot = manifest.partitions.iter().find(|partition| {
|
|
||||||
partition.has_partition_name() && partition.partition_name() == "init_boot"
|
|
||||||
});
|
|
||||||
|
|
||||||
let boot = match boot {
|
|
||||||
Some(boot) => Some(boot),
|
|
||||||
None => manifest.partitions.iter().find(|partition| {
|
|
||||||
partition.has_partition_name() && partition.partition_name() == "boot"
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
let boot = boot.ok_or(data_err!("boot partition not found"))?;
|
|
||||||
|
|
||||||
let base_offset = reader.stream_position()?;
|
|
||||||
|
|
||||||
let base_offset = base_offset + manifest_sig_len as u64;
|
|
||||||
|
|
||||||
let block_size = manifest
|
|
||||||
.block_size
|
|
||||||
.ok_or(data_err!("block size not found"))? as u64;
|
|
||||||
|
|
||||||
let mut out_file = File::create(out_path)?;
|
|
||||||
|
|
||||||
for operation in boot.operations.iter() {
|
|
||||||
let data_len = operation
|
|
||||||
.data_length
|
|
||||||
.ok_or(data_err!("data length not found"))?;
|
|
||||||
|
|
||||||
let data_offset = operation
|
|
||||||
.data_offset
|
|
||||||
.ok_or(data_err!("data offset not found"))?;
|
|
||||||
|
|
||||||
let data_type = operation
|
|
||||||
.type_
|
|
||||||
.ok_or(data_err!("operation type not found"))?;
|
|
||||||
|
|
||||||
let data_type = data_type
|
|
||||||
.enum_value()
|
|
||||||
.map_err(|_| data_err!("operation type not valid"))?;
|
|
||||||
|
|
||||||
let mut buf = vec![0; data_len as usize];
|
|
||||||
|
|
||||||
reader.seek(SeekFrom::Start(base_offset + data_offset))?;
|
|
||||||
reader.read_exact(&mut buf)?;
|
|
||||||
|
|
||||||
let out_offset = operation
|
|
||||||
.dst_extents
|
|
||||||
.get(0)
|
|
||||||
.ok_or(data_err!("dst extents not found"))?
|
|
||||||
.start_block
|
|
||||||
.ok_or(data_err!("start block not found"))?
|
|
||||||
* block_size;
|
|
||||||
|
|
||||||
match data_type {
|
|
||||||
Type::REPLACE => {
|
|
||||||
out_file.seek(SeekFrom::Start(out_offset))?;
|
|
||||||
out_file.write_all(&buf)?;
|
|
||||||
}
|
|
||||||
Type::ZERO => {
|
|
||||||
for ext in operation.dst_extents.iter() {
|
|
||||||
let out_seek =
|
|
||||||
ext.start_block.ok_or(data_err!("start block not found"))? * block_size;
|
|
||||||
let num_blocks = ext.num_blocks.ok_or(data_err!("num blocks not found"))?;
|
|
||||||
out_file.seek(SeekFrom::Start(out_seek))?;
|
|
||||||
out_file.write_all(&vec![0; num_blocks as usize])?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Type::REPLACE_BZ | Type::REPLACE_XZ => {
|
|
||||||
out_file.seek(SeekFrom::Start(out_offset))?;
|
|
||||||
unsafe {
|
|
||||||
if !ffi::decompress(buf.as_ptr(), buf.len() as u64, out_file.as_raw_fd()) {
|
|
||||||
return Err(data_err!("decompression failed"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
return Err(data_err!(
|
|
||||||
"unsupported operation type: {}",
|
|
||||||
data_type.descriptor().name()
|
|
||||||
));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn extract_boot_from_payload(in_path: *const c_char, out_path: *const c_char) -> bool {
|
|
||||||
let in_path = unsafe { CStr::from_ptr(in_path) };
|
|
||||||
let out_path = unsafe { CStr::from_ptr(out_path) };
|
|
||||||
let in_path = match in_path.to_str() {
|
|
||||||
Ok(path) => path,
|
|
||||||
Err(_) => {
|
|
||||||
error!("Failed to extract boot from payload: input path invalid");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
let out_path = match out_path.to_str() {
|
|
||||||
Ok(path) => path,
|
|
||||||
Err(_) => {
|
|
||||||
error!("Failed to extract boot from payload: output path invalid");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
match do_extract_boot_from_payload(in_path, out_path) {
|
|
||||||
Ok(_) => true,
|
|
||||||
Err(err) => {
|
|
||||||
error!("Failed to extract boot from payload: \"{}\"", err);
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
158
native/src/boot/payload.rs
Normal file
158
native/src/boot/payload.rs
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
use std::fs::File;
|
||||||
|
use std::io;
|
||||||
|
use std::io::{BufReader, ErrorKind, Read, Seek, SeekFrom, Write};
|
||||||
|
use std::os::fd::AsRawFd;
|
||||||
|
|
||||||
|
use byteorder::{BigEndian, ReadBytesExt};
|
||||||
|
use protobuf::{EnumFull, Message};
|
||||||
|
|
||||||
|
use base::libc::c_char;
|
||||||
|
use base::WriteExt;
|
||||||
|
use base::{error, ptr_to_str};
|
||||||
|
|
||||||
|
use crate::ffi;
|
||||||
|
use crate::update_metadata::install_operation::Type;
|
||||||
|
use crate::update_metadata::DeltaArchiveManifest;
|
||||||
|
|
||||||
|
macro_rules! data_err {
|
||||||
|
($fmt:expr) => { io::Error::new(ErrorKind::InvalidData, format!($fmt)) };
|
||||||
|
($fmt:expr, $($args:tt)*) => {
|
||||||
|
io::Error::new(ErrorKind::InvalidData, format!($fmt, $($args)*))
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static PAYLOAD_MAGIC: &str = "CrAU";
|
||||||
|
|
||||||
|
fn do_extract_boot_from_payload(in_path: &str, out_path: &str) -> io::Result<()> {
|
||||||
|
let mut reader = BufReader::new(File::open(in_path)?);
|
||||||
|
|
||||||
|
let buf = &mut [0u8; 4];
|
||||||
|
reader.read_exact(buf)?;
|
||||||
|
|
||||||
|
if buf != PAYLOAD_MAGIC.as_bytes() {
|
||||||
|
return Err(data_err!("invalid payload magic"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let version = reader.read_u64::<BigEndian>()?;
|
||||||
|
if version != 2 {
|
||||||
|
return Err(data_err!("unsupported version {}", version));
|
||||||
|
}
|
||||||
|
|
||||||
|
let manifest_len = reader.read_u64::<BigEndian>()?;
|
||||||
|
if manifest_len == 0 {
|
||||||
|
return Err(data_err!("manifest length is zero"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let manifest_sig_len = reader.read_u32::<BigEndian>()?;
|
||||||
|
if manifest_sig_len == 0 {
|
||||||
|
return Err(data_err!("manifest signature length is zero"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut buf = vec![0; manifest_len as usize];
|
||||||
|
|
||||||
|
reader.read_exact(&mut buf)?;
|
||||||
|
|
||||||
|
let manifest = DeltaArchiveManifest::parse_from_bytes(&buf)?;
|
||||||
|
if !manifest.has_minor_version() || manifest.minor_version() != 0 {
|
||||||
|
return Err(data_err!(
|
||||||
|
"delta payloads are not supported, please use a full payload file"
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if !manifest.has_block_size() {
|
||||||
|
return Err(data_err!("block size not found"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let boot = manifest.partitions.iter().find(|partition| {
|
||||||
|
partition.has_partition_name() && partition.partition_name() == "init_boot"
|
||||||
|
});
|
||||||
|
let boot = match boot {
|
||||||
|
Some(boot) => Some(boot),
|
||||||
|
None => manifest.partitions.iter().find(|partition| {
|
||||||
|
partition.has_partition_name() && partition.partition_name() == "boot"
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
let boot = boot.ok_or(data_err!("boot partition not found"))?;
|
||||||
|
|
||||||
|
let base_offset = reader.stream_position()? + manifest_sig_len as u64;
|
||||||
|
|
||||||
|
let block_size = manifest
|
||||||
|
.block_size
|
||||||
|
.ok_or(data_err!("block size not found"))? as u64;
|
||||||
|
|
||||||
|
let mut out_file = File::create(out_path)?;
|
||||||
|
|
||||||
|
for operation in boot.operations.iter() {
|
||||||
|
let data_len = operation
|
||||||
|
.data_length
|
||||||
|
.ok_or(data_err!("data length not found"))?;
|
||||||
|
|
||||||
|
let data_offset = operation
|
||||||
|
.data_offset
|
||||||
|
.ok_or(data_err!("data offset not found"))?;
|
||||||
|
|
||||||
|
let data_type = operation
|
||||||
|
.type_
|
||||||
|
.ok_or(data_err!("operation type not found"))?;
|
||||||
|
|
||||||
|
let data_type = data_type
|
||||||
|
.enum_value()
|
||||||
|
.map_err(|_| data_err!("operation type not valid"))?;
|
||||||
|
|
||||||
|
let mut buf = vec![0; data_len as usize];
|
||||||
|
|
||||||
|
reader.seek(SeekFrom::Start(base_offset + data_offset))?;
|
||||||
|
reader.read_exact(&mut buf)?;
|
||||||
|
|
||||||
|
let out_offset = operation
|
||||||
|
.dst_extents
|
||||||
|
.get(0)
|
||||||
|
.ok_or(data_err!("dst extents not found"))?
|
||||||
|
.start_block
|
||||||
|
.ok_or(data_err!("start block not found"))?
|
||||||
|
* block_size;
|
||||||
|
|
||||||
|
match data_type {
|
||||||
|
Type::REPLACE => {
|
||||||
|
out_file.seek(SeekFrom::Start(out_offset))?;
|
||||||
|
out_file.write_all(&buf)?;
|
||||||
|
}
|
||||||
|
Type::ZERO => {
|
||||||
|
for ext in operation.dst_extents.iter() {
|
||||||
|
let out_seek =
|
||||||
|
ext.start_block.ok_or(data_err!("start block not found"))? * block_size;
|
||||||
|
let num_blocks = ext.num_blocks.ok_or(data_err!("num blocks not found"))?;
|
||||||
|
out_file.seek(SeekFrom::Start(out_seek))?;
|
||||||
|
out_file.write_zeros(num_blocks as usize)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Type::REPLACE_BZ | Type::REPLACE_XZ => {
|
||||||
|
out_file.seek(SeekFrom::Start(out_offset))?;
|
||||||
|
unsafe {
|
||||||
|
if !ffi::decompress(buf.as_ptr(), buf.len() as u64, out_file.as_raw_fd()) {
|
||||||
|
return Err(data_err!("decompression failed"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
return Err(data_err!(
|
||||||
|
"unsupported operation type: {}",
|
||||||
|
data_type.descriptor().name()
|
||||||
|
));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn extract_boot_from_payload(in_path: *const c_char, out_path: *const c_char) -> bool {
|
||||||
|
let in_path = ptr_to_str(in_path);
|
||||||
|
let out_path = ptr_to_str(out_path);
|
||||||
|
match do_extract_boot_from_payload(in_path, out_path) {
|
||||||
|
Ok(_) => true,
|
||||||
|
Err(err) => {
|
||||||
|
error!("Failed to extract boot from payload: {}", err);
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user