#include <sys/param.h>
#include <sys/systm.h>
#include <sys/types.h>
#include <machine/sbi.h>
#include <dev/cons.h>
extern void (*cpuresetfn)(void);
extern void (*powerdownfn)(void);
#define OPENSBI_VERSION_MAJOR_OFFSET 16
#define OPENSBI_VERSION_MINOR_MASK 0xFFFF
u_long sbi_spec_version;
u_long sbi_impl_id;
u_long sbi_impl_version;
static struct sbi_ret
sbi_get_spec_version(void)
{
return (SBI_CALL0(SBI_EXT_ID_BASE, SBI_BASE_GET_SPEC_VERSION));
}
static struct sbi_ret
sbi_get_impl_id(void)
{
return (SBI_CALL0(SBI_EXT_ID_BASE, SBI_BASE_GET_IMPL_ID));
}
static struct sbi_ret
sbi_get_impl_version(void)
{
return (SBI_CALL0(SBI_EXT_ID_BASE, SBI_BASE_GET_IMPL_VERSION));
}
void
sbi_print_version(void)
{
u_int major;
u_int minor;
if (sbi_spec_version == 0) {
printf("SBI: Unknown (Legacy) Implementation\n");
printf("SBI Specification Version: 0.1\n");
return;
}
switch (sbi_impl_id) {
case (SBI_IMPL_ID_BBL):
printf("SBI: Berkely Boot Loader %lu", sbi_impl_version);
break;
case (SBI_IMPL_ID_OPENSBI):
major = sbi_impl_version >> OPENSBI_VERSION_MAJOR_OFFSET;
minor = sbi_impl_version & OPENSBI_VERSION_MINOR_MASK;
printf("SBI: OpenSBI v%u.%u", major, minor);
break;
default:
printf("SBI: Unrecognized Implementation: %lu", sbi_impl_id);
break;
}
major = (sbi_spec_version & SBI_SPEC_VERS_MAJOR_MASK) >>
SBI_SPEC_VERS_MAJOR_OFFSET;
minor = (sbi_spec_version & SBI_SPEC_VERS_MINOR_MASK);
printf(", SBI Specification Version %u.%u\n", major, minor);
}
#ifdef MULTIPROCESSOR
int
sbi_hsm_hart_start(u_long hart, u_long start_addr, u_long priv)
{
struct sbi_ret ret;
ret = SBI_CALL3(SBI_EXT_ID_HSM, SBI_HSM_HART_START, hart, start_addr,
priv);
return (ret.error != 0 ? (int)ret.error : 0);
}
void
sbi_hsm_hart_stop(void)
{
(void)SBI_CALL0(SBI_EXT_ID_HSM, SBI_HSM_HART_STOP);
}
int
sbi_hsm_hart_status(u_long hart)
{
struct sbi_ret ret;
ret = SBI_CALL1(SBI_EXT_ID_HSM, SBI_HSM_HART_STATUS, hart);
return (ret.error != 0 ? (int)ret.error : (int)ret.value);
}
#endif
void
sbi_reset(void)
{
SBI_CALL2(SBI_EXT_ID_SRST, SBI_SRST_RESET,
SBI_SRST_RESET_WARM_REBOOT, 0);
}
void
sbi_powerdown(void)
{
SBI_CALL2(SBI_EXT_ID_SRST, SBI_SRST_RESET,
SBI_SRST_RESET_SHUTDOWN, 0);
}
void
sbi_init(void)
{
struct sbi_ret sret;
sret = sbi_get_spec_version();
if (sret.error != 0) {
sbi_spec_version = 0;
return;
}
sbi_spec_version = sret.value;
sbi_impl_id = sbi_get_impl_id().value;
sbi_impl_version = sbi_get_impl_version().value;
KASSERTMSG(sbi_probe_extension(SBI_SET_TIMER) != 0,
"SBI doesn't implement sbi_set_timer()");
KASSERTMSG(sbi_probe_extension(SBI_CONSOLE_PUTCHAR) != 0,
"SBI doesn't implement sbi_console_putchar()");
KASSERTMSG(sbi_probe_extension(SBI_CONSOLE_GETCHAR) != 0,
"SBI doesn't implement sbi_console_getchar()");
KASSERTMSG(sbi_probe_extension(SBI_CLEAR_IPI) != 0,
"SBI doesn't implement sbi_clear_ipi()");
KASSERTMSG(sbi_probe_extension(SBI_SEND_IPI) != 0,
"SBI doesn't implement sbi_send_ipi()");
KASSERTMSG(sbi_probe_extension(SBI_REMOTE_FENCE_I) != 0,
"SBI doesn't implement sbi_remote_fence_i()");
KASSERTMSG(sbi_probe_extension(SBI_REMOTE_SFENCE_VMA) != 0,
"SBI doesn't implement sbi_remote_sfence_vma()");
KASSERTMSG(sbi_probe_extension(SBI_REMOTE_SFENCE_VMA_ASID) != 0,
"SBI doesn't implement sbi_remote_sfence_vma_asid()");
KASSERTMSG(sbi_probe_extension(SBI_SHUTDOWN) != 0,
"SBI doesn't implement sbi_shutdown()");
if (sbi_probe_extension(SBI_EXT_ID_SRST) != 0) {
cpuresetfn = sbi_reset;
powerdownfn = sbi_powerdown;
}
}
void
sbi_cnprobe(struct consdev *cd)
{
}
void
sbi_cninit(struct consdev *cd)
{
}
int
sbi_cngetc(dev_t dev)
{
int c;
for (;;) {
c = sbi_console_getchar();
if (c != -1)
return c;
}
}
void
sbi_cnputc(dev_t dev, int c)
{
sbi_console_putchar(c);
}
void
sbi_cnpollc(dev_t dev, int on)
{
}
struct consdev sbi_consdev = {
.cn_probe = sbi_cnprobe,
.cn_init = sbi_cninit,
.cn_getc = sbi_cngetc,
.cn_putc = sbi_cnputc,
.cn_pollc = sbi_cnpollc,
};
struct consdev *cn_tab = &sbi_consdev;