root/arch/csky/kernel/cpu-probe.c
// 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,
};