use std::convert::TryFrom; use std::convert::TryInto; use std::fs::{File, OpenOptions}; use std::os::unix::fs::OpenOptionsExt; use std::os::unix::io::AsRawFd; use std::path::Path; use std::time::SystemTime; use anyhow::{bail, format_err, Error}; use endian_trait::Endian; use nix::fcntl::{fcntl, FcntlArg, OFlag}; mod encryption; pub use encryption::*; mod volume_statistics; pub use volume_statistics::*; mod tape_alert_flags; pub use tape_alert_flags::*; mod mam; pub use mam::*; mod report_density; pub use report_density::*; use proxmox_io::{ReadExt, WriteExt}; use proxmox_sys::error::SysResult; use pbs_api_types::{Lp17VolumeStatistics, LtoDriveAndMediaStatus, MamAttribute}; use crate::{ sgutils2::{ alloc_page_aligned_buffer, scsi_inquiry, scsi_mode_sense, scsi_request_sense, InquiryInfo, ModeBlockDescriptor, ModeParameterHeader, ScsiError, SenseInfo, SgRaw, }, BlockRead, BlockReadError, BlockWrite, BlockedReader, BlockedWriter, }; #[repr(C, packed)] #[derive(Endian, Debug, Copy, Clone)] pub struct ReadPositionLongPage { flags: u8, reserved: [u8; 3], partition_number: u32, pub logical_object_number: u64, pub logical_file_id: u64, obsolete: [u8; 8], } #[repr(C, packed)] #[derive(Endian, Debug, Copy, Clone)] struct DataCompressionModePage { page_code: u8, // 0x0f page_length: u8, // 0x0e flags2: u8, flags3: u8, compression_algorithm: u32, decompression_algorithm: u32, reserved: [u8; 4], } impl DataCompressionModePage { pub fn set_compression(&mut self, enable: bool) { if enable { self.flags2 |= 128; } else { self.flags2 &= 127; } } pub fn compression_enabled(&self) -> bool { (self.flags2 & 0b1000_0000) != 0 } } #[repr(C, packed)] #[derive(Endian)] struct MediumConfigurationModePage { page_code: u8, // 0x1d page_length: u8, // 0x1e flags2: u8, reserved: [u8; 29], } impl MediumConfigurationModePage { pub fn is_worm(&self) -> bool { (self.flags2 & 1) == 1 } } #[derive(Debug)] pub struct LtoTapeStatus { pub block_length: u32, pub density_code: u8, pub buffer_mode: u8, pub write_protect: bool, pub compression: bool, } pub struct SgTape { file: File, locate_offset: Option, info: InquiryInfo, encryption_key_loaded: bool, } impl SgTape { const SCSI_TAPE_DEFAULT_TIMEOUT: usize = 60 * 10; // 10 minutes /// Create a new instance /// /// Uses scsi_inquiry to check the device type. pub fn new(mut file: File) -> Result { let info = scsi_inquiry(&mut file)?; if info.peripheral_type != 1 { bail!( "not a tape device (peripheral_type = {})", info.peripheral_type ); } Ok(Self { file, info, encryption_key_loaded: false, locate_offset: None, }) } /// Access to file descriptor - useful for testing pub fn file_mut(&mut self) -> &mut File { &mut self.file } pub fn info(&self) -> &InquiryInfo { &self.info } /// Return the maximum supported density code /// /// This can be used to detect the drive generation. pub fn max_density_code(&mut self) -> Result { report_density(&mut self.file) } pub fn open>(path: P) -> Result { // do not wait for media, use O_NONBLOCK let file = OpenOptions::new() .read(true) .write(true) .custom_flags(libc::O_NONBLOCK) .open(path)?; // then clear O_NONBLOCK let flags = fcntl(file.as_raw_fd(), FcntlArg::F_GETFL).into_io_result()?; let mut flags = OFlag::from_bits_truncate(flags); flags.remove(OFlag::O_NONBLOCK); fcntl(file.as_raw_fd(), FcntlArg::F_SETFL(flags)).into_io_result()?; Self::new(file) } pub fn inquiry(&mut self) -> Result { scsi_inquiry(&mut self.file) } /// Erase medium. /// /// EOD is written at the current position, which marks it as end /// of data. After the command is successfully completed, the /// drive is positioned immediately before End Of Data (not End Of /// Tape). pub fn erase_media(&mut self, fast: bool) -> Result<(), Error> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.push(0x19); if fast { cmd.push(0); // LONG=0 } else { cmd.push(1); // LONG=1 } cmd.extend(&[0, 0, 0, 0]); sg_raw .do_command(&cmd) .map_err(|err| format_err!("erase failed - {}", err))?; Ok(()) } /// Format media, single partition pub fn format_media(&mut self, fast: bool) -> Result<(), Error> { // try to get info about loaded media first let (has_format, is_worm) = match self.read_medium_configuration_page() { Ok((_head, block_descriptor, page)) => { // FORMAT requires LTO5 or newer let has_format = block_descriptor.density_code >= 0x58; let is_worm = page.is_worm(); (has_format, is_worm) } Err(_) => { // LTO3 and older do not supprt medium configuration mode page (false, false) } }; if is_worm { // We cannot FORMAT WORM media! Instead we check if its empty. self.move_to_eom(false)?; let pos = self.position()?; if pos.logical_object_number != 0 { bail!("format failed - detected WORM media with data."); } Ok(()) } else { self.rewind()?; let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); if has_format { cmd.extend(&[0x04, 0, 0, 0, 0, 0]); // FORMAT sg_raw.do_command(&cmd)?; if !fast { self.erase_media(false)?; // overwrite everything } } else { // try rewind/erase instead self.erase_media(fast)? } Ok(()) } } /// Lock/Unlock drive door pub fn set_medium_removal(&mut self, allow: bool) -> Result<(), ScsiError> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.extend(&[0x1E, 0, 0, 0]); if allow { cmd.push(0); } else { cmd.push(1); } cmd.push(0); // control sg_raw.do_command(&cmd)?; Ok(()) } pub fn rewind(&mut self) -> Result<(), Error> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.extend(&[0x01, 0, 0, 0, 0, 0]); // REWIND sg_raw .do_command(&cmd) .map_err(|err| format_err!("rewind failed - {}", err))?; Ok(()) } pub fn locate_file(&mut self, position: u64) -> Result<(), Error> { if position == 0 { return self.rewind(); } const SPACE_ONE_FILEMARK: &[u8] = &[0x11, 0x01, 0, 0, 1, 0]; // Special case for position 1, because LOCATE 0 does not work if position == 1 { self.rewind()?; let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); sg_raw .do_command(SPACE_ONE_FILEMARK) .map_err(|err| format_err!("locate file {} (space) failed - {}", position, err))?; return Ok(()); } let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); // Note: LOCATE(16) works for LTO4 or newer // // It seems the LOCATE command behaves slightly different across vendors // e.g. for IBM drives, LOCATE 1 moves to File #2, but // for HP drives, LOCATE 1 move to File #1 let fixed_position = if let Some(locate_offset) = self.locate_offset { if locate_offset < 0 { position.saturating_sub((-locate_offset) as u64) } else { position.saturating_add(locate_offset as u64) } } else { position }; // always sub(1), so that it works for IBM drives without locate_offset let fixed_position = fixed_position.saturating_sub(1); let mut cmd = Vec::new(); cmd.extend(&[0x92, 0b000_01_000, 0, 0]); // LOCATE(16) filemarks cmd.extend(&fixed_position.to_be_bytes()); cmd.extend(&[0, 0, 0, 0]); sg_raw .do_command(&cmd) .map_err(|err| format_err!("locate file {} failed - {}", position, err))?; // LOCATE always position at the BOT side of the filemark, so // we need to move to other side of filemark sg_raw .do_command(SPACE_ONE_FILEMARK) .map_err(|err| format_err!("locate file {} (space) failed - {}", position, err))?; if self.locate_offset.is_none() { // check if we landed at correct position let current_file = self.current_file_number()?; if current_file != position { let offset: i64 = i64::try_from((position as i128) - (current_file as i128)) .map_err(|err| { format_err!( "locate_file: offset between {} and {} invalid: {}", position, current_file, err ) })?; self.locate_offset = Some(offset); self.locate_file(position)?; let current_file = self.current_file_number()?; if current_file != position { bail!("locate_file: compensating offset did not work, aborting..."); } } else { self.locate_offset = Some(0); } } Ok(()) } pub fn position(&mut self) -> Result { let expected_size = std::mem::size_of::(); let mut sg_raw = SgRaw::new(&mut self.file, 32)?; sg_raw.set_timeout(30); // use short timeout let mut cmd = Vec::new(); // READ POSITION LONG FORM works on LTO4 or newer (with recent // firmware), although it is missing in the IBM LTO4 SSCI // reference manual. cmd.extend(&[0x34, 0x06, 0, 0, 0, 0, 0, 0, 0, 0]); // READ POSITION LONG FORM let data = sg_raw .do_command(&cmd) .map_err(|err| format_err!("read position failed - {}", err))?; let page = proxmox_lang::try_block!({ if data.len() != expected_size { bail!( "got unexpected data len ({} != {}", data.len(), expected_size ); } let mut reader = data; let page: ReadPositionLongPage = unsafe { reader.read_be_value()? }; Ok(page) }) .map_err(|err: Error| format_err!("decode position page failed - {}", err))?; if page.partition_number != 0 { bail!("detecthed partitioned tape - not supported"); } Ok(page) } pub fn current_file_number(&mut self) -> Result { let position = self.position()?; Ok(position.logical_file_id) } /// Check if we are positioned after a filemark (or BOT) pub fn check_filemark(&mut self) -> Result { let pos = self.position()?; if pos.logical_object_number == 0 { // at BOT, Ok (no filemark required) return Ok(true); } // Note: SPACE blocks returns Err at filemark match self.space(-1, true) { Ok(_) => { self.space(1, true) // move back to end .map_err(|err| { format_err!("check_filemark failed (space forward) - {}", err) })?; Ok(false) } Err(ScsiError::Sense(SenseInfo { sense_key: 0, asc: 0, ascq: 1, })) => { // Filemark detected - good self.space(1, false) // move to EOT side of filemark .map_err(|err| { format_err!( "check_filemark failed (move to EOT side of filemark) - {}", err ) })?; Ok(true) } Err(err) => { bail!("check_filemark failed - {:?}", err); } } } pub fn move_to_eom(&mut self, write_missing_eof: bool) -> Result<(), Error> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.extend(&[0x11, 0x03, 0, 0, 0, 0]); // SPACE(6) move to EOD sg_raw .do_command(&cmd) .map_err(|err| format_err!("move to EOD failed - {}", err))?; if write_missing_eof && !self.check_filemark()? { self.write_filemarks(1, false)?; } Ok(()) } fn space(&mut self, count: isize, blocks: bool) -> Result<(), ScsiError> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); // Use short command if possible (supported by all drives) if (count <= 0x7fffff) && (count > -0x7fffff) { cmd.push(0x11); // SPACE(6) if blocks { cmd.push(0); // blocks } else { cmd.push(1); // filemarks } cmd.push(((count >> 16) & 0xff) as u8); cmd.push(((count >> 8) & 0xff) as u8); cmd.push((count & 0xff) as u8); cmd.push(0); //control byte } else { cmd.push(0x91); // SPACE(16) if blocks { cmd.push(0); // blocks } else { cmd.push(1); // filemarks } cmd.extend(&[0, 0]); // reserved let count: i64 = count as i64; cmd.extend(&count.to_be_bytes()); cmd.extend(&[0, 0, 0, 0]); // reserved } sg_raw.do_command(&cmd)?; Ok(()) } pub fn space_filemarks(&mut self, count: isize) -> Result<(), Error> { self.space(count, false) .map_err(|err| format_err!("space filemarks failed - {}", err)) } pub fn space_blocks(&mut self, count: isize) -> Result<(), Error> { self.space(count, true) .map_err(|err| format_err!("space blocks failed - {}", err)) } pub fn eject(&mut self) -> Result<(), Error> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.extend(&[0x1B, 0, 0, 0, 0, 0]); // LODA/UNLOAD HOLD=0, LOAD=0 sg_raw .do_command(&cmd) .map_err(|err| format_err!("eject failed - {}", err))?; Ok(()) } pub fn load(&mut self) -> Result<(), Error> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.extend(&[0x1B, 0, 0, 0, 0b0000_0001, 0]); // LODA/UNLOAD HOLD=0, LOAD=1 sg_raw .do_command(&cmd) .map_err(|err| format_err!("load media failed - {}", err))?; Ok(()) } pub fn write_filemarks(&mut self, count: usize, immediate: bool) -> Result<(), std::io::Error> { if count > 255 { proxmox_lang::io_bail!("write_filemarks failed: got strange count '{}'", count); } let mut sg_raw = SgRaw::new(&mut self.file, 16).map_err(|err| { proxmox_lang::io_format_err!("write_filemarks failed (alloc) - {}", err) })?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.push(0x10); if immediate { cmd.push(1); // IMMED=1 } else { cmd.push(0); // IMMED=0 } cmd.extend(&[0, 0, count as u8]); // COUNT cmd.push(0); // control byte match sg_raw.do_command(&cmd) { Ok(_) => { /* OK */ } Err(ScsiError::Sense(SenseInfo { sense_key: 0, asc: 0, ascq: 2, })) => { /* LEOM - ignore */ } Err(err) => { proxmox_lang::io_bail!("write filemark failed - {}", err); } } Ok(()) } // Flush tape buffers (WEOF with count 0 => flush) pub fn sync(&mut self) -> Result<(), std::io::Error> { self.write_filemarks(0, false)?; Ok(()) } pub fn test_unit_ready(&mut self) -> Result<(), Error> { let mut sg_raw = SgRaw::new(&mut self.file, 16)?; sg_raw.set_timeout(30); // use short timeout let mut cmd = Vec::new(); cmd.extend(&[0x00, 0, 0, 0, 0, 0]); // TEST UNIT READY match sg_raw.do_command(&cmd) { Ok(_) => Ok(()), Err(err) => { bail!("test_unit_ready failed - {}", err); } } } pub fn wait_until_ready(&mut self) -> Result<(), Error> { let start = SystemTime::now(); let max_wait = std::time::Duration::new(Self::SCSI_TAPE_DEFAULT_TIMEOUT as u64, 0); loop { match self.test_unit_ready() { Ok(()) => return Ok(()), _ => { std::thread::sleep(std::time::Duration::new(1, 0)); if start.elapsed()? > max_wait { bail!("wait_until_ready failed - got timeout"); } } } } } /// Read Tape Alert Flags pub fn tape_alert_flags(&mut self) -> Result { read_tape_alert_flags(&mut self.file) } /// Read Cartridge Memory (MAM Attributes) pub fn cartridge_memory(&mut self) -> Result, Error> { read_mam_attributes(&mut self.file) } /// Read Volume Statistics pub fn volume_statistics(&mut self) -> Result { read_volume_statistics(&mut self.file) } pub fn set_encryption(&mut self, key: Option<[u8; 32]>) -> Result<(), Error> { self.encryption_key_loaded = key.is_some(); set_encryption(&mut self.file, key) } // Note: use alloc_page_aligned_buffer to alloc data transfer buffer // // Returns true if the drive reached the Logical End Of Media (early warning) fn write_block(&mut self, data: &[u8]) -> Result { let transfer_len = data.len(); if transfer_len > 0x800000 { proxmox_lang::io_bail!("write failed - data too large"); } let mut sg_raw = SgRaw::new(&mut self.file, 0).unwrap(); // cannot fail with size 0 sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.push(0x0A); // WRITE cmd.push(0x00); // VARIABLE SIZED BLOCKS cmd.push(((transfer_len >> 16) & 0xff) as u8); cmd.push(((transfer_len >> 8) & 0xff) as u8); cmd.push((transfer_len & 0xff) as u8); cmd.push(0); // control byte //println!("WRITE {:?}", cmd); //println!("WRITE {:?}", data); match sg_raw.do_out_command(&cmd, data) { Ok(()) => Ok(false), Err(ScsiError::Sense(SenseInfo { sense_key: 0, asc: 0, ascq: 2, })) => { Ok(true) // LEOM } Err(err) => { proxmox_lang::io_bail!("write failed - {}", err); } } } fn read_block(&mut self, buffer: &mut [u8]) -> Result { let transfer_len = buffer.len(); if transfer_len > 0xFFFFFF { return Err(BlockReadError::Error(proxmox_lang::io_format_err!( "read failed - buffer too large" ))); } let mut sg_raw = SgRaw::new(&mut self.file, 0).unwrap(); // cannot fail with size 0 sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); let mut cmd = Vec::new(); cmd.push(0x08); // READ cmd.push(0x02); // VARIABLE SIZED BLOCKS, SILI=1 //cmd.push(0x00); // VARIABLE SIZED BLOCKS, SILI=0 cmd.push(((transfer_len >> 16) & 0xff) as u8); cmd.push(((transfer_len >> 8) & 0xff) as u8); cmd.push((transfer_len & 0xff) as u8); cmd.push(0); // control byte let data = match sg_raw.do_in_command(&cmd, buffer) { Ok(data) => data, Err(ScsiError::Sense(SenseInfo { sense_key: 0, asc: 0, ascq: 1, })) => { return Err(BlockReadError::EndOfFile); } Err(ScsiError::Sense(SenseInfo { sense_key: 8, asc: 0, ascq: 5, })) => { return Err(BlockReadError::EndOfStream); } Err(err) => { return Err(BlockReadError::Error(proxmox_lang::io_format_err!( "read failed - {}", err ))); } }; if data.len() != transfer_len { return Err(BlockReadError::Error(proxmox_lang::io_format_err!( "read failed - unexpected block len ({} != {})", data.len(), buffer.len() ))); } Ok(transfer_len) } pub fn open_writer(&mut self) -> BlockedWriter { let writer = SgTapeWriter::new(self); BlockedWriter::new(writer) } pub fn open_reader(&mut self) -> Result, BlockReadError> { let reader = SgTapeReader::new(self); BlockedReader::open(reader) } /// Set all options we need/want pub fn set_default_options(&mut self) -> Result<(), Error> { let compression = Some(true); let block_length = Some(0); // variable length mode let buffer_mode = Some(true); // Always use drive buffer self.set_drive_options(compression, block_length, buffer_mode)?; Ok(()) } /// Set important drive options pub fn set_drive_options( &mut self, compression: Option, block_length: Option, buffer_mode: Option, ) -> Result<(), Error> { // Note: Read/Modify/Write let (mut head, mut block_descriptor, mut page) = self.read_compression_page()?; let mut sg_raw = SgRaw::new(&mut self.file, 0)?; sg_raw.set_timeout(Self::SCSI_TAPE_DEFAULT_TIMEOUT); head.mode_data_len = 0; // need to b e zero if let Some(compression) = compression { page.set_compression(compression); } if let Some(block_length) = block_length { block_descriptor.set_block_length(block_length)?; } if let Some(buffer_mode) = buffer_mode { head.set_buffer_mode(buffer_mode); } let mut data = Vec::new(); unsafe { data.write_be_value(head)?; data.write_be_value(block_descriptor)?; data.write_be_value(page)?; } let mut cmd = Vec::new(); cmd.push(0x55); // MODE SELECT(10) cmd.push(0b0001_0000); // PF=1 cmd.extend(&[0, 0, 0, 0, 0]); //reserved let param_list_len: u16 = data.len() as u16; cmd.extend(¶m_list_len.to_be_bytes()); cmd.push(0); // control let mut buffer = alloc_page_aligned_buffer(4096)?; buffer[..data.len()].copy_from_slice(&data[..]); sg_raw .do_out_command(&cmd, &buffer[..data.len()]) .map_err(|err| format_err!("set drive options failed - {}", err))?; Ok(()) } fn read_medium_configuration_page( &mut self, ) -> Result< ( ModeParameterHeader, ModeBlockDescriptor, MediumConfigurationModePage, ), Error, > { let (head, block_descriptor, page): (_, _, MediumConfigurationModePage) = scsi_mode_sense(&mut self.file, false, 0x1d, 0)?; proxmox_lang::try_block!({ if (page.page_code & 0b0011_1111) != 0x1d { bail!("wrong page code {}", page.page_code); } if page.page_length != 0x1e { bail!("wrong page length {}", page.page_length); } let block_descriptor = match block_descriptor { Some(block_descriptor) => block_descriptor, None => bail!("missing block descriptor"), }; Ok((head, block_descriptor, page)) }) .map_err(|err| format_err!("read_medium_configuration failed - {}", err)) } fn read_compression_page( &mut self, ) -> Result< ( ModeParameterHeader, ModeBlockDescriptor, DataCompressionModePage, ), Error, > { let (head, block_descriptor, page): (_, _, DataCompressionModePage) = scsi_mode_sense(&mut self.file, false, 0x0f, 0)?; proxmox_lang::try_block!({ if (page.page_code & 0b0011_1111) != 0x0f { bail!("wrong page code {}", page.page_code); } if page.page_length != 0x0e { bail!("wrong page length {}", page.page_length); } let block_descriptor = match block_descriptor { Some(block_descriptor) => block_descriptor, None => bail!("missing block descriptor"), }; Ok((head, block_descriptor, page)) }) .map_err(|err| format_err!("read_compression_page failed: {}", err)) } /// Read drive options/status /// /// We read the drive compression page, including the /// block_descriptor. This is all information we need for now. pub fn read_drive_status(&mut self) -> Result { // We do a Request Sense, but ignore the result. // This clears deferred error or media changed events. let _ = scsi_request_sense(&mut self.file); let (head, block_descriptor, page) = self.read_compression_page()?; Ok(LtoTapeStatus { block_length: block_descriptor.block_length(), write_protect: head.write_protect(), buffer_mode: head.buffer_mode(), compression: page.compression_enabled(), density_code: block_descriptor.density_code, }) } /// Get Tape and Media status pub fn get_drive_and_media_status(&mut self) -> Result { let drive_status = self.read_drive_status()?; let alert_flags = self .tape_alert_flags() .map(|flags| format!("{:?}", flags)) .ok(); let mut status = LtoDriveAndMediaStatus { vendor: self.info().vendor.clone(), product: self.info().product.clone(), revision: self.info().revision.clone(), blocksize: drive_status.block_length, compression: drive_status.compression, buffer_mode: drive_status.buffer_mode, density: drive_status.density_code.try_into()?, alert_flags, write_protect: None, file_number: None, block_number: None, manufactured: None, bytes_read: None, bytes_written: None, medium_passes: None, medium_wearout: None, volume_mounts: None, }; if self.test_unit_ready().is_ok() { if drive_status.write_protect { status.write_protect = Some(drive_status.write_protect); } let position = self.position()?; status.file_number = Some(position.logical_file_id); status.block_number = Some(position.logical_object_number); if let Ok(mam) = self.cartridge_memory() { let usage = mam_extract_media_usage(&mam)?; status.manufactured = Some(usage.manufactured); status.bytes_read = Some(usage.bytes_read); status.bytes_written = Some(usage.bytes_written); if let Ok(volume_stats) = self.volume_statistics() { let passes = std::cmp::max( volume_stats.beginning_of_medium_passes, volume_stats.middle_of_tape_passes, ); // assume max. 16000 medium passes // see: https://en.wikipedia.org/wiki/Linear_Tape-Open let wearout: f64 = (passes as f64) / 16000.0_f64; status.medium_passes = Some(passes); status.medium_wearout = Some(wearout); status.volume_mounts = Some(volume_stats.volume_mounts); } } } Ok(status) } } impl Drop for SgTape { fn drop(&mut self) { // For security reasons, clear the encryption key if self.encryption_key_loaded { let _ = self.set_encryption(None); } } } pub struct SgTapeReader<'a> { sg_tape: &'a mut SgTape, end_of_file: bool, } impl<'a> SgTapeReader<'a> { pub fn new(sg_tape: &'a mut SgTape) -> Self { Self { sg_tape, end_of_file: false, } } } impl<'a> BlockRead for SgTapeReader<'a> { fn read_block(&mut self, buffer: &mut [u8]) -> Result { if self.end_of_file { return Err(BlockReadError::Error(proxmox_lang::io_format_err!( "detected read after EOF!" ))); } match self.sg_tape.read_block(buffer) { Ok(usize) => Ok(usize), Err(BlockReadError::EndOfFile) => { self.end_of_file = true; Err(BlockReadError::EndOfFile) } Err(err) => Err(err), } } } pub struct SgTapeWriter<'a> { sg_tape: &'a mut SgTape, _leom_sent: bool, } impl<'a> SgTapeWriter<'a> { pub fn new(sg_tape: &'a mut SgTape) -> Self { Self { sg_tape, _leom_sent: false, } } } impl<'a> BlockWrite for SgTapeWriter<'a> { fn write_block(&mut self, buffer: &[u8]) -> Result { self.sg_tape.write_block(buffer) } fn write_filemark(&mut self) -> Result<(), std::io::Error> { self.sg_tape.write_filemarks(1, true) } }