1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! x86 definitions of non-translated MSI address and data.

use crate::apic::APIC_BASE_ADDRESS;
use bitfield_struct::bitfield;

/// The layout of the MSI address element.
#[bitfield(u32)]
pub struct MsiAddress {
    #[bits(2)]
    _reserved: u32,
    pub destination_mode_logical: bool,
    pub redirection_hint: bool,
    pub extended_destination: u8,
    pub destination: u8,
    #[bits(12)]
    pub address: u16,
}

/// The expected value for MsiAddress::address.
pub const MSI_ADDRESS: u16 = (APIC_BASE_ADDRESS >> 20) as u16;

impl MsiAddress {
    /// Returns a 15-bit destination encoded in the MSI address. This is not
    /// architectural--normally only an 8-bit destination is supported unless
    /// interrupt redirection is enabled--but this is supported by some
    /// virtualization platforms (including Hyper-V and KVM).
    ///
    /// The high 7 bits are encoded as the high 7 bits of the extended
    /// destination field. The low bit of that field is ignored and presumed to
    /// be zero in this configuration.
    pub fn virt_destination(&self) -> u16 {
        self.destination() as u16 | ((self.extended_destination() as u16 & !1) << 7)
    }

    /// Returns a value with a 15-bit destination encoded as guests expect when
    /// running with Hyper-V or KVM virtualization extensions.
    ///
    /// This updates the destination and extended destination fields.
    pub fn with_virt_destination(self, destination: u16) -> Self {
        self.with_destination(destination as u8)
            .with_extended_destination((destination >> 7) as u8 & !1)
    }

    /// Updates the value with a 15-bit destination encoded as guests expect
    /// when running with Hyper-V or KVM virtualization extensions.
    ///
    /// This updates the destination and extended destination fields.
    pub fn set_virt_destination(&mut self, destination: u16) {
        *self = self.with_virt_destination(destination);
    }
}

/// The layout of the MSI data element.
///
/// Note that the significant bits correspond to low bits of
/// [`Icr`](super::apic::Icr).
#[bitfield(u32)]
pub struct MsiData {
    pub vector: u8,
    #[bits(3)]
    pub delivery_mode: u8,
    pub destination_mode_logical: bool,
    #[bits(2)]
    _reserved: u8,
    pub assert: bool,
    pub trigger_mode_level: bool,
    _reserved: u16,
}