|  | // 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, | 
|  | }; |