1use crate::util::is_set;
5use bitfield::bitfield;
6use fidl_fuchsia_hardware_pci::Capability as FidlCapability;
7use std::fmt;
8use zerocopy::{FromBytes, Immutable, IntoBytes, KnownLayout, Ref};
9
10enum CapabilityType {
12 Null,
13 PciPowerManagement,
14 Agp,
15 VitalProductData,
16 SlotIdentification,
17 Msi,
18 CompactPciHotSwap,
19 PciX,
20 HyperTransport,
21 Vendor,
22 DebugPort,
23 CompactPciCrc,
24 PciHotplug,
25 PciBridgeSubsystemVendorId,
26 Agp8x,
27 SecureDevice,
28 PciExpress,
29 MsiX,
30 SataDataNdxCfg,
31 AdvancedFeatures,
32 EnhancedAllocation,
33 FlatteningPortalBridge,
34 Unknown(u8),
35}
36
37impl From<u8> for CapabilityType {
38 fn from(value: u8) -> Self {
39 match value {
40 0x00 => CapabilityType::Null,
41 0x01 => CapabilityType::PciPowerManagement,
42 0x02 => CapabilityType::Agp,
43 0x03 => CapabilityType::VitalProductData,
44 0x04 => CapabilityType::SlotIdentification,
45 0x05 => CapabilityType::Msi,
46 0x06 => CapabilityType::CompactPciHotSwap,
47 0x07 => CapabilityType::PciX,
48 0x08 => CapabilityType::HyperTransport,
49 0x09 => CapabilityType::Vendor,
50 0x0a => CapabilityType::DebugPort,
51 0x0b => CapabilityType::CompactPciCrc,
52 0x0c => CapabilityType::PciHotplug,
53 0x0d => CapabilityType::PciBridgeSubsystemVendorId,
54 0x0e => CapabilityType::Agp8x,
55 0x0f => CapabilityType::SecureDevice,
56 0x10 => CapabilityType::PciExpress,
57 0x11 => CapabilityType::MsiX,
58 0x12 => CapabilityType::SataDataNdxCfg,
59 0x13 => CapabilityType::AdvancedFeatures,
60 0x14 => CapabilityType::EnhancedAllocation,
61 0x15 => CapabilityType::FlatteningPortalBridge,
62 _ => CapabilityType::Unknown(value),
63 }
64 }
65}
66
67pub struct Capability<'a> {
68 offset: usize,
69 config: &'a [u8],
70 cap_type: CapabilityType,
71}
72
73impl<'a> fmt::Display for Capability<'a> {
74 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
75 write!(f, "Capabilities: [{:#2x}] ", self.offset)?;
76 match self.cap_type {
77 CapabilityType::Null => write!(f, "Null"),
78 CapabilityType::PciPowerManagement => write!(f, "PCI Power Management"),
79 CapabilityType::Agp => write!(f, "AGP"),
80 CapabilityType::VitalProductData => write!(f, "Vital Product Data"),
81 CapabilityType::SlotIdentification => write!(f, "Slot Identification"),
82 CapabilityType::Msi => self.msi(f),
83 CapabilityType::CompactPciHotSwap => write!(f, "CompactPCI Hotswap"),
84 CapabilityType::PciX => write!(f, "PCI-X"),
85 CapabilityType::HyperTransport => write!(f, "HyperTransport"),
86 CapabilityType::Vendor => self.vendor(f),
87 CapabilityType::DebugPort => write!(f, "Debug Port"),
88 CapabilityType::CompactPciCrc => write!(f, "CompactPCI CRC"),
89 CapabilityType::PciHotplug => write!(f, "PCI Hotplug"),
90 CapabilityType::PciBridgeSubsystemVendorId => write!(f, "PCI Bridge Subsystem VID"),
91 CapabilityType::Agp8x => write!(f, "AGP 8x"),
92 CapabilityType::SecureDevice => write!(f, "Secure Device"),
93 CapabilityType::PciExpress => self.pci_express(f),
94 CapabilityType::MsiX => self.msi_x(f),
95 CapabilityType::SataDataNdxCfg => write!(f, "SATA Data Ndx Config"),
96 CapabilityType::AdvancedFeatures => write!(f, "Advanced Features"),
97 CapabilityType::EnhancedAllocation => write!(f, "Enhanced Allocations"),
98 CapabilityType::FlatteningPortalBridge => write!(f, "Flattening Portal Bridge"),
99 CapabilityType::Unknown(id) => write!(f, "Unknown Capability (id = {:#2x})", id),
100 }
101 }
102}
103
104impl<'a> Capability<'a> {
105 pub fn new(capability: &'a FidlCapability, config: &'a [u8]) -> Self {
106 Capability {
107 offset: capability.offset as usize,
108 config,
109 cap_type: CapabilityType::from(capability.id),
110 }
111 }
112
113 fn msi(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
114 let control = MsiControl(
117 ((self.config[self.offset + 3] as u16) << 8) | self.config[self.offset + 2] as u16,
118 );
119 write!(
120 f,
121 "MSI: Enable{} Count={}/{} Maskable{} 64bit{}\n",
122 is_set(control.enable()),
123 msi_mms_to_value(control.mms_enabled()),
124 msi_mms_to_value(control.mms_capable()),
125 is_set(control.pvm_capable()),
126 is_set(control.can_be_64bit())
127 )?;
128
129 if control.can_be_64bit() {
130 let (msi, _) = Ref::<_, Msi64Capability>::from_prefix(
131 &self.config[self.offset..self.config.len()],
132 )
133 .unwrap();
134 write!(
135 f,
136 "\t\tAddress: {:#010x} {:#08x} Data: {:#06x}",
137 { msi.address_upper },
138 { msi.address },
139 { msi.data }
140 )
141 } else {
142 let (msi, _) = Ref::<_, Msi32Capability>::from_prefix(
143 &self.config[self.offset..self.config.len()],
144 )
145 .unwrap();
146 write!(f, "\t\tAddress: {:#010x} Data: {:#06x}", { msi.address }, { msi.data })
147 }
148 }
149
150 fn msi_x(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
151 let (msix, _) =
152 Ref::<_, MsixCapability>::from_prefix(&self.config[self.offset..self.config.len()])
153 .unwrap();
154 let control = MsixControl(msix.control);
155 let table = MsixBarField(msix.table);
156 let pba = MsixBarField(msix.pba);
157 write!(
158 f,
159 "MSI-X: Enable{} Count={} Masked{} TBIR={} TOff={:#x} PBIR={} POff={:#x}",
160 is_set(control.enable()),
161 control.table_size() + 1,
162 is_set(control.function_mask()),
163 table.bir(),
164 table.offset() << 3,
165 pba.bir(),
166 pba.offset() << 3,
167 )
168 }
169
170 fn pci_express(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
171 let (pcie, _) = Ref::<_, PciExpressCapability>::from_prefix(
172 &self.config[self.offset..self.config.len()],
173 )
174 .unwrap();
175 let pcie_capabilities = PcieCapabilitiesField(pcie.pcie_capabilities);
177 let dev_type = PcieDevicePortType::from(pcie_capabilities.device_type());
178 let slot = match dev_type {
179 PcieDevicePortType::PCIEEP
180 | PcieDevicePortType::LPCIEEP
181 | PcieDevicePortType::RCIEP
182 | PcieDevicePortType::RCEC => String::from(""),
183 _ => format!(" (Slot{})", is_set(pcie_capabilities.slot_implemented())),
184 };
185 write!(
186 f,
187 "Express (v{}) {}{}, MSI {:#02x}",
188 pcie_capabilities.version(),
189 dev_type,
190 slot,
191 pcie_capabilities.irq_number()
192 )
193 }
194
195 fn vendor(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
196 write!(f, "Vendor Specific Information: Len={:#2x}", self.config[self.offset + 2])
197 }
198}
199
200fn msi_mms_to_value(mms: u16) -> u8 {
201 match mms {
202 0x000 => 1,
203 0x001 => 2,
204 0x010 => 4,
205 0x011 => 8,
206 0x100 => 16,
207 0x101 => 32,
208 _ => 0,
209 }
210}
211
212bitfield! {
213 struct MsiControl(u16);
214 enable, _: 0;
215 mms_capable, _: 3, 1;
216 mms_enabled, _: 6, 4;
217 can_be_64bit, _: 7;
218 pvm_capable, _: 8;
219 _reserved, _: 15, 9;
220}
221
222#[derive(IntoBytes, KnownLayout, FromBytes, Immutable)]
223#[repr(C, packed)]
224struct Msi32Capability {
225 id: u8,
226 next: u8,
227 control: u16,
228 address: u32,
229 data: u16,
230}
231
232#[derive(IntoBytes, KnownLayout, FromBytes, Immutable)]
233#[repr(C, packed)]
234struct Msi64Capability {
235 id: u8,
236 next: u8,
237 control: u16,
238 address: u32,
239 address_upper: u32,
240 data: u16,
241}
242
243#[derive(IntoBytes, KnownLayout, FromBytes, Immutable)]
244#[repr(C, packed)]
245struct MsixCapability {
246 id: u8,
247 next: u8,
248 control: u16,
249 table: u32,
250 pba: u32,
251}
252
253bitfield! {
254 pub struct MsixControl(u16);
255 table_size, _: 10, 0;
256 _reserved, _: 13, 11;
257 function_mask, _: 14;
258 enable, _: 15;
259}
260
261bitfield! {
262 pub struct MsixBarField(u32);
263 bir, _: 2, 0;
264 offset, _: 31, 3;
265}
266
267bitfield! {
268 pub struct PcieCapabilitiesField(u16);
269 version, _: 3, 0;
270 device_type, _: 7, 4;
271 slot_implemented, _: 8;
272 irq_number, _: 13, 9;
273 _reserved, _: 15, 14;
274}
275
276enum PcieDevicePortType {
278 PCIEEP,
279 LPCIEEP,
280 RCIEP,
281 RCEC,
282 RPPCIERC,
283 UPPCIES,
284 DPPCIES,
285 PCIE2PCIB,
286 PCI2PCIEB,
287 Unknown(u16),
288}
289
290impl From<u16> for PcieDevicePortType {
291 fn from(value: u16) -> Self {
292 match value {
293 0b0000 => PcieDevicePortType::PCIEEP,
294 0b0001 => PcieDevicePortType::LPCIEEP,
295 0b1001 => PcieDevicePortType::RCIEP,
296 0b1010 => PcieDevicePortType::RCEC,
297 0b0100 => PcieDevicePortType::RPPCIERC,
298 0b0101 => PcieDevicePortType::UPPCIES,
299 0b0110 => PcieDevicePortType::DPPCIES,
300 0b0111 => PcieDevicePortType::PCIE2PCIB,
301 0b1000 => PcieDevicePortType::PCI2PCIEB,
302 _ => PcieDevicePortType::Unknown(value),
303 }
304 }
305}
306
307impl fmt::Display for PcieDevicePortType {
308 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
309 write!(
310 f,
311 "{}",
312 match self {
313 PcieDevicePortType::PCIEEP => "PCI Express Endpoint",
314 PcieDevicePortType::LPCIEEP => "Legacy PCI Express Endpoint",
315 PcieDevicePortType::RCIEP => "Root Complex Integrated Endpoint",
316 PcieDevicePortType::RCEC => "Root Complex Event Collector",
317 PcieDevicePortType::RPPCIERC => "Root Port of PCI Express Root Complex",
318 PcieDevicePortType::UPPCIES => "Upstream Port of PCI Express Switch",
319 PcieDevicePortType::DPPCIES => "Downstream Port of PCI Express Switch",
320 PcieDevicePortType::PCIE2PCIB => "PCI Express to PCI/PCI-X Bridge",
321 PcieDevicePortType::PCI2PCIEB => "PCI/PCI-X to PCI Express Bridge",
322 PcieDevicePortType::Unknown(x) => return write!(f, "Unknown PCIe Type ({:#x})", x),
323 }
324 )
325 }
326}
327
328#[derive(IntoBytes, KnownLayout, FromBytes, Immutable)]
329#[repr(C, packed)]
330struct PciExpressCapability {
331 id: u8,
332 next: u8,
333 pcie_capabilities: u16,
334 device_capabilities: u32,
335 device_control: u16,
336 device_status: u16,
337}