use core::marker::PhantomData;
use kernel::{
device,
firmware,
prelude::*,
str::CString,
transmute::FromBytes, };
use crate::{
dma::DmaObject,
falcon::FalconFirmware,
gpu,
num::{
FromSafeCast,
IntoSafeCast, },
};
pub(crate) mod booter;
pub(crate) mod fwsec;
pub(crate) mod gsp;
pub(crate) mod riscv;
pub(crate) const FIRMWARE_VERSION: &str = "570.144";
fn request_firmware(
dev: &device::Device,
chipset: gpu::Chipset,
name: &str,
ver: &str,
) -> Result<firmware::Firmware> {
let chip_name = chipset.name();
CString::try_from_fmt(fmt!("nvidia/{chip_name}/gsp/{name}-{ver}.bin"))
.and_then(|path| firmware::Firmware::request(&path, dev))
}
#[repr(C)]
#[derive(Debug, Clone)]
pub(crate) struct FalconUCodeDescV3 {
hdr: u32,
stored_size: u32,
pub(crate) pkc_data_offset: u32,
pub(crate) interface_offset: u32,
pub(crate) imem_phys_base: u32,
pub(crate) imem_load_size: u32,
pub(crate) imem_virt_base: u32,
pub(crate) dmem_phys_base: u32,
pub(crate) dmem_load_size: u32,
pub(crate) engine_id_mask: u16,
pub(crate) ucode_id: u8,
pub(crate) signature_count: u8,
pub(crate) signature_versions: u16,
_reserved: u16,
}
impl FalconUCodeDescV3 {
pub(crate) fn size(&self) -> usize {
const HDR_SIZE_SHIFT: u32 = 16;
const HDR_SIZE_MASK: u32 = 0xffff0000;
((self.hdr & HDR_SIZE_MASK) >> HDR_SIZE_SHIFT).into_safe_cast()
}
}
trait SignedState {}
struct Unsigned;
impl SignedState for Unsigned {}
struct Signed;
impl SignedState for Signed {}
struct FirmwareDmaObject<F: FalconFirmware, S: SignedState>(DmaObject, PhantomData<(F, S)>);
trait FirmwareSignature<F: FalconFirmware>: AsRef<[u8]> {}
impl<F: FalconFirmware> FirmwareDmaObject<F, Unsigned> {
fn patch_signature<S: FirmwareSignature<F>>(
mut self,
signature: &S,
sig_base_img: usize,
) -> Result<FirmwareDmaObject<F, Signed>> {
let signature_bytes = signature.as_ref();
if sig_base_img + signature_bytes.len() > self.0.size() {
return Err(EINVAL);
}
let dst = unsafe { self.0.start_ptr_mut().add(sig_base_img) };
unsafe {
core::ptr::copy_nonoverlapping(signature_bytes.as_ptr(), dst, signature_bytes.len())
};
Ok(FirmwareDmaObject(self.0, PhantomData))
}
fn no_patch_signature(self) -> FirmwareDmaObject<F, Signed> {
FirmwareDmaObject(self.0, PhantomData)
}
}
#[repr(C)]
#[derive(Debug, Clone)]
struct BinHdr {
bin_magic: u32,
bin_ver: u32,
bin_size: u32,
header_offset: u32,
data_offset: u32,
data_size: u32,
}
unsafe impl FromBytes for BinHdr {}
struct BinFirmware<'a> {
hdr: BinHdr,
fw: &'a [u8],
}
impl<'a> BinFirmware<'a> {
fn new(fw: &'a firmware::Firmware) -> Result<Self> {
const BIN_MAGIC: u32 = 0x10de;
let fw = fw.data();
fw.get(0..size_of::<BinHdr>())
.and_then(BinHdr::from_bytes_copy)
.and_then(|hdr| {
if hdr.bin_magic == BIN_MAGIC {
Some(hdr)
} else {
None
}
})
.map(|hdr| Self { hdr, fw })
.ok_or(EINVAL)
}
fn data(&self) -> Option<&[u8]> {
let fw_start = usize::from_safe_cast(self.hdr.data_offset);
let fw_size = usize::from_safe_cast(self.hdr.data_size);
self.fw.get(fw_start..fw_start + fw_size)
}
}
pub(crate) struct ModInfoBuilder<const N: usize>(firmware::ModInfoBuilder<N>);
impl<const N: usize> ModInfoBuilder<N> {
const fn make_entry_file(self, chipset: &str, fw: &str) -> Self {
ModInfoBuilder(
self.0
.new_entry()
.push("nvidia/")
.push(chipset)
.push("/gsp/")
.push(fw)
.push("-")
.push(FIRMWARE_VERSION)
.push(".bin"),
)
}
const fn make_entry_chipset(self, chipset: &str) -> Self {
self.make_entry_file(chipset, "booter_load")
.make_entry_file(chipset, "booter_unload")
.make_entry_file(chipset, "bootloader")
.make_entry_file(chipset, "gsp")
}
pub(crate) const fn create(
module_name: &'static kernel::str::CStr,
) -> firmware::ModInfoBuilder<N> {
let mut this = Self(firmware::ModInfoBuilder::new(module_name));
let mut i = 0;
while i < gpu::Chipset::ALL.len() {
this = this.make_entry_chipset(gpu::Chipset::ALL[i].name());
i += 1;
}
this.0
}
}