2019-01-30 11:15:17 -08:00
/* SPDX-License-Identifier: GPL-2.0 */
/*
* Common codes for both the skx_edac driver and Intel 10 nm server EDAC driver .
* Originally split out from the skx_edac driver .
*
* Copyright ( c ) 2018 , Intel Corporation .
*/
# ifndef _SKX_COMM_EDAC_H
# define _SKX_COMM_EDAC_H
2021-06-11 10:01:18 -07:00
# include <linux/bits.h>
2019-01-30 11:15:17 -08:00
# define MSG_SIZE 1024
/*
* Debug macros
*/
# define skx_printk(level, fmt, arg...) \
edac_printk ( level , " skx " , fmt , # # arg )
# define skx_mc_printk(mci, level, fmt, arg...) \
edac_mc_chipset_printk ( mci , level , " skx " , fmt , # # arg )
/*
* Get a bit field at register value < v > , from bit < lo > to bit < hi >
*/
# define GET_BITFIELD(v, lo, hi) \
( ( ( v ) & GENMASK_ULL ( ( hi ) , ( lo ) ) ) > > ( lo ) )
# define SKX_NUM_IMC 2 /* Memory controllers per socket */
# define SKX_NUM_CHANNELS 3 /* Channels per memory controller */
# define SKX_NUM_DIMMS 2 /* Max DIMMS per channel */
2021-06-11 10:01:20 -07:00
# define I10NM_NUM_DDR_IMC 4
# define I10NM_NUM_DDR_CHANNELS 2
# define I10NM_NUM_DDR_DIMMS 2
# define I10NM_NUM_HBM_IMC 16
# define I10NM_NUM_HBM_CHANNELS 2
# define I10NM_NUM_HBM_DIMMS 1
# define I10NM_NUM_IMC (I10NM_NUM_DDR_IMC + I10NM_NUM_HBM_IMC)
# define I10NM_NUM_CHANNELS MAX(I10NM_NUM_DDR_CHANNELS, I10NM_NUM_HBM_CHANNELS)
# define I10NM_NUM_DIMMS MAX(I10NM_NUM_DDR_DIMMS, I10NM_NUM_HBM_DIMMS)
2019-01-30 11:15:17 -08:00
# define MAX(a, b) ((a) > (b) ? (a) : (b))
# define NUM_IMC MAX(SKX_NUM_IMC, I10NM_NUM_IMC)
# define NUM_CHANNELS MAX(SKX_NUM_CHANNELS, I10NM_NUM_CHANNELS)
# define NUM_DIMMS MAX(SKX_NUM_DIMMS, I10NM_NUM_DIMMS)
# define IS_DIMM_PRESENT(r) GET_BITFIELD(r, 15, 15)
# define IS_NVDIMM_PRESENT(r, i) GET_BITFIELD(r, i, i)
/*
* Each cpu socket contains some pci devices that provide global
* information , and also some that are local to each of the two
* memory controllers on the die .
*/
struct skx_dev {
struct list_head list ;
u8 bus [ 4 ] ;
int seg ;
struct pci_dev * sad_all ;
struct pci_dev * util_all ;
struct pci_dev * uracu ; /* for i10nm CPU */
2021-06-11 10:01:20 -07:00
struct pci_dev * pcu_cr3 ; /* for HBM memory detection */
2019-01-30 11:15:17 -08:00
u32 mcroute ;
struct skx_imc {
struct mem_ctl_info * mci ;
struct pci_dev * mdev ; /* for i10nm CPU */
void __iomem * mbase ; /* for i10nm CPU */
2020-11-17 20:49:53 +08:00
int chan_mmio_sz ; /* for i10nm CPU */
2021-06-11 10:01:20 -07:00
int num_channels ; /* channels per memory controller */
int num_dimms ; /* dimms per channel */
bool hbm_mc ;
2019-01-30 11:15:17 -08:00
u8 mc ; /* system wide mc# */
u8 lmc ; /* socket relative mc# */
u8 src_id , node_id ;
struct skx_channel {
struct pci_dev * cdev ;
2019-08-15 14:53:28 -07:00
struct pci_dev * edev ;
2021-08-18 10:57:01 -07:00
u32 retry_rd_err_log_s ;
u32 retry_rd_err_log_d ;
2019-01-30 11:15:17 -08:00
struct skx_dimm {
u8 close_pg ;
u8 bank_xor_enable ;
u8 fine_grain_bank ;
u8 rowbits ;
u8 colbits ;
} dimms [ NUM_DIMMS ] ;
} chan [ NUM_CHANNELS ] ;
} imc [ NUM_IMC ] ;
} ;
struct skx_pvt {
struct skx_imc * imc ;
} ;
enum type {
SKX ,
2020-11-17 20:49:53 +08:00
I10NM ,
SPR
2019-01-30 11:15:17 -08:00
} ;
enum {
INDEX_SOCKET ,
INDEX_MEMCTRL ,
INDEX_CHANNEL ,
INDEX_DIMM ,
2021-06-11 10:01:18 -07:00
INDEX_NM_FIRST ,
INDEX_NM_MEMCTRL = INDEX_NM_FIRST ,
INDEX_NM_CHANNEL ,
INDEX_NM_DIMM ,
2019-01-30 11:15:17 -08:00
INDEX_MAX
} ;
2021-06-11 10:01:18 -07:00
# define BIT_NM_MEMCTRL BIT_ULL(INDEX_NM_MEMCTRL)
# define BIT_NM_CHANNEL BIT_ULL(INDEX_NM_CHANNEL)
# define BIT_NM_DIMM BIT_ULL(INDEX_NM_DIMM)
2019-01-30 11:15:17 -08:00
struct decoded_addr {
struct skx_dev * dev ;
u64 addr ;
int socket ;
int imc ;
int channel ;
u64 chan_addr ;
int sktways ;
int chanways ;
int dimm ;
int rank ;
int channel_rank ;
u64 rank_address ;
int row ;
int column ;
int bank_address ;
int bank_group ;
} ;
2020-04-24 20:18:14 +08:00
struct res_config {
enum type type ;
/* Configuration agent device ID */
unsigned int decs_did ;
/* Default bus number configuration register offset */
int busno_cfg_offset ;
2020-11-17 20:49:53 +08:00
/* Per DDR channel memory-mapped I/O size */
int ddr_chan_mmio_sz ;
2021-06-11 10:01:20 -07:00
/* Per HBM channel memory-mapped I/O size */
int hbm_chan_mmio_sz ;
2020-11-17 20:49:53 +08:00
bool support_ddr5 ;
2021-06-11 10:01:19 -07:00
/* SAD device number and function number */
unsigned int sad_all_devfn ;
int sad_all_offset ;
2021-08-18 10:57:01 -07:00
/* Offsets of retry_rd_err_log registers */
u32 * offsets_scrub ;
u32 * offsets_demand ;
2020-04-24 20:18:14 +08:00
} ;
2020-11-17 20:49:53 +08:00
typedef int ( * get_dimm_config_f ) ( struct mem_ctl_info * mci ,
struct res_config * cfg ) ;
2019-01-30 11:15:17 -08:00
typedef bool ( * skx_decode_f ) ( struct decoded_addr * res ) ;
2021-08-18 10:57:01 -07:00
typedef void ( * skx_show_retry_log_f ) ( struct decoded_addr * res , char * msg , int len , bool scrub_err ) ;
2019-01-30 11:15:17 -08:00
int __init skx_adxl_get ( void ) ;
void __exit skx_adxl_put ( void ) ;
2019-08-15 14:53:28 -07:00
void skx_set_decode ( skx_decode_f decode , skx_show_retry_log_f show_retry_log ) ;
2021-06-11 10:01:18 -07:00
void skx_set_mem_cfg ( bool mem_cfg_2lm ) ;
2019-01-30 11:15:17 -08:00
2019-06-26 14:16:55 +08:00
int skx_get_src_id ( struct skx_dev * d , int off , u8 * id ) ;
2019-01-30 11:15:17 -08:00
int skx_get_node_id ( struct skx_dev * d , u8 * id ) ;
2020-04-24 20:18:14 +08:00
int skx_get_all_bus_mappings ( struct res_config * cfg , struct list_head * * list ) ;
2019-01-30 11:15:17 -08:00
int skx_get_hi_lo ( unsigned int did , int off [ ] , u64 * tolm , u64 * tohm ) ;
2020-05-15 20:34:06 +08:00
int skx_get_dimm_info ( u32 mtr , u32 mcmtr , u32 amap , struct dimm_info * dimm ,
2020-11-17 20:49:53 +08:00
struct skx_imc * imc , int chan , int dimmno ,
struct res_config * cfg ) ;
2019-01-30 11:15:17 -08:00
int skx_get_nvdimm_info ( struct dimm_info * dimm , struct skx_imc * imc ,
int chan , int dimmno , const char * mod_str ) ;
int skx_register_mci ( struct skx_imc * imc , struct pci_dev * pdev ,
const char * ctl_name , const char * mod_str ,
2020-11-17 20:49:53 +08:00
get_dimm_config_f get_dimm_config ,
struct res_config * cfg ) ;
2019-01-30 11:15:17 -08:00
int skx_mce_check_error ( struct notifier_block * nb , unsigned long val ,
void * data ) ;
void skx_remove ( void ) ;
# endif /* _SKX_COMM_EDAC_H */