mirror of
https://github.com/riscv-software-src/opensbi.git
synced 2025-12-17 04:21:39 +00:00
lib: sbi: Introduce hart protection abstraction
Currently, PMP and ePMP are the only hart protection mechanisms available in OpenSBI but new protection mechanisms (such as Smmpt) will be added in the near future. To allow multiple hart protection mechanisms, introduce hart protection abstraction and related APIs. Signed-off-by: Anup Patel <apatel@ventanamicro.com> Link: https://lore.kernel.org/r/20251209135235.423391-3-apatel@ventanamicro.com Signed-off-by: Anup Patel <anup@brainfault.org>
This commit is contained in:
96
lib/sbi/sbi_hart_protection.c
Normal file
96
lib/sbi/sbi_hart_protection.c
Normal file
@@ -0,0 +1,96 @@
|
||||
/*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*
|
||||
* Copyright (c) 2025 Ventana Micro Systems Inc.
|
||||
*/
|
||||
|
||||
#include <sbi/sbi_error.h>
|
||||
#include <sbi/sbi_hart_protection.h>
|
||||
#include <sbi/sbi_scratch.h>
|
||||
|
||||
static SBI_LIST_HEAD(hart_protection_list);
|
||||
|
||||
struct sbi_hart_protection *sbi_hart_protection_best(void)
|
||||
{
|
||||
if (sbi_list_empty(&hart_protection_list))
|
||||
return NULL;
|
||||
|
||||
return sbi_list_first_entry(&hart_protection_list, struct sbi_hart_protection, head);
|
||||
}
|
||||
|
||||
int sbi_hart_protection_register(struct sbi_hart_protection *hprot)
|
||||
{
|
||||
struct sbi_hart_protection *pos = NULL;
|
||||
bool found_pos = false;
|
||||
|
||||
if (!hprot)
|
||||
return SBI_EINVAL;
|
||||
|
||||
sbi_list_for_each_entry(pos, &hart_protection_list, head) {
|
||||
if (hprot->rating > pos->rating) {
|
||||
found_pos = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (found_pos)
|
||||
sbi_list_add_tail(&hprot->head, &pos->head);
|
||||
else
|
||||
sbi_list_add_tail(&hprot->head, &hart_protection_list);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void sbi_hart_protection_unregister(struct sbi_hart_protection *hprot)
|
||||
{
|
||||
if (!hprot)
|
||||
return;
|
||||
|
||||
sbi_list_del(&hprot->head);
|
||||
}
|
||||
|
||||
int sbi_hart_protection_configure(struct sbi_scratch *scratch)
|
||||
{
|
||||
struct sbi_hart_protection *hprot = sbi_hart_protection_best();
|
||||
|
||||
if (!hprot)
|
||||
return SBI_EINVAL;
|
||||
if (!hprot->configure)
|
||||
return SBI_ENOSYS;
|
||||
|
||||
return hprot->configure(scratch);
|
||||
}
|
||||
|
||||
void sbi_hart_protection_unconfigure(struct sbi_scratch *scratch)
|
||||
{
|
||||
struct sbi_hart_protection *hprot = sbi_hart_protection_best();
|
||||
|
||||
if (!hprot || !hprot->unconfigure)
|
||||
return;
|
||||
|
||||
hprot->unconfigure(scratch);
|
||||
}
|
||||
|
||||
int sbi_hart_protection_map_range(unsigned long base, unsigned long size)
|
||||
{
|
||||
struct sbi_hart_protection *hprot = sbi_hart_protection_best();
|
||||
|
||||
if (!hprot)
|
||||
return SBI_EINVAL;
|
||||
if (!hprot->map_range)
|
||||
return 0;
|
||||
|
||||
return hprot->map_range(sbi_scratch_thishart_ptr(), base, size);
|
||||
}
|
||||
|
||||
int sbi_hart_protection_unmap_range(unsigned long base, unsigned long size)
|
||||
{
|
||||
struct sbi_hart_protection *hprot = sbi_hart_protection_best();
|
||||
|
||||
if (!hprot)
|
||||
return SBI_EINVAL;
|
||||
if (!hprot->unmap_range)
|
||||
return 0;
|
||||
|
||||
return hprot->unmap_range(sbi_scratch_thishart_ptr(), base, size);
|
||||
}
|
||||
Reference in New Issue
Block a user