vmotherboard/chipset/backing/arc_mutex/
pci.rs

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
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

use crate::BusIdPci;
use crate::chipset::PciConflict;
use crate::chipset::PciConflictReason;
use chipset_device::ChipsetDevice;
use closeable_mutex::CloseableMutex;
use std::collections::HashMap;
use std::sync::Arc;
use std::sync::Weak;

/// An abstraction over a PCI bus implementation that is able to route accesses
/// to `Weak<CloseableMutex<dyn ChipsetDevice>>` devices.
pub trait RegisterWeakMutexPci: Send {
    /// Try to add a PCI device to the bus, reporting any conflicts.
    fn add_pci_device(
        &mut self,
        bus: u8,
        device: u8,
        function: u8,
        name: Arc<str>,
        dev: Weak<CloseableMutex<dyn ChipsetDevice>>,
    ) -> Result<(), PciConflict>;
}

pub struct WeakMutexPciEntry {
    pub bdf: (u8, u8, u8),
    pub name: Arc<str>,
    pub dev: Weak<CloseableMutex<dyn ChipsetDevice>>,
}

#[derive(Default)]
pub struct BusResolverWeakMutexPci {
    pub buses: HashMap<BusIdPci, Box<dyn RegisterWeakMutexPci>>,
    pub devices: HashMap<BusIdPci, Vec<WeakMutexPciEntry>>,
}

impl BusResolverWeakMutexPci {
    pub fn resolve(mut self) -> Result<(), Vec<PciConflict>> {
        let mut errs = Vec::new();

        for (bus_id, entries) in self.devices {
            for WeakMutexPciEntry { bdf, name, dev } in entries {
                let pci_bus = match self.buses.get_mut(&bus_id) {
                    Some(bus) => bus,
                    None => {
                        errs.push(PciConflict {
                            bdf,
                            conflict_dev: name.clone(),
                            reason: PciConflictReason::MissingBus,
                        });
                        continue;
                    }
                };

                let (bus, device, function) = bdf;
                match pci_bus.add_pci_device(bus, device, function, name, dev) {
                    Ok(()) => {}
                    Err(conflict) => {
                        errs.push(conflict);
                        continue;
                    }
                };
            }
        }

        if !errs.is_empty() { Err(errs) } else { Ok(()) }
    }
}