// SPDX-License-Identifier: GPL-2.0 // Copyright (C) 2018 Hangzhou C-SKY Microsystems co.,ltd. #include <linux/of.h> #include <linux/init.h> #include <linux/seq_file.h> #include <linux/memblock.h> #include <abi/reg_ops.h> static void percpu_print(void *arg) { struct seq_file *m = (struct seq_file *)arg; unsigned int cur, next, i; seq_printf(m, "processor : %d\n", smp_processor_id()); seq_printf(m, "C-SKY CPU model : %s\n", CSKYCPU_DEF_NAME); /* read processor id, max is 100 */ cur = mfcr("cr13"); for (i = 0; i < 100; i++) { seq_printf(m, "product info[%d] : 0x%08x\n", i, cur); next = mfcr("cr13"); /* some CPU only has one id reg */ if (cur == next) break; cur = next; /* cpid index is 31-28, reset */ if (!(next >> 28)) { while ((mfcr("cr13") >> 28) != i); break; } } /* CPU feature regs, setup by bootloader or gdbinit */ seq_printf(m, "hint (CPU funcs): 0x%08x\n", mfcr_hint()); seq_printf(m, "ccr (L1C & MMU): 0x%08x\n", mfcr("cr18")); seq_printf(m, "ccr2 (L2C) : 0x%08x\n", mfcr_ccr2()); seq_printf(m, "\n"); } static int c_show(struct seq_file *m, void *v) { int cpu; for_each_online_cpu(cpu) smp_call_function_single(cpu, percpu_print, m, true); #ifdef CSKY_ARCH_VERSION seq_printf(m, "arch-version : %s\n", CSKY_ARCH_VERSION); seq_printf(m, "\n"); #endif return 0; } static void *c_start(struct seq_file *m, loff_t *pos) { return *pos < 1 ? (void *)1 : NULL; } static void *c_next(struct seq_file *m, void *v, loff_t *pos) { ++*pos; return NULL; } static void c_stop(struct seq_file *m, void *v) {} const struct seq_operations cpuinfo_op = { .start = c_start, .next = c_next, .stop = c_stop, .show = c_show, };