5
0
mirror of git://git.proxmox.com/git/proxmox-backup.git synced 2025-01-10 01:18:06 +03:00

tape: assert encryption mode when using the PoolWriter

by introducing an 'assert_encryption_mode' that checks the desired
state, and bails out if it's different, called directly where we
previously set the encryption mode (which is now done automatically)

Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
 [ TL: add drive_ prefix and fleece in comment ]
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
This commit is contained in:
Dominik Csapak 2024-01-22 12:50:34 +01:00 committed by Thomas Lamprecht
parent 1343dcaf01
commit 3579d724a3
5 changed files with 44 additions and 2 deletions

View File

@ -9,7 +9,7 @@ use endian_trait::Endian;
use nix::fcntl::{fcntl, FcntlArg, OFlag}; use nix::fcntl::{fcntl, FcntlArg, OFlag};
mod encryption; mod encryption;
pub use encryption::{drive_set_encryption, has_encryption}; pub use encryption::{drive_set_encryption, drive_get_encryption, has_encryption};
mod volume_statistics; mod volume_statistics;
use proxmox_uuid::Uuid; use proxmox_uuid::Uuid;

View File

@ -57,6 +57,27 @@ pub fn drive_set_encryption<F: AsRawFd>(file: &mut F, key: Option<[u8; 32]>) ->
bail!("got unexpected encryption mode {:?}", status.mode); bail!("got unexpected encryption mode {:?}", status.mode);
} }
/// Returns if encryption is enabled on the drive
pub fn drive_get_encryption<F: AsRawFd>(file: &mut F) -> Result<bool, Error> {
let data = match sg_spin_data_encryption_status(file) {
Ok(data) => data,
Err(_) => {
// Assume device does not support HW encryption
return Ok(false);
}
};
let status = decode_spin_data_encryption_status(&data)?;
match status.mode {
// these three below have all encryption enabled, and only differ in how decryption is
// handled
DataEncryptionMode::On => Ok(true),
DataEncryptionMode::Mixed => Ok(true),
DataEncryptionMode::RawRead => Ok(true),
// currently, the mode below is the only one that has encryption actually disabled
DataEncryptionMode::Off => Ok(false),
}
}
#[derive(Endian)] #[derive(Endian)]
#[repr(C, packed)] #[repr(C, packed)]
struct SspSetDataEncryptionPage { struct SspSetDataEncryptionPage {
@ -187,7 +208,7 @@ fn sg_spin_data_encryption_caps<F: AsRawFd>(file: &mut F) -> Result<Vec<u8>, Err
.map(|v| v.to_vec()) .map(|v| v.to_vec())
} }
#[derive(Debug)] #[derive(Debug, PartialEq, Eq)]
enum DataEncryptionMode { enum DataEncryptionMode {
On, On,
Mixed, Mixed,

View File

@ -16,6 +16,7 @@ use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
use anyhow::{bail, format_err, Error}; use anyhow::{bail, format_err, Error};
use pbs_tape::sg_tape::drive_get_encryption;
use proxmox_uuid::Uuid; use proxmox_uuid::Uuid;
use pbs_api_types::{ use pbs_api_types::{
@ -311,6 +312,14 @@ impl TapeDriver for LtoTapeHandle {
self.sg_tape.set_encryption(None) self.sg_tape.set_encryption(None)
} }
} }
fn assert_encryption_mode(&mut self, encryption_wanted: bool) -> Result<(), Error> {
let encryption_set = drive_get_encryption(self.sg_tape.file_mut())?;
if encryption_wanted != encryption_set {
bail!("Set encryption mode not what was desired (set: {encryption_set}, wanted: {encryption_wanted})");
}
Ok(())
}
} }
fn run_sg_tape_cmd(subcmd: &str, args: &[&str], fd: RawFd) -> Result<String, Error> { fn run_sg_tape_cmd(subcmd: &str, args: &[&str], fd: RawFd) -> Result<String, Error> {

View File

@ -224,6 +224,14 @@ pub trait TapeDriver {
} }
Ok(()) Ok(())
} }
/// Asserts that the encryption mode is set to the given value
fn assert_encryption_mode(&mut self, encryption_wanted: bool) -> Result<(), Error> {
if encryption_wanted {
bail!("drive does not support encryption");
}
Ok(())
}
} }
/// A boxed implementor of [`MediaChange`]. /// A boxed implementor of [`MediaChange`].

View File

@ -270,6 +270,10 @@ impl PoolWriter {
self.catalog_set.lock().unwrap().append_catalog(catalog)?; self.catalog_set.lock().unwrap().append_catalog(catalog)?;
let media_set = media.media_set_label().unwrap();
drive.assert_encryption_mode(media_set.encryption_key_fingerprint.is_some())?;
self.status = Some(PoolWriterState { self.status = Some(PoolWriterState {
drive, drive,
media_uuid: media_uuid.clone(), media_uuid: media_uuid.clone(),