18c2ecf20Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0 */ 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (C) 2020 Loongson Technology Corporation Limited 48c2ecf20Sopenharmony_ci */ 58c2ecf20Sopenharmony_ci#ifndef __ASM_CPU_INFO_H 68c2ecf20Sopenharmony_ci#define __ASM_CPU_INFO_H 78c2ecf20Sopenharmony_ci 88c2ecf20Sopenharmony_ci#include <linux/cache.h> 98c2ecf20Sopenharmony_ci#include <linux/types.h> 108c2ecf20Sopenharmony_ci 118c2ecf20Sopenharmony_ci#include <asm/loongarchregs.h> 128c2ecf20Sopenharmony_ci 138c2ecf20Sopenharmony_ci/* cache_desc->flags */ 148c2ecf20Sopenharmony_cienum { 158c2ecf20Sopenharmony_ci CACHE_PRESENT = (1 << 0), 168c2ecf20Sopenharmony_ci CACHE_PRIVATE = (1 << 1), /* core private cache */ 178c2ecf20Sopenharmony_ci CACHE_INCLUSIVE = (1 << 2), /* include the inner level caches */ 188c2ecf20Sopenharmony_ci}; 198c2ecf20Sopenharmony_ci 208c2ecf20Sopenharmony_ci/* 218c2ecf20Sopenharmony_ci * Descriptor for a cache 228c2ecf20Sopenharmony_ci */ 238c2ecf20Sopenharmony_cistruct cache_desc { 248c2ecf20Sopenharmony_ci unsigned char type; 258c2ecf20Sopenharmony_ci unsigned char level; 268c2ecf20Sopenharmony_ci unsigned short sets; /* Number of lines per set */ 278c2ecf20Sopenharmony_ci unsigned char ways; /* Number of ways */ 288c2ecf20Sopenharmony_ci unsigned char linesz; /* Size of line in bytes */ 298c2ecf20Sopenharmony_ci unsigned char flags; /* Flags describing cache properties */ 308c2ecf20Sopenharmony_ci}; 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_ci#define CACHE_LEVEL_MAX 3 338c2ecf20Sopenharmony_ci#define CACHE_LEAVES_MAX 6 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_cistruct guest_info { 368c2ecf20Sopenharmony_ci unsigned long ases; 378c2ecf20Sopenharmony_ci unsigned long ases_dyn; 388c2ecf20Sopenharmony_ci unsigned long long options; 398c2ecf20Sopenharmony_ci unsigned long long options_dyn; 408c2ecf20Sopenharmony_ci u8 conf; 418c2ecf20Sopenharmony_ci unsigned int kscratch_mask; 428c2ecf20Sopenharmony_ci}; 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_cistruct cpuinfo_loongarch { 458c2ecf20Sopenharmony_ci u64 asid_cache; 468c2ecf20Sopenharmony_ci unsigned long asid_mask; 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_ci /* 498c2ecf20Sopenharmony_ci * Capability and feature descriptor structure for LoongArch CPU 508c2ecf20Sopenharmony_ci */ 518c2ecf20Sopenharmony_ci unsigned long ases; 528c2ecf20Sopenharmony_ci unsigned long long options; 538c2ecf20Sopenharmony_ci unsigned int processor_id; 548c2ecf20Sopenharmony_ci unsigned int fpu_vers; 558c2ecf20Sopenharmony_ci unsigned int fpu_csr0; 568c2ecf20Sopenharmony_ci unsigned int fpu_mask; 578c2ecf20Sopenharmony_ci unsigned int cputype; 588c2ecf20Sopenharmony_ci int isa_level; 598c2ecf20Sopenharmony_ci int tlbsize; 608c2ecf20Sopenharmony_ci int tlbsizemtlb; 618c2ecf20Sopenharmony_ci int tlbsizestlbsets; 628c2ecf20Sopenharmony_ci int tlbsizestlbways; 638c2ecf20Sopenharmony_ci int cache_leaves_present; /* number of cache_leaves[] elements */ 648c2ecf20Sopenharmony_ci struct cache_desc cache_leaves[CACHE_LEAVES_MAX]; 658c2ecf20Sopenharmony_ci int core; /* physical core number in package */ 668c2ecf20Sopenharmony_ci int package;/* physical package number */ 678c2ecf20Sopenharmony_ci int global_id; /* physical global thread number */ 688c2ecf20Sopenharmony_ci int vabits; /* Virtual Address size in bits */ 698c2ecf20Sopenharmony_ci int pabits; /* Physical Address size in bits */ 708c2ecf20Sopenharmony_ci unsigned int ksave_mask; /* Usable KSave mask. */ 718c2ecf20Sopenharmony_ci unsigned int watch_dreg_count; /* Number data breakpoints */ 728c2ecf20Sopenharmony_ci unsigned int watch_ireg_count; /* Number instruction breakpoints */ 738c2ecf20Sopenharmony_ci unsigned int watch_reg_use_cnt; /* min(NUM_WATCH_REGS, watch_dreg_count + watch_ireg_count), Usable by ptrace */ 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci /* VZ & Guest features */ 768c2ecf20Sopenharmony_ci struct guest_info guest; 778c2ecf20Sopenharmony_ci unsigned long guest_cfg; 788c2ecf20Sopenharmony_ci} __attribute__((aligned(SMP_CACHE_BYTES))); 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_ciextern struct cpuinfo_loongarch cpu_data[]; 818c2ecf20Sopenharmony_ci#define boot_cpu_data cpu_data[0] 828c2ecf20Sopenharmony_ci#define current_cpu_data cpu_data[smp_processor_id()] 838c2ecf20Sopenharmony_ci#define raw_current_cpu_data cpu_data[raw_smp_processor_id()] 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_ciextern void cpu_probe(void); 868c2ecf20Sopenharmony_ci 878c2ecf20Sopenharmony_ciextern const char *__cpu_family[]; 888c2ecf20Sopenharmony_ciextern const char *__cpu_full_name[]; 898c2ecf20Sopenharmony_ci#define cpu_family_string() __cpu_family[raw_smp_processor_id()] 908c2ecf20Sopenharmony_ci#define cpu_full_name_string() __cpu_full_name[raw_smp_processor_id()] 918c2ecf20Sopenharmony_ci 928c2ecf20Sopenharmony_cistruct seq_file; 938c2ecf20Sopenharmony_cistruct notifier_block; 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_ciextern int register_proc_cpuinfo_notifier(struct notifier_block *nb); 968c2ecf20Sopenharmony_ciextern int proc_cpuinfo_notifier_call_chain(unsigned long val, void *v); 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_ci#define proc_cpuinfo_notifier(fn, pri) \ 998c2ecf20Sopenharmony_ci({ \ 1008c2ecf20Sopenharmony_ci static struct notifier_block fn##_nb = { \ 1018c2ecf20Sopenharmony_ci .notifier_call = fn, \ 1028c2ecf20Sopenharmony_ci .priority = pri \ 1038c2ecf20Sopenharmony_ci }; \ 1048c2ecf20Sopenharmony_ci \ 1058c2ecf20Sopenharmony_ci register_proc_cpuinfo_notifier(&fn##_nb); \ 1068c2ecf20Sopenharmony_ci}) 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_cistruct proc_cpuinfo_notifier_args { 1098c2ecf20Sopenharmony_ci struct seq_file *m; 1108c2ecf20Sopenharmony_ci unsigned long n; 1118c2ecf20Sopenharmony_ci}; 1128c2ecf20Sopenharmony_ci 1138c2ecf20Sopenharmony_cistatic inline bool cpus_are_siblings(int cpua, int cpub) 1148c2ecf20Sopenharmony_ci{ 1158c2ecf20Sopenharmony_ci struct cpuinfo_loongarch *infoa = &cpu_data[cpua]; 1168c2ecf20Sopenharmony_ci struct cpuinfo_loongarch *infob = &cpu_data[cpub]; 1178c2ecf20Sopenharmony_ci 1188c2ecf20Sopenharmony_ci if (infoa->package != infob->package) 1198c2ecf20Sopenharmony_ci return false; 1208c2ecf20Sopenharmony_ci 1218c2ecf20Sopenharmony_ci if (infoa->core != infob->core) 1228c2ecf20Sopenharmony_ci return false; 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_ci return true; 1258c2ecf20Sopenharmony_ci} 1268c2ecf20Sopenharmony_ci 1278c2ecf20Sopenharmony_cistatic inline unsigned long cpu_asid_mask(struct cpuinfo_loongarch *cpuinfo) 1288c2ecf20Sopenharmony_ci{ 1298c2ecf20Sopenharmony_ci return cpuinfo->asid_mask; 1308c2ecf20Sopenharmony_ci} 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_cistatic inline void set_cpu_asid_mask(struct cpuinfo_loongarch *cpuinfo, 1338c2ecf20Sopenharmony_ci unsigned long asid_mask) 1348c2ecf20Sopenharmony_ci{ 1358c2ecf20Sopenharmony_ci cpuinfo->asid_mask = asid_mask; 1368c2ecf20Sopenharmony_ci} 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci#endif /* __ASM_CPU_INFO_H */ 139