Skip to content

Commit

Permalink
overlay: rm temp_img and add flag
Browse files Browse the repository at this point in the history
rm temp_img
when bind_mount set APATCH_BIND_MOUNT=true in sh
  • Loading branch information
Admirepowered committed Dec 6, 2024
1 parent 462b555 commit d1200d0
Show file tree
Hide file tree
Showing 5 changed files with 162 additions and 83 deletions.
3 changes: 2 additions & 1 deletion apd/Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions apd/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ serde = { version = "1", features = ["derive"] }
serde_json = "1"
regex-lite = "0.1.6"
encoding_rs = "0.8"
walkdir="2.4"
retry = "2"
humansize = "2"
libc = "0.2"
Expand Down
1 change: 1 addition & 0 deletions apd/src/defs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ pub const MODULE_UPDATE_TMP_IMG: &str = concatcp!(WORKING_DIR, "update_tmp.img")

// warning: this directory should not change, or you need to change the code in module_installer.sh!!!
pub const MODULE_UPDATE_TMP_DIR: &str = concatcp!(ADB_DIR, "modules_update/");
pub const MODULE_MOUNT_DIR: &str = concatcp!(ADB_DIR, "modules_mount/");

pub const SYSTEM_RW_DIR: &str = concatcp!(MODULE_DIR, ".rw/");

Expand Down
191 changes: 131 additions & 60 deletions apd/src/event.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
use crate::module;
use crate::supercall::fork_for_result;
use crate::utils::switch_cgroups;
use crate::utils::{switch_cgroups,ensure_dir_exists,get_work_dir,ensure_file_exists};
use crate::{
assets, defs, mount, restorecon, supercall,
supercall::{init_load_package_uid_config, init_load_su_path, refresh_ap_package_list},
Expand All @@ -19,9 +19,38 @@ use std::process::Command;
use std::sync::{Arc, Mutex};
use std::time::Duration;
use std::{collections::HashMap, thread};
use std::{env, fs};
use std::{env, fs,io};
use rustix::{fd::AsFd, fs::CWD, mount::*};
use std::fs::{remove_dir_all, rename};
use walkdir::WalkDir;
use extattr::{lsetxattr, lgetxattr, Flags as XattrFlags};

fn copy_with_xattr(src: &Path, dest: &Path) -> io::Result<()> {
fs::copy(src, dest)?;

if let Ok(xattr_value) = lgetxattr(src, "security.selinux") {
lsetxattr(dest, "security.selinux", &xattr_value, XattrFlags::empty())?;
}

Ok(())
}

fn copy_dir_with_xattr(src: &Path, dest: &Path) -> io::Result<()> {
for entry in WalkDir::new(src) {
let entry = entry?;
let rel_path = entry
.path()
.strip_prefix(src)
.map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))?;
let target_path = dest.join(rel_path);
if entry.file_type().is_dir() {
fs::create_dir_all(&target_path)?;
} else if entry.file_type().is_file() {
copy_with_xattr(entry.path(), &target_path)?;
}
}
Ok(())
}

fn mount_partition(partition_name: &str, lowerdir: &Vec<String>) -> Result<()> {
if lowerdir.is_empty() {
Expand All @@ -48,7 +77,7 @@ fn mount_partition(partition_name: &str, lowerdir: &Vec<String>) -> Result<()> {
mount::mount_overlay(&partition, lowerdir, workdir, upperdir)
}

pub fn mount_systemlessly(module_dir: &str) -> Result<()> {
pub fn mount_systemlessly(module_dir: &str,is_img: bool) -> Result<()> {
// construct overlay mount params
let dir = fs::read_dir(module_dir);
let Ok(dir) = dir else {
Expand Down Expand Up @@ -99,6 +128,50 @@ pub fn mount_systemlessly(module_dir: &str) -> Result<()> {
// mount /system first
if let Err(e) = mount_partition("system", &system_lowerdir) {
warn!("mount system failed: {:#}", e);
//ensure_file_exists(format!("{}",defs::BIND_MOUNT_FILE))?;
//ensure_clean_dir(defs::MODULE_DIR)?;
//info!("bind_mount enable,overlayfs is not work,clear module_dir");
if !is_img {

info!("fallback to modules.img");
let module_update_dir = defs::MODULE_DIR;
let module_dir = defs::MODULE_MOUNT_DIR;
let tmp_module_img = defs::MODULE_UPDATE_TMP_IMG;
let tmp_module_path = Path::new(tmp_module_img);

ensure_clean_dir(module_dir)?;
info!("- Preparing image");
if tmp_module_path.exists() { //if it have update,remove tmp file
std::fs::remove_file(tmp_module_path)?;
}
let total_size = calculate_total_size(Path::new(module_update_dir))?; //create modules adapt size
info!("Total size of files in '{}': {} bytes",tmp_module_path.display(),total_size);
let grow_size = 128 * 1024 * 1024 + total_size;
fs::File::create(tmp_module_img)
.context("Failed to create ext4 image file")?
.set_len(grow_size)
.context("Failed to extend ext4 image")?;
let result = Command::new("mkfs.ext4")
.arg("-b")
.arg("1024")
.arg(tmp_module_img)
.stdout(std::process::Stdio::piped())
.output()?;
ensure!(result.status.success(),"Failed to format ext4 image: {}",String::from_utf8(result.stderr).unwrap());
info!("Checking Image");
module::check_image(tmp_module_img)?;
info!("- Mounting image");
mount::AutoMountExt4::try_new(tmp_module_img, module_dir, false)
.with_context(|| "mount module image failed".to_string())?;
info!("mounted {} to {}", tmp_module_img, module_dir);
let _ = restorecon::setsyscon(module_dir);
let command_string = format!("cp --preserve=context -R {}* {};",module_update_dir,module_dir);
let args = vec!["-c",&command_string];
let _ = utils::run_command("sh", &args, None)?.wait()?;
mount_systemlessly(module_dir,true)?;
return Ok(());
}

}

// mount other partitions
Expand Down Expand Up @@ -138,7 +211,26 @@ pub fn calculate_total_size(path: &Path) -> std::io::Result<u64> {
}
Ok(total_size)
}
pub fn move_file(module_update_dir: &str,module_dir: &str)-> Result<()> {
for entry in fs::read_dir(module_update_dir)? {
let entry = entry?;
let file_name = entry.file_name();
let file_name_str = file_name.to_string_lossy();

if entry.path().is_dir() {
let source_path = Path::new(module_update_dir).join(file_name_str.as_ref());
let target_path = Path::new(module_dir).join(file_name_str.as_ref());
if target_path.exists() {
info!("Removing existing folder in target directory: {}", file_name_str);
remove_dir_all(&target_path)?;
}

info!("Moving {} to target directory", file_name_str);
rename(&source_path, &target_path)?;
}
}
return Ok(());
}
pub fn on_post_data_fs(superkey: Option<String>) -> Result<()> {
utils::umask(0);
use std::process::Stdio;
Expand Down Expand Up @@ -166,7 +258,7 @@ pub fn on_post_data_fs(superkey: Option<String>) -> Result<()> {
.expect("Failed to set permissions");
}
let mut command_string = format!(
"rm {}*.old.log; for file in {}*; do mv \"$file\" \"$file.old.log\"; done",
"rm -rf {}*.old.log; for file in {}*; do mv \"$file\" \"$file.old.log\"; done",
defs::APATCH_LOG_FOLDER,
defs::APATCH_LOG_FOLDER
);
Expand Down Expand Up @@ -239,65 +331,17 @@ pub fn on_post_data_fs(superkey: Option<String>) -> Result<()> {

let tmp_module_img = defs::MODULE_UPDATE_TMP_IMG;
let tmp_module_path = Path::new(tmp_module_img);
move_file(module_update_dir,module_dir)?;

if (module_update_flag.exists() || !tmp_module_path.exists()) && utils::should_enable_overlay()? {// only if modules change,then renew modules file
ensure_clean_dir(module_dir)?;
info!("remove update flag");
let _ = fs::remove_file(module_update_flag);
info!("- Preparing image");
if tmp_module_path.exists() { //if it have update,remove tmp file
std::fs::remove_file(tmp_module_path)?;
}
let total_size = calculate_total_size(Path::new(module_update_dir))?; //create modules adapt size
info!("Total size of files in '{}': {} bytes",tmp_module_path.display(),total_size);
let grow_size = 128 * 1024 * 1024 + total_size;
fs::File::create(tmp_module_img)
.context("Failed to create ext4 image file")?
.set_len(grow_size)
.context("Failed to extend ext4 image")?;
let result = Command::new("mkfs.ext4")
.arg("-b")
.arg("1024")
.arg(tmp_module_img)
.stdout(std::process::Stdio::piped())
.output()?;
ensure!(result.status.success(),"Failed to format ext4 image: {}",String::from_utf8(result.stderr).unwrap());
info!("Checking Image");
module::check_image(tmp_module_img)?;
info!("- Mounting image");
mount::AutoMountExt4::try_new(tmp_module_img, module_dir, false)
.with_context(|| "mount module image failed".to_string())?;
info!("mounted {} to {}", tmp_module_img, module_dir);
let _ = restorecon::setsyscon(module_dir);
command_string = format!("cp --preserve=context -R {}* {};",module_update_dir,module_dir);
args = vec!["-c",&command_string];
let _ = utils::run_command("sh", &args, None)?.wait()?;
}else if utils::should_enable_overlay()? {//mounting last time img file
ensure_clean_dir(module_dir)?;
info!("- Mounting image");
mount::AutoMountExt4::try_new(tmp_module_img, module_dir, false)
.with_context(|| "mount module image failed".to_string())?;
}else {
for entry in fs::read_dir(module_update_dir)? {
let entry = entry?;
let file_name = entry.file_name();
let file_name_str = file_name.to_string_lossy();

if entry.path().is_dir() {
let source_path = Path::new(module_update_dir).join(file_name_str.as_ref());
let target_path = Path::new(module_dir).join(file_name_str.as_ref());
if target_path.exists() {
info!("Removing existing folder in target directory: {}", file_name_str);
remove_dir_all(&target_path)?;
}

info!("Moving {} to target directory", file_name_str);
rename(&source_path, &target_path)?;
}
}


info!("remove update flag");
let _ = fs::remove_file(module_update_flag);
if tmp_module_path.exists() { //if it have update,remove tmp file
std::fs::remove_file(tmp_module_path)?;
}



if safe_mode {
warn!("safe mode, skip post-fs-data scripts and disable all modules!");
Expand Down Expand Up @@ -336,9 +380,36 @@ pub fn on_post_data_fs(superkey: Option<String>) -> Result<()> {
}
if utils::should_enable_overlay()? {
// mount module systemlessly by overlay
if let Err(e) = mount_systemlessly(module_dir) {
let work_dir = get_work_dir();
let tmp_dir = PathBuf::from(work_dir.clone());
ensure_dir_exists(&tmp_dir)?;
mount(defs::AP_OVERLAY_SOURCE, &tmp_dir, "tmpfs", MountFlags::empty(), "").context("mount tmp")?;
mount_change(&tmp_dir, MountPropagationFlags::PRIVATE).context("make tmp private")?;
let dir_names = vec!["vendor", "product", "system_ext", "odm", "oem", "system"];
let dir = fs::read_dir(module_dir)?;
for entry in dir.flatten() {
let module_path = entry.path();
if module_path.is_dir() {
let module_name = module_path.file_name().unwrap().to_string_lossy();
let module_dest = Path::new(&work_dir).join(module_name.as_ref());

for sub_dir in dir_names.iter() {
let sub_dir_path = module_path.join(sub_dir);
if sub_dir_path.exists() && sub_dir_path.is_dir() {
let sub_dir_dest = module_dest.join(sub_dir);
fs::create_dir_all(&sub_dir_dest)?;

copy_dir_with_xattr(&sub_dir_path, &sub_dir_dest)?;
}
}
}
}
if let Err(e) = mount_systemlessly(&get_work_dir(),false) {
warn!("do systemless mount failed: {}", e);
}
if let Err(e) = unmount(&tmp_dir, UnmountFlags::DETACH) {
log::error!("failed to unmount tmp {}", e);
}
}else{
if let Err(e) = systemless_bind_mount(module_dir) {
warn!("do systemless bind_mount failed: {}", e);
Expand Down
49 changes: 27 additions & 22 deletions apd/src/module.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#[allow(clippy::wildcard_imports)]
use crate::utils::*;
use crate::{
assets, defs, restorecon,
assets, defs, restorecon,utils,
};
use regex_lite::Regex;
use anyhow::{anyhow, bail, ensure, Context, Result};
Expand Down Expand Up @@ -67,6 +67,7 @@ fn exec_install_script(module_file: &str) -> Result<()> {
.env("APATCH", "true")
.env("APATCH_VER", defs::VERSION_NAME)
.env("APATCH_VER_CODE", defs::VERSION_CODE)
.env("APATCH_BIND_MOUNT", format!("{}", !utils::should_enable_overlay()?))
.env("OUTFD", "1")
.env("ZIPFILE", realpath)
.status()?;
Expand Down Expand Up @@ -197,6 +198,7 @@ fn exec_script<T: AsRef<Path>>(path: T, wait: bool) -> Result<()> {
.env("APATCH", "true")
.env("APATCH_VER", defs::VERSION_NAME)
.env("APATCH_VER_CODE", defs::VERSION_CODE)
.env("APATCH_BIND_MOUNT", format!("{}", !utils::should_enable_overlay()?))
.env(
"PATH",
format!(
Expand Down Expand Up @@ -428,12 +430,13 @@ pub fn _uninstall_module(id: &str, update_dir: &str) -> Result<()> {

}
pub fn uninstall_module(id: &str) -> Result<()> {
let result = _uninstall_module(id, defs::MODULE_DIR);
if should_enable_overlay()?{
_uninstall_module(id, defs::MODULE_UPDATE_TMP_DIR)?;
}else{
return result;
}
//let result = _uninstall_module(id, defs::MODULE_DIR);
//if should_enable_overlay()?{
// _uninstall_module(id, defs::MODULE_UPDATE_TMP_DIR)?;
//}else{
// return result;
//}
_uninstall_module(id, defs::MODULE_DIR)?;
Ok(())
}

Expand Down Expand Up @@ -468,12 +471,13 @@ pub fn enable_module(id: &str) -> Result<()> {
let update_dir = Path::new(defs::MODULE_DIR);
let update_dir_update = Path::new(defs::MODULE_UPDATE_TMP_DIR);

let result = enable_module_update(id, update_dir);
if should_enable_overlay()?{
enable_module_update(id, update_dir_update)?;
}else{
return result;
}
//let result = enable_module_update(id, update_dir);
//if should_enable_overlay()?{
// enable_module_update(id, update_dir_update)?;
//}else{
// return result;
//}
enable_module_update(id, update_dir)?;
Ok(())
}

Expand All @@ -489,14 +493,14 @@ pub fn enable_module_update(id: &str,update_dir: &Path) -> Result<()> {
pub fn disable_module(id: &str) -> Result<()> {
let update_dir = Path::new(defs::MODULE_DIR);
let update_dir_update = Path::new(defs::MODULE_UPDATE_TMP_DIR);
let result = disable_module_update(id, update_dir);

if should_enable_overlay()?{
disable_module_update(id, update_dir_update)?;
}else{
return result;
}

//let result = disable_module_update(id, update_dir);
//if should_enable_overlay()?{
// disable_module_update(id, update_dir_update)?;
//}else{
// return result;
//}
disable_module_update(id, update_dir)?;

Ok(())
}
Expand All @@ -518,8 +522,9 @@ pub fn disable_all_modules() -> Result<()> {
}

// we assume the module dir is already mounted
let _ = disable_all_modules_update(defs::MODULE_DIR);
disable_all_modules_update(defs::MODULE_UPDATE_TMP_DIR)?;
//let _ = disable_all_modules_update(defs::MODULE_DIR);
//disable_all_modules_update(defs::MODULE_UPDATE_TMP_DIR)?;
disable_all_modules_update(defs::MODULE_DIR)?;
Ok(())
}

Expand Down

0 comments on commit d1200d0

Please sign in to comment.