Struct AdminQueueFaultConfig

Source
pub struct AdminQueueFaultConfig {
    pub admin_submission_queue_faults: Vec<(CommandMatch, AdminQueueFaultBehavior<Command>)>,
    pub admin_completion_queue_faults: Vec<(CommandMatch, AdminQueueFaultBehavior<Completion>)>,
}
Expand description

A fault configuration to inject faults into the admin submission and completion queues.

This struct maintains a mapping from CommandMatch to AdminQueueFaultBehavior for submission and completion queues. When a command match is found, (and fault_active == true) the associated fault is applied. Both submission and completion queue faults match on commands because completions do not contain enough identifying information to match against. If there is more than one match for a given command, the match defined first is prioritized. Faults are added via the with_submission_queue_fault and with_completion_queue_fault methods and can be chained. AdminQueueFaultConfig::new() creates an empty fault.

§Panics

Panics if a duplicate CommandMatch is added for either submission or completion queues

§Example

Panic on CREATE_IO_COMPLETION_QUEUE and delay before sending completion for 500ms after GET_LOG_PAGE command is processed.

use mesh::CellUpdater;
use nvme_resources::fault::AdminQueueFaultConfig;
use nvme_resources::fault::CommandMatch;
use nvme_resources::fault::FaultConfiguration;
use nvme_resources::fault::AdminQueueFaultBehavior;
use nvme_spec::Command;
use std::time::Duration;
use zerocopy::FromZeros;
use zerocopy::IntoBytes;

pub fn build_admin_queue_fault() -> FaultConfiguration {
    let mut fault_start_updater = CellUpdater::new(false);

    // Setup command matches
    let mut command_io_queue = Command::new_zeroed();
    let mut command_log_page = Command::new_zeroed();
    let mut mask = Command::new_zeroed();

    command_io_queue.cdw0 = command_io_queue.cdw0.with_opcode(nvme_spec::AdminOpcode::CREATE_IO_COMPLETION_QUEUE.0);
    command_log_page.cdw0 = command_log_page.cdw0.with_opcode(nvme_spec::AdminOpcode::GET_LOG_PAGE.0);
    mask.cdw0 = mask.cdw0.with_opcode(u8::MAX);

    return FaultConfiguration::new(fault_start_updater.cell())
        .with_admin_queue_fault(
            AdminQueueFaultConfig::new().with_submission_queue_fault(
                CommandMatch {
                    command: command_io_queue,
                    mask: mask.as_bytes().try_into().expect("mask should be 64 bytes"),
                },
                AdminQueueFaultBehavior::Panic("Received a CREATE_IO_COMPLETION_QUEUE command".to_string()),
            ).with_completion_queue_fault(
                CommandMatch {
                    command: command_log_page,
                    mask: mask.as_bytes().try_into().expect("mask should be 64 bytes"),
                },
                AdminQueueFaultBehavior::Delay(Duration::from_millis(500)),
            )
        );
}

Fields§

§admin_submission_queue_faults: Vec<(CommandMatch, AdminQueueFaultBehavior<Command>)>

A map of NVME opcodes to the submission fault behavior for each. (This would ideally be a HashMap, but mesh doesn’t support that type. Given that this is not performance sensitive, the lookup is okay)

§admin_completion_queue_faults: Vec<(CommandMatch, AdminQueueFaultBehavior<Completion>)>

A map of NVME opcodes to the completion fault behavior for each.

Implementations§

Source§

impl AdminQueueFaultConfig

Source

pub fn new() -> Self

Create an empty fault configuration

Source

pub fn with_submission_queue_fault( self, pattern: CommandMatch, behaviour: AdminQueueFaultBehavior<Command>, ) -> Self

Add a CommandMatch -> AdminQueueFaultBehavior mapping for the submission queue.

§Panics

Panics if an identical CommandMatch has already been configured.

Source

pub fn with_completion_queue_fault( self, pattern: CommandMatch, behaviour: AdminQueueFaultBehavior<Completion>, ) -> Self

Add a CommandMatch -> AdminQueueFaultBehavior mapping for the completion queue.

§Panics

Panics if an identical CommandMatch has already been configured.

Trait Implementations§

Source§

impl DefaultEncoding for AdminQueueFaultConfig

Source§

type Encoding = TableEncoder

The encoding to use for the serialization. Read more
Source§

impl<'encoding> StructDecodeMetadata<'encoding, Resource> for AdminQueueFaultConfig

Source§

const DECODERS: &'static [ErasedDecoderEntry]

The list of decoder vtables.
Source§

impl StructEncodeMetadata<Resource> for AdminQueueFaultConfig

Source§

const ENCODERS: &'static [ErasedEncoderEntry]

The list of encoder vtables.
Source§

impl StructMetadata for AdminQueueFaultConfig

Source§

const NUMBERS: &'static [u32]

The field numbers for each field.
Source§

const OFFSETS: &'static [usize]

The byte offset to each field within the struct.

Auto Trait Implementations§

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

§

impl<T> Instrument for T

§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided [Span], returning an Instrumented wrapper. Read more
§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

§

impl<T> MeshField for T
where T: DefaultEncoding, <T as DefaultEncoding>::Encoding: FieldEncode<T, Resource> + for<'a> FieldDecode<'a, T, Resource> + Send + Sync,

§

type Encoding = <T as DefaultEncoding>::Encoding

§

impl<T> MeshPayload for T
where T: DefaultEncoding + Any + Send + 'static, <T as DefaultEncoding>::Encoding: MessageEncode<T, Resource> + for<'a> MessageDecode<'a, T, Resource> + FieldEncode<T, Resource> + for<'a> FieldDecode<'a, T, Resource> + Send + Sync,

§

type Encoding = <T as DefaultEncoding>::Encoding

§

impl<T> SerializeMessage for T
where T: 'static + MeshPayload + Send,

§

type Concrete = T

The underlying concrete message type.
§

fn compute_message_size(&mut self, sizer: MessageSizer<'_>)

Computes the message size, as in [MessageEncode::compute_message_size].
§

fn write_message(self, writer: MessageWriter<'_, '_, Resource>)

Writes the message, as in [MessageEncode::write_message].
§

fn extract(self) -> <T as SerializeMessage>::Concrete

Extract the concrete message.
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
§

impl<T> WithSubscriber for T

§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a [WithDispatch] wrapper. Read more