mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2026-05-14 21:38:46 +02:00
015b1d3650
The GSP bootloader is a small RISC-V firmware that is loaded by Booter onto the GSP core and is in charge of loading, validating, and starting the actual GSP firmware. It is a regular binary firmware file containing a specific header. Create a type holding the DMA-mapped firmware as well as useful information extracted from the header, and hook it into our firmware structure for later use. The GSP bootloader is stored into the `GspFirmware` structure, since it is part of the GSP firmware package. This makes the `Firmware` structure empty, so remove it. Reviewed-by: John Hubbard <jhubbard@nvidia.com> Acked-by: Danilo Krummrich <dakr@kernel.org> Link: https://lore.kernel.org/r/20250913-nova_firmware-v6-8-9007079548b0@nvidia.com Signed-off-by: Alexandre Courbot <acourbot@nvidia.com>
92 lines
2.7 KiB
Rust
92 lines
2.7 KiB
Rust
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
//! Support for firmware binaries designed to run on a RISC-V core. Such firmwares files have a
|
|
//! dedicated header.
|
|
|
|
use core::mem::size_of;
|
|
|
|
use kernel::device;
|
|
use kernel::firmware::Firmware;
|
|
use kernel::prelude::*;
|
|
use kernel::transmute::FromBytes;
|
|
|
|
use crate::dma::DmaObject;
|
|
use crate::firmware::BinFirmware;
|
|
|
|
/// Descriptor for microcode running on a RISC-V core.
|
|
#[repr(C)]
|
|
#[derive(Debug)]
|
|
struct RmRiscvUCodeDesc {
|
|
version: u32,
|
|
bootloader_offset: u32,
|
|
bootloader_size: u32,
|
|
bootloader_param_offset: u32,
|
|
bootloader_param_size: u32,
|
|
riscv_elf_offset: u32,
|
|
riscv_elf_size: u32,
|
|
app_version: u32,
|
|
manifest_offset: u32,
|
|
manifest_size: u32,
|
|
monitor_data_offset: u32,
|
|
monitor_data_size: u32,
|
|
monitor_code_offset: u32,
|
|
monitor_code_size: u32,
|
|
}
|
|
|
|
// SAFETY: all bit patterns are valid for this type, and it doesn't use interior mutability.
|
|
unsafe impl FromBytes for RmRiscvUCodeDesc {}
|
|
|
|
impl RmRiscvUCodeDesc {
|
|
/// Interprets the header of `bin_fw` as a [`RmRiscvUCodeDesc`] and returns it.
|
|
///
|
|
/// Fails if the header pointed at by `bin_fw` is not within the bounds of the firmware image.
|
|
fn new(bin_fw: &BinFirmware<'_>) -> Result<Self> {
|
|
let offset = bin_fw.hdr.header_offset as usize;
|
|
|
|
bin_fw
|
|
.fw
|
|
.get(offset..offset + size_of::<Self>())
|
|
.and_then(Self::from_bytes_copy)
|
|
.ok_or(EINVAL)
|
|
}
|
|
}
|
|
|
|
/// A parsed firmware for a RISC-V core, ready to be loaded and run.
|
|
#[expect(unused)]
|
|
pub(crate) struct RiscvFirmware {
|
|
/// Offset at which the code starts in the firmware image.
|
|
code_offset: u32,
|
|
/// Offset at which the data starts in the firmware image.
|
|
data_offset: u32,
|
|
/// Offset at which the manifest starts in the firmware image.
|
|
manifest_offset: u32,
|
|
/// Application version.
|
|
app_version: u32,
|
|
/// Device-mapped firmware image.
|
|
ucode: DmaObject,
|
|
}
|
|
|
|
impl RiscvFirmware {
|
|
/// Parses the RISC-V firmware image contained in `fw`.
|
|
pub(crate) fn new(dev: &device::Device<device::Bound>, fw: &Firmware) -> Result<Self> {
|
|
let bin_fw = BinFirmware::new(fw)?;
|
|
|
|
let riscv_desc = RmRiscvUCodeDesc::new(&bin_fw)?;
|
|
|
|
let ucode = {
|
|
let start = bin_fw.hdr.data_offset as usize;
|
|
let len = bin_fw.hdr.data_size as usize;
|
|
|
|
DmaObject::from_data(dev, fw.data().get(start..start + len).ok_or(EINVAL)?)?
|
|
};
|
|
|
|
Ok(Self {
|
|
ucode,
|
|
code_offset: riscv_desc.monitor_code_offset,
|
|
data_offset: riscv_desc.monitor_data_offset,
|
|
manifest_offset: riscv_desc.manifest_offset,
|
|
app_version: riscv_desc.app_version,
|
|
})
|
|
}
|
|
}
|