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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
|
use alloc::boxed::Box;
use core::{
format_args,
pin::Pin,
};
use kernel::{
bindings,
c_str,
driver,
error::code::*,
io_mem::IoMem,
pci, pci::define_pci_id_table,
prelude::*,
};
use crate::gpu::Gpu;
use crate::gpu::CardType;
struct NovaDevice;
/// BAR Size
pub(crate) const BAR_SIZE: usize = 16777216;
impl pci::Driver for NovaDevice {
define_pci_id_table! {
(),
[ (pci::DeviceId::new(bindings::PCI_VENDOR_ID_NVIDIA, bindings::PCI_ANY_ID as u32), None) ]
}
fn probe(dev: &mut pci::Device, _id_info: Option<&Self::IdInfo>) -> Result {
pr_info!("probe()\n");
dev.enable_device_mem()?;
dev.set_master();
let _bars = dev.select_bars(bindings::IORESOURCE_MEM.into());
let res = dev.take_resource(0).ok_or(ENXIO)?;
let bar = unsafe { IoMem::<BAR_SIZE>::try_new(res) }?;
let boot0 = u64::from_le(bar.readq(0));
if boot0 & 0x1f000000 == 0 {
return Err(ENODEV);
}
pr_info!("nvidia hw rev 0x{:#x}", boot0);
let chipset = (boot0 & 0x1ff00000) >> 20;
let _chiprev = boot0 & 0xff;
let card_type = match chipset & 0x1f0 {
0x160 => { pr_info!("TU100"); CardType::TU100 },
0x170 => { pr_info!("GA100"); CardType::GA100 },
0x190 => { pr_info!("AD100"); CardType::AD100 },
_ => return Err(ENODEV)
};
let mut gpu = Gpu::new(card_type, bar);
let _ = gpu.init(dev)?;
Ok(())
}
fn remove(_data: &Self::Data) {
pr_info!("remove()\n");
}
}
pub(crate) struct NovaModule {
_registration: Pin<Box<driver::Registration<pci::Adapter<NovaDevice>>>>,
}
impl kernel::Module for NovaModule {
fn init(_name: &'static CStr, module: &'static ThisModule) -> Result<Self> {
pr_info!("Module loaded!\n");
let registration = driver::Registration::new_pinned(c_str!("nova"), module)?;
pr_info!("Registerd PCI driver.\n");
Ok(Self {
_registration: registration,
})
}
}
|