From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: <pbs-devel-bounces@lists.proxmox.com> Received: from firstgate.proxmox.com (firstgate.proxmox.com [IPv6:2a01:7e0:0:424::9]) by lore.proxmox.com (Postfix) with ESMTPS id 6F54E1FF16B for <inbox@lore.proxmox.com>; Thu, 6 Feb 2025 16:02:30 +0100 (CET) Received: from firstgate.proxmox.com (localhost [127.0.0.1]) by firstgate.proxmox.com (Proxmox) with ESMTP id AB8882A731; Thu, 6 Feb 2025 16:02:28 +0100 (CET) Date: Thu, 6 Feb 2025 16:01:54 +0100 From: Wolfgang Bumiller <w.bumiller@proxmox.com> To: Filip Schauer <f.schauer@proxmox.com> Message-ID: <tto4hi2gxxlta3yumimb3gtjc2lyptrkhu3hf6cyfhj33supwr@twn4jty73ixv> References: <20241204103331.69226-1-f.schauer@proxmox.com> MIME-Version: 1.0 Content-Disposition: inline In-Reply-To: <20241204103331.69226-1-f.schauer@proxmox.com> X-SPAM-LEVEL: Spam detection results: 0 AWL 0.082 Adjusted score from AWL reputation of From: address BAYES_00 -1.9 Bayes spam probability is 0 to 1% DMARC_MISSING 0.1 Missing DMARC policy KAM_DMARC_STATUS 0.01 Test Rule for DKIM or SPF Failure with Strict Alignment RCVD_IN_VALIDITY_CERTIFIED_BLOCKED 0.001 ADMINISTRATOR NOTICE: The query to Validity was blocked. See https://knowledge.validity.com/hc/en-us/articles/20961730681243 for more information. RCVD_IN_VALIDITY_RPBL_BLOCKED 0.001 ADMINISTRATOR NOTICE: The query to Validity was blocked. See https://knowledge.validity.com/hc/en-us/articles/20961730681243 for more information. RCVD_IN_VALIDITY_SAFE_BLOCKED 0.001 ADMINISTRATOR NOTICE: The query to Validity was blocked. See https://knowledge.validity.com/hc/en-us/articles/20961730681243 for more information. SPF_HELO_NONE 0.001 SPF: HELO does not publish an SPF Record SPF_PASS -0.001 SPF: sender matches SPF record Subject: Re: [pbs-devel] [PATCH backup] fix #5946: disks: wipe: ensure GPT header backup is wiped X-BeenThere: pbs-devel@lists.proxmox.com X-Mailman-Version: 2.1.29 Precedence: list List-Id: Proxmox Backup Server development discussion <pbs-devel.lists.proxmox.com> List-Unsubscribe: <https://lists.proxmox.com/cgi-bin/mailman/options/pbs-devel>, <mailto:pbs-devel-request@lists.proxmox.com?subject=unsubscribe> List-Archive: <http://lists.proxmox.com/pipermail/pbs-devel/> List-Post: <mailto:pbs-devel@lists.proxmox.com> List-Help: <mailto:pbs-devel-request@lists.proxmox.com?subject=help> List-Subscribe: <https://lists.proxmox.com/cgi-bin/mailman/listinfo/pbs-devel>, <mailto:pbs-devel-request@lists.proxmox.com?subject=subscribe> Reply-To: Proxmox Backup Server development discussion <pbs-devel@lists.proxmox.com> Cc: pbs-devel@lists.proxmox.com Content-Type: text/plain; charset="us-ascii" Content-Transfer-Encoding: 7bit Errors-To: pbs-devel-bounces@lists.proxmox.com Sender: "pbs-devel" <pbs-devel-bounces@lists.proxmox.com> On Wed, Dec 04, 2024 at 11:33:31AM +0100, Filip Schauer wrote: > When wiping a block device with a GUID partition table, the header > backup might get left behind at the end of the disk. This commit also > wipes the last 4096 bytes of the disk, making sure that a GPT header > backup is erased, even from disks with 4k sector sizes. > > Signed-off-by: Filip Schauer <f.schauer@proxmox.com> > --- > src/tools/disks/mod.rs | 47 +++++++++++++++++++++++++----------------- > 1 file changed, 28 insertions(+), 19 deletions(-) > > diff --git a/src/tools/disks/mod.rs b/src/tools/disks/mod.rs > index 61aceccd..26e14603 100644 > --- a/src/tools/disks/mod.rs > +++ b/src/tools/disks/mod.rs > @@ -1176,29 +1176,38 @@ pub fn wipe_blockdev(disk: &Disk) -> Result<(), Error> { > let wipefs_output = proxmox_sys::command::run_command(wipefs_command, None)?; > info!("wipefs output: {wipefs_output}"); > > - let size = disk.size().map(|size| size / 1024 / 1024)?; > - let count = size.min(200); > - > - let mut dd_command = std::process::Command::new("dd"); > - let mut of_path = OsString::from("of="); > - of_path.push(disk_path); > - let mut count_str = OsString::from("count="); > - count_str.push(count.to_string()); > - let args = [ > - "if=/dev/zero".into(), > - of_path, > - "bs=1M".into(), > - "conv=fdatasync".into(), > - count_str, > - ]; > - dd_command.args(args); > - > - let dd_output = proxmox_sys::command::run_command(dd_command, None)?; > - info!("dd output: {dd_output}"); > + let dd_zero = |seek: Option<&str>, bs: &str, count: &str| -> Result<(), Error> { > + let mut dd_command = std::process::Command::new("dd"); > + let mut of_path = OsString::from("of="); > + of_path.push(disk_path); > + dd_command > + .arg("if=/dev/zero") > + .arg(of_path) > + .arg(format!("bs={bs}")) > + .arg("conv=fdatasync") > + .arg(format!("count={count}")); > + > + if let Some(seek) = seek { > + dd_command.arg(format!("seek={seek}")); > + } > + > + let dd_output = proxmox_sys::command::run_command(dd_command, None)?; > + info!("dd output: {dd_output}"); > + > + Ok(()) > + }; > + > + let size = disk.size()?; > + let count = (size / 1024 / 1024).min(200).to_string(); > + dd_zero(None, "1M", &count)?; > > if is_partition { > // set the partition type to 0x83 'Linux filesystem' > change_parttype(disk, "8300")?; > + } else { > + // Wipe the end of the disk to remove a potential GPT header backup > + let seek = (size / 512 - 8).to_string(); > + dd_zero(Some(&seek), "512", "8")?; > } > > Ok(()) > -- > 2.39.5 I don't know why we call out to dd for simple writes. I'd suggest just using `write_all_at` like so: ---8<--- diff --git a/src/tools/disks/mod.rs b/src/tools/disks/mod.rs index 571446db..e22691b7 100644 --- a/src/tools/disks/mod.rs +++ b/src/tools/disks/mod.rs @@ -4,11 +4,11 @@ use std::collections::{HashMap, HashSet}; use std::ffi::{OsStr, OsString}; use std::io; use std::os::unix::ffi::{OsStrExt, OsStringExt}; -use std::os::unix::fs::MetadataExt; +use std::os::unix::fs::{FileExt, MetadataExt, OpenOptionsExt}; use std::path::{Path, PathBuf}; use std::sync::{Arc, LazyLock}; -use anyhow::{bail, format_err, Error}; +use anyhow::{bail, format_err, Context as _, Error}; use libc::dev_t; use once_cell::sync::OnceCell; @@ -1176,25 +1176,7 @@ pub fn wipe_blockdev(disk: &Disk) -> Result<(), Error> { let wipefs_output = proxmox_sys::command::run_command(wipefs_command, None)?; info!("wipefs output: {wipefs_output}"); - let size = disk.size().map(|size| size / 1024 / 1024)?; - let count = size.min(200); - - let mut dd_command = std::process::Command::new("dd"); - let mut of_path = OsString::from("of="); - of_path.push(disk_path); - let mut count_str = OsString::from("count="); - count_str.push(count.to_string()); - let args = [ - "if=/dev/zero".into(), - of_path, - "bs=1M".into(), - "conv=fdatasync".into(), - count_str, - ]; - dd_command.args(args); - - let dd_output = proxmox_sys::command::run_command(dd_command, None)?; - info!("dd output: {dd_output}"); + zero_start_and_end(disk)?; if is_partition { // set the partition type to 0x83 'Linux filesystem' @@ -1204,6 +1186,29 @@ pub fn wipe_blockdev(disk: &Disk) -> Result<(), Error> { Ok(()) } +pub fn zero_start_and_end(disk: &Disk) -> Result<(), Error> { + let disk_path = match disk.device_path() { + Some(path) => path, + None => bail!("disk {:?} has no node in /dev", disk.syspath()), + }; + + let disk_size = disk.size()?; + let file = std::fs::OpenOptions::new() + .write(true) + .custom_flags(libc::O_CLOEXEC | libc::O_DSYNC) + .open(disk_path) + .with_context(|| "failed to open device {disk_path:?} for writing")?; + let write_size = disk_size.min(1024 * 1024); + let zeroes = proxmox_io::boxed::zeroed(write_size as usize); + file.write_all_at(&zeroes, 0) + .with_context(|| "failed to wipe start of device {disk_path:?}")?; + if disk_size > 4096 { + file.write_all_at(&zeroes[0..4096], disk_size - 4096) + .with_context(|| "failed to wipe end of device {disk_path:?}")?; + } + Ok(()) +} + pub fn change_parttype(part_disk: &Disk, part_type: &str) -> Result<(), Error> { let part_path = match part_disk.device_path() { Some(path) => path, _______________________________________________ pbs-devel mailing list pbs-devel@lists.proxmox.com https://lists.proxmox.com/cgi-bin/mailman/listinfo/pbs-devel