#include "opt_platform.h"
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/ctype.h>
#include <sys/kernel.h>
#include <sys/pcpu.h>
#include <sys/sysctl.h>
#include <machine/cpu.h>
#include <machine/cpufunc.h>
#include <machine/elf.h>
#include <machine/md_var.h>
#include <machine/thead.h>
#include <machine/cbo.h>
#ifdef FDT
#include <dev/fdt/fdt_common.h>
#include <dev/ofw/openfirm.h>
#include <dev/ofw/ofw_bus_subr.h>
#endif
const char machine[] = "riscv";
SYSCTL_CONST_STRING(_hw, HW_MACHINE, machine, CTLFLAG_RD | CTLFLAG_CAPRD,
machine, "Machine class");
register_t mvendorid;
register_t marchid;
register_t mimpid;
u_int mmu_caps;
bool has_hyp;
bool __read_frequently has_sstc;
bool __read_frequently has_sscofpmf;
bool has_svpbmt;
bool has_zicbom;
bool has_zicboz;
bool has_zicbop;
struct cpu_desc {
const char *cpu_mvendor_name;
const char *cpu_march_name;
u_int isa_extensions;
u_int mmu_caps;
u_int smode_extensions;
#define SV_SSTC (1 << 0)
#define SV_SVNAPOT (1 << 1)
#define SV_SVPBMT (1 << 2)
#define SV_SVINVAL (1 << 3)
#define SV_SSCOFPMF (1 << 4)
u_int z_extensions;
#define Z_ZICBOM (1 << 0)
#define Z_ZICBOZ (1 << 1)
#define Z_ZICBOP (1 << 2)
int cbom_block_size;
int cboz_block_size;
};
struct cpu_desc cpu_desc[MAXCPU];
struct marchid_entry {
register_t march_id;
const char *march_name;
};
#define MARCHID_END { -1ul, NULL }
static const struct marchid_entry global_marchids[] = {
{ MARCHID_UCB_ROCKET, "UC Berkeley Rocket" },
{ MARCHID_UCB_BOOM, "UC Berkeley Boom" },
{ MARCHID_UCB_SPIKE, "UC Berkeley Spike" },
{ MARCHID_UCAM_RVBS, "University of Cambridge RVBS" },
MARCHID_END
};
static const struct marchid_entry sifive_marchids[] = {
{ MARCHID_SIFIVE_U7, "6/7/P200/X200-Series Processor" },
{ MARCHID_SIFIVE_P5, "P550/P650 Processor" },
MARCHID_END
};
static const struct {
register_t mvendor_id;
const char *mvendor_name;
const struct marchid_entry *marchid_table;
} mvendor_ids[] = {
{ MVENDORID_UNIMPL, "Unspecified", NULL },
{ MVENDORID_SIFIVE, "SiFive", sifive_marchids },
{ MVENDORID_THEAD, "T-Head", NULL },
};
#define ISA_PREFIX ("rv" __XSTRING(__riscv_xlen))
#define ISA_PREFIX_LEN (sizeof(ISA_PREFIX) - 1)
static __inline int
parse_ext_s(struct cpu_desc *desc, char *isa, int idx, int len)
{
#define CHECK_S_EXT(str, flag) \
do { \
if (strncmp(&isa[idx], (str), \
MIN(strlen(str), len - idx)) == 0) { \
desc->smode_extensions |= flag; \
return (idx + strlen(str)); \
} \
} while (0)
CHECK_S_EXT("sstc", SV_SSTC);
CHECK_S_EXT("svnapot", SV_SVNAPOT);
CHECK_S_EXT("svpbmt", SV_SVPBMT);
CHECK_S_EXT("svinval", SV_SVINVAL);
CHECK_S_EXT("sscofpmf", SV_SSCOFPMF);
#undef CHECK_S_EXT
while (isa[idx] != '_' && idx < len) {
idx++;
}
return (idx);
}
static __inline int
parse_ext_x(struct cpu_desc *desc __unused, char *isa, int idx, int len)
{
while (isa[idx] != '_' && idx < len) {
idx++;
}
return (idx);
}
static __inline int
parse_ext_z(struct cpu_desc *desc __unused, char *isa, int idx, int len)
{
#define CHECK_Z_EXT(str, flag) \
do { \
if (strncmp(&isa[idx], (str), \
MIN(strlen(str), len - idx)) == 0) { \
desc->z_extensions |= flag; \
return (idx + strlen(str)); \
} \
} while (0)
CHECK_Z_EXT("zicbom", Z_ZICBOM);
CHECK_Z_EXT("zicboz", Z_ZICBOZ);
CHECK_Z_EXT("zicbop", Z_ZICBOP);
#undef CHECK_Z_EXT
while (isa[idx] != '_' && idx < len) {
idx++;
}
return (idx);
}
static __inline int
parse_ext_version(char *isa, int idx, u_int *majorp __unused,
u_int *minorp __unused)
{
while (isdigit(isa[idx]))
idx++;
if (isa[idx] != 'p')
return (idx);
else
idx++;
while (isdigit(isa[idx]))
idx++;
return (idx);
}
static int
parse_riscv_isa(struct cpu_desc *desc, char *isa, int len)
{
int i;
if (strncmp(isa, ISA_PREFIX, ISA_PREFIX_LEN) != 0) {
printf("%s: Unrecognized ISA string: %s\n", __func__, isa);
return (-1);
}
i = ISA_PREFIX_LEN;
while (i < len) {
switch(isa[i]) {
case 'a':
case 'b':
case 'c':
case 'd':
case 'f':
case 'h':
case 'i':
case 'm':
desc->isa_extensions |= HWCAP_ISA_BIT(isa[i]);
i++;
break;
case 'g':
desc->isa_extensions |= HWCAP_ISA_G;
i++;
break;
case 's':
if (isa[i - 1] != '_' && isa[i + 1] == 'u') {
i += 2;
continue;
}
i = parse_ext_s(desc, isa, i, len);
break;
case 'x':
i = parse_ext_x(desc, isa, i, len);
break;
case 'z':
i = parse_ext_z(desc, isa, i, len);
break;
case '_':
i++;
continue;
default:
i++;
break;
}
i = parse_ext_version(isa, i, NULL, NULL);
}
return (0);
}
#ifdef FDT
static void
parse_mmu_fdt(struct cpu_desc *desc, phandle_t node)
{
char mmu[16];
desc->mmu_caps |= MMU_SV39;
if (OF_getprop(node, "mmu-type", mmu, sizeof(mmu)) > 0) {
if (strcmp(mmu, "riscv,sv48") == 0)
desc->mmu_caps |= MMU_SV48;
else if (strcmp(mmu, "riscv,sv57") == 0)
desc->mmu_caps |= MMU_SV48 | MMU_SV57;
}
}
static void
parse_cbo_fdt(struct cpu_desc *desc, phandle_t node)
{
int error;
error = OF_getencprop(node, "riscv,cbom-block-size",
&desc->cbom_block_size, sizeof(desc->cbom_block_size));
if (error == -1)
desc->cbom_block_size = 0;
error = OF_getencprop(node, "riscv,cboz-block-size",
&desc->cboz_block_size, sizeof(desc->cboz_block_size));
if (error == -1)
desc->cboz_block_size = 0;
}
static void
identify_cpu_features_fdt(u_int cpu, struct cpu_desc *desc)
{
char isa[1024];
phandle_t node;
ssize_t len;
pcell_t reg;
u_int hart;
node = OF_finddevice("/cpus");
if (node == -1) {
printf("%s: could not find /cpus node in FDT\n", __func__);
return;
}
hart = pcpu_find(cpu)->pc_hart;
for (node = OF_child(node); node > 0; node = OF_peer(node)) {
if (!ofw_bus_node_is_compatible(node, "riscv"))
continue;
if (OF_getencprop(node, "reg", ®, sizeof(reg)) <= 0 ||
reg != hart)
continue;
len = OF_getprop(node, "riscv,isa", isa, sizeof(isa));
KASSERT(len <= sizeof(isa), ("ISA string truncated"));
if (len == -1) {
printf("%s: could not find 'riscv,isa' property "
"for CPU %d, hart %u\n", __func__, cpu, hart);
return;
}
for (int i = 0; i < len; i++)
isa[i] = tolower(isa[i]);
if (parse_riscv_isa(desc, isa, len) != 0)
return;
parse_mmu_fdt(desc, node);
parse_cbo_fdt(desc, node);
break;
}
if (node <= 0) {
printf("%s: could not find FDT node for CPU %u, hart %u\n",
__func__, cpu, hart);
}
}
#endif
static void
identify_cpu_features(u_int cpu, struct cpu_desc *desc)
{
#ifdef FDT
identify_cpu_features_fdt(cpu, desc);
#endif
}
static void
update_global_capabilities(u_int cpu, struct cpu_desc *desc)
{
#define UPDATE_CAP(t, v) \
do { \
if (cpu == 0) { \
(t) = (v); \
} else { \
(t) &= (v); \
} \
} while (0)
UPDATE_CAP(elf_hwcap, (u_long)desc->isa_extensions);
UPDATE_CAP(mmu_caps, desc->mmu_caps);
UPDATE_CAP(has_hyp, (desc->isa_extensions & HWCAP_ISA_H) != 0);
UPDATE_CAP(has_sstc, (desc->smode_extensions & SV_SSTC) != 0);
UPDATE_CAP(has_sscofpmf, (desc->smode_extensions & SV_SSCOFPMF) != 0);
UPDATE_CAP(has_svpbmt, (desc->smode_extensions & SV_SVPBMT) != 0);
UPDATE_CAP(has_zicbom, (desc->z_extensions & Z_ZICBOM) != 0);
UPDATE_CAP(has_zicboz, (desc->z_extensions & Z_ZICBOZ) != 0);
UPDATE_CAP(has_zicbop, (desc->z_extensions & Z_ZICBOP) != 0);
#undef UPDATE_CAP
}
static void
identify_cpu_ids(struct cpu_desc *desc)
{
const struct marchid_entry *table = NULL;
int i;
desc->cpu_mvendor_name = "Unknown";
desc->cpu_march_name = "Unknown";
for (i = 0; i < nitems(mvendor_ids); i++) {
if (mvendorid == mvendor_ids[i].mvendor_id) {
desc->cpu_mvendor_name = mvendor_ids[i].mvendor_name;
table = mvendor_ids[i].marchid_table;
break;
}
}
if (marchid == MARCHID_UNIMPL) {
desc->cpu_march_name = "Unspecified";
return;
}
if (MARCHID_IS_OPENSOURCE(marchid)) {
table = global_marchids;
} else if (table == NULL)
return;
for (i = 0; table[i].march_name != NULL; i++) {
if (marchid == table[i].march_id) {
desc->cpu_march_name = table[i].march_name;
break;
}
}
}
static void
handle_thead_quirks(u_int cpu, struct cpu_desc *desc)
{
if (cpu != 0)
return;
has_errata_thead_pbmt = true;
thead_setup_cache();
}
static void
handle_cpu_quirks(u_int cpu, struct cpu_desc *desc)
{
switch (mvendorid) {
case MVENDORID_THEAD:
handle_thead_quirks(cpu, desc);
break;
}
}
void
identify_cpu(u_int cpu)
{
struct cpu_desc *desc = &cpu_desc[cpu];
identify_cpu_ids(desc);
identify_cpu_features(cpu, desc);
update_global_capabilities(cpu, desc);
handle_cpu_quirks(cpu, desc);
if (has_zicbom && cpu == 0)
cbo_zicbom_setup_cache(desc->cbom_block_size);
}
void
printcpuinfo(u_int cpu)
{
struct cpu_desc *desc;
u_int hart;
desc = &cpu_desc[cpu];
hart = pcpu_find(cpu)->pc_hart;
KASSERT(desc->isa_extensions != 0,
("Empty extension set for CPU %u, did parsing fail?", cpu));
#define SHOULD_PRINT(_field) \
(cpu == 0 || desc[0]._field != desc[-1]._field)
printf("CPU %-3u: Vendor=%s Core=%s (Hart %u)\n", cpu,
desc->cpu_mvendor_name, desc->cpu_march_name, hart);
if (cpu == 0)
printf(" marchid=%#lx, mimpid=%#lx\n", marchid, mimpid);
if (SHOULD_PRINT(mmu_caps)) {
printf(" MMU: %#b\n", desc->mmu_caps,
"\020"
"\01Sv39"
"\02Sv48"
"\03Sv57");
}
if (SHOULD_PRINT(isa_extensions)) {
printf(" ISA: %#b\n", desc->isa_extensions,
"\020"
"\01Atomic"
"\03Compressed"
"\04Double"
"\06Float"
"\10Hypervisor"
"\15Mult/Div");
}
if (SHOULD_PRINT(smode_extensions)) {
printf(" S-mode Extensions: %#b\n", desc->smode_extensions,
"\020"
"\01Sstc"
"\02Svnapot"
"\03Svpbmt"
"\04Svinval"
"\05Sscofpmf");
}
#undef SHOULD_PRINT
}