2009-05-04 19:25:34 +02:00
/*
* AMD64 class Memory Controller kernel module
*
* Copyright ( c ) 2009 SoftwareBitMaker .
2015-09-28 06:43:13 -05:00
* Copyright ( c ) 2009 - 15 Advanced Micro Devices , Inc .
2009-05-04 19:25:34 +02:00
*
* This file may be distributed under the terms of the
* GNU General Public License .
*/
# include <linux/module.h>
# include <linux/ctype.h>
# include <linux/init.h>
# include <linux/pci.h>
# include <linux/pci_ids.h>
# include <linux/slab.h>
# include <linux/mmzone.h>
# include <linux/edac.h>
2017-01-27 11:24:23 -06:00
# include <asm/cpu_device_id.h>
2009-04-27 19:46:08 +02:00
# include <asm/msr.h>
2016-10-29 15:16:34 -02:00
# include "edac_module.h"
2010-09-27 15:30:39 +02:00
# include "mce_amd.h"
2009-05-04 19:25:34 +02:00
2010-10-07 18:29:15 +02:00
# define amd64_info(fmt, arg...) \
edac_printk ( KERN_INFO , " amd64 " , fmt , # # arg )
# define amd64_warn(fmt, arg...) \
2016-12-01 11:35:07 +01:00
edac_printk ( KERN_WARNING , " amd64 " , " Warning: " fmt , # # arg )
2010-10-07 18:29:15 +02:00
# define amd64_err(fmt, arg...) \
2016-12-01 11:35:07 +01:00
edac_printk ( KERN_ERR , " amd64 " , " Error: " fmt , # # arg )
2010-10-07 18:29:15 +02:00
# define amd64_mc_warn(mci, fmt, arg...) \
edac_mc_chipset_printk ( mci , KERN_WARNING , " amd64 " , fmt , # # arg )
# define amd64_mc_err(mci, fmt, arg...) \
edac_mc_chipset_printk ( mci , KERN_ERR , " amd64 " , fmt , # # arg )
2009-05-04 19:25:34 +02:00
/*
* Throughout the comments in this code , the following terms are used :
*
* SysAddr , DramAddr , and InputAddr
*
* These terms come directly from the amd64 documentation
* ( AMD publication # 26094 ) . They are defined as follows :
*
* SysAddr :
* This is a physical address generated by a CPU core or a device
* doing DMA . If generated by a CPU core , a SysAddr is the result of
* a virtual to physical address translation by the CPU core ' s address
* translation mechanism ( MMU ) .
*
* DramAddr :
* A DramAddr is derived from a SysAddr by subtracting an offset that
* depends on which node the SysAddr maps to and whether the SysAddr
* is within a range affected by memory hoisting . The DRAM Base
* ( section 3.4 .4 .1 ) and DRAM Limit ( section 3.4 .4 .2 ) registers
* determine which node a SysAddr maps to .
*
* If the DRAM Hole Address Register ( DHAR ) is enabled and the SysAddr
* is within the range of addresses specified by this register , then
* a value x from the DHAR is subtracted from the SysAddr to produce a
* DramAddr . Here , x represents the base address for the node that
* the SysAddr maps to plus an offset due to memory hoisting . See
* section 3.4 .8 and the comments in amd64_get_dram_hole_info ( ) and
* sys_addr_to_dram_addr ( ) below for more information .
*
* If the SysAddr is not affected by the DHAR then a value y is
* subtracted from the SysAddr to produce a DramAddr . Here , y is the
* base address for the node that the SysAddr maps to . See section
* 3.4 .4 and the comments in sys_addr_to_dram_addr ( ) below for more
* information .
*
* InputAddr :
* A DramAddr is translated to an InputAddr before being passed to the
* memory controller for the node that the DramAddr is associated
* with . The memory controller then maps the InputAddr to a csrow .
* If node interleaving is not in use , then the InputAddr has the same
* value as the DramAddr . Otherwise , the InputAddr is produced by
* discarding the bits used for node interleaving from the DramAddr .
* See section 3.4 .4 for more information .
*
* The memory controller for a given node uses its DRAM CS Base and
* DRAM CS Mask registers to map an InputAddr to a csrow . See
* sections 3.5 .4 and 3.5 .5 for more information .
*/
2017-02-14 11:58:05 +01:00
# define EDAC_AMD64_VERSION "3.5.0"
2009-05-04 19:25:34 +02:00
# define EDAC_MOD_STR "amd64_edac"
/* Extended Model from CPUID, for CPU Revision numbers */
2009-10-21 13:44:36 +02:00
# define K8_REV_D 1
# define K8_REV_E 2
# define K8_REV_F 4
2009-05-04 19:25:34 +02:00
/* Hardware limit on ChipSelect rows per MC and processors per system */
2010-10-21 18:52:53 +02:00
# define NUM_CHIPSELECTS 8
# define DRAM_RANGES 8
2019-08-21 23:59:55 +00:00
# define NUM_CONTROLLERS 8
2009-05-04 19:25:34 +02:00
2009-11-03 15:29:26 +01:00
# define ON true
# define OFF false
2009-05-04 19:25:34 +02:00
/*
* PCI - defined configuration space registers
*/
2011-01-19 18:15:10 +01:00
# define PCI_DEVICE_ID_AMD_15H_NB_F1 0x1601
# define PCI_DEVICE_ID_AMD_15H_NB_F2 0x1602
2014-10-30 12:16:09 +01:00
# define PCI_DEVICE_ID_AMD_15H_M30H_NB_F1 0x141b
# define PCI_DEVICE_ID_AMD_15H_M30H_NB_F2 0x141c
# define PCI_DEVICE_ID_AMD_15H_M60H_NB_F1 0x1571
# define PCI_DEVICE_ID_AMD_15H_M60H_NB_F2 0x1572
2013-04-17 14:57:13 -05:00
# define PCI_DEVICE_ID_AMD_16H_NB_F1 0x1531
# define PCI_DEVICE_ID_AMD_16H_NB_F2 0x1532
2014-02-20 10:28:46 -06:00
# define PCI_DEVICE_ID_AMD_16H_M30H_NB_F1 0x1581
# define PCI_DEVICE_ID_AMD_16H_M30H_NB_F2 0x1582
2016-11-17 17:57:35 -05:00
# define PCI_DEVICE_ID_AMD_17H_DF_F0 0x1460
# define PCI_DEVICE_ID_AMD_17H_DF_F6 0x1466
2018-08-16 15:28:40 -04:00
# define PCI_DEVICE_ID_AMD_17H_M10H_DF_F0 0x15e8
# define PCI_DEVICE_ID_AMD_17H_M10H_DF_F6 0x15ee
2019-02-28 15:36:09 +00:00
# define PCI_DEVICE_ID_AMD_17H_M30H_DF_F0 0x1490
# define PCI_DEVICE_ID_AMD_17H_M30H_DF_F6 0x1496
2020-05-10 20:48:42 +00:00
# define PCI_DEVICE_ID_AMD_17H_M60H_DF_F0 0x1448
# define PCI_DEVICE_ID_AMD_17H_M60H_DF_F6 0x144e
2019-09-06 23:21:38 +00:00
# define PCI_DEVICE_ID_AMD_17H_M70H_DF_F0 0x1440
# define PCI_DEVICE_ID_AMD_17H_M70H_DF_F6 0x1446
2020-01-10 01:56:50 +00:00
# define PCI_DEVICE_ID_AMD_19H_DF_F0 0x1650
# define PCI_DEVICE_ID_AMD_19H_DF_F6 0x1656
2009-05-04 19:25:34 +02:00
/*
* Function 1 - Address Map
*/
2010-10-21 18:52:53 +02:00
# define DRAM_BASE_LO 0x40
# define DRAM_LIMIT_LO 0x44
2013-08-09 11:54:49 -05:00
/*
* F15 M30h D18F1x2 [ 1 C : 00 ]
*/
# define DRAM_CONT_BASE 0x200
# define DRAM_CONT_LIMIT 0x204
/*
* F15 M30h D18F1x2 [ 4 C : 40 ]
*/
# define DRAM_CONT_HIGH_OFF 0x240
2011-02-21 19:33:10 +01:00
# define dram_rw(pvt, i) ((u8)(pvt->ranges[i].base.lo & 0x3))
# define dram_intlv_sel(pvt, i) ((u8)((pvt->ranges[i].lim.lo >> 8) & 0x7))
# define dram_dst_node(pvt, i) ((u8)(pvt->ranges[i].lim.lo & 0x7))
2010-10-21 18:52:53 +02:00
2010-11-11 17:29:13 +01:00
# define DHAR 0xf0
2010-12-10 19:49:19 +01:00
# define dhar_mem_hoist_valid(pvt) ((pvt)->dhar & BIT(1))
# define dhar_base(pvt) ((pvt)->dhar & 0xff000000)
# define k8_dhar_offset(pvt) (((pvt)->dhar & 0x0000ff00) << 16)
2009-05-04 19:25:34 +02:00
/* NOTE: Extra mask bit vs K8 */
2010-12-10 19:49:19 +01:00
# define f10_dhar_offset(pvt) (((pvt)->dhar & 0x0000ff80) << 16)
2009-05-04 19:25:34 +02:00
2010-10-08 18:32:29 +02:00
# define DCT_CFG_SEL 0x10C
2009-05-04 19:25:34 +02:00
2011-03-30 15:42:10 +02:00
# define DRAM_LOCAL_NODE_BASE 0x120
2011-03-21 20:45:06 +01:00
# define DRAM_LOCAL_NODE_LIM 0x124
2010-10-21 18:52:53 +02:00
# define DRAM_BASE_HI 0x140
# define DRAM_LIMIT_HI 0x144
2009-05-04 19:25:34 +02:00
/*
* Function 2 - DRAM controller
*/
2010-11-29 19:49:02 +01:00
# define DCSB0 0x40
# define DCSB1 0x140
# define DCSB_CS_ENABLE BIT(0)
2009-05-04 19:25:34 +02:00
2010-11-29 19:49:02 +01:00
# define DCSM0 0x60
# define DCSM1 0x160
2009-05-04 19:25:34 +02:00
2019-08-22 00:00:02 +00:00
# define csrow_enabled(i, dct, pvt) ((pvt)->csels[(dct)].csbases[(i)] & DCSB_CS_ENABLE)
# define csrow_sec_enabled(i, dct, pvt) ((pvt)->csels[(dct)].csbases_sec[(i)] & DCSB_CS_ENABLE)
2009-05-04 19:25:34 +02:00
2014-10-30 12:16:09 +01:00
# define DRAM_CONTROL 0x78
2009-05-04 19:25:34 +02:00
# define DBAM0 0x80
# define DBAM1 0x180
/* Extract the DIMM 'type' on the i'th DIMM from the DBAM reg value passed */
2012-09-12 18:16:01 +02:00
# define DBAM_DIMM(i, reg) ((((reg) >> (4*(i)))) & 0xF)
2009-05-04 19:25:34 +02:00
# define DBAM_MAX_VALUE 11
2010-12-22 14:28:24 +01:00
# define DCLR0 0x90
# define DCLR1 0x190
2009-05-04 19:25:34 +02:00
# define REVE_WIDTH_128 BIT(16)
2011-01-18 19:16:08 +01:00
# define WIDTH_128 BIT(11)
2009-05-04 19:25:34 +02:00
2010-12-22 14:28:24 +01:00
# define DCHR0 0x94
# define DCHR1 0x194
2009-10-21 13:44:36 +02:00
# define DDR3_MODE BIT(8)
2009-05-04 19:25:34 +02:00
2010-12-22 19:31:45 +01:00
# define DCT_SEL_LO 0x110
# define dct_high_range_enabled(pvt) ((pvt)->dct_sel_lo & BIT(0))
# define dct_interleave_enabled(pvt) ((pvt)->dct_sel_lo & BIT(2))
2010-12-22 14:28:24 +01:00
2010-12-22 19:31:45 +01:00
# define dct_ganging_enabled(pvt) ((boot_cpu_data.x86 == 0x10) && ((pvt)->dct_sel_lo & BIT(4)))
2010-12-22 14:28:24 +01:00
2010-12-22 19:31:45 +01:00
# define dct_data_intlv_enabled(pvt) ((pvt)->dct_sel_lo & BIT(5))
# define dct_memory_cleared(pvt) ((pvt)->dct_sel_lo & BIT(10))
2009-05-04 19:25:34 +02:00
2011-01-11 22:08:07 +01:00
# define SWAP_INTLV_REG 0x10c
2010-12-22 19:31:45 +01:00
# define DCT_SEL_HI 0x114
2009-05-04 19:25:34 +02:00
2015-09-28 06:43:12 -05:00
# define F15H_M60H_SCRCTRL 0x1C8
2016-11-17 17:57:42 -05:00
# define F17H_SCR_BASE_ADDR 0x48
# define F17H_SCR_LIMIT_ADDR 0x4C
2015-09-28 06:43:12 -05:00
2009-05-04 19:25:34 +02:00
/*
* Function 3 - Misc Control
*/
2010-12-22 19:48:20 +01:00
# define NBCTL 0x40
2009-05-04 19:25:34 +02:00
2010-12-23 14:07:18 +01:00
# define NBCFG 0x44
# define NBCFG_CHIPKILL BIT(23)
# define NBCFG_ECC_ENABLE BIT(22)
2009-05-04 19:25:34 +02:00
2011-01-07 16:26:49 +01:00
/* F3x48: NBSL */
2009-05-04 19:25:34 +02:00
# define F10_NBSL_EXT_ERR_ECC 0x8
2011-01-07 16:26:49 +01:00
# define NBSL_PP_OBS 0x2
2009-05-04 19:25:34 +02:00
2011-01-07 16:26:49 +01:00
# define SCRCTRL 0x58
2009-05-04 19:25:34 +02:00
# define F10_ONLINE_SPARE 0xB0
2011-01-13 18:02:22 +01:00
# define online_spare_swap_done(pvt, c) (((pvt)->online_spare >> (1 + 2 * (c))) & 0x1)
# define online_spare_bad_dramcs(pvt, c) (((pvt)->online_spare >> (4 + 4 * (c))) & 0x7)
2009-05-04 19:25:34 +02:00
# define F10_NB_ARRAY_ADDR 0xB8
2012-08-09 18:23:53 +02:00
# define F10_NB_ARRAY_DRAM BIT(31)
2009-05-04 19:25:34 +02:00
/* Bits [2:1] are used to select 16-byte section within a 64-byte cacheline */
2012-08-09 18:23:53 +02:00
# define SET_NB_ARRAY_ADDR(section) (((section) & 0x3) << 1)
2009-05-04 19:25:34 +02:00
# define F10_NB_ARRAY_DATA 0xBC
2012-08-09 18:41:07 +02:00
# define F10_NB_ARR_ECC_WR_REQ BIT(17)
2012-08-09 18:23:53 +02:00
# define SET_NB_DRAM_INJECTION_WRITE(inj) \
( BIT ( ( ( inj . word ) & 0xF ) + 20 ) | \
2012-08-09 18:41:07 +02:00
F10_NB_ARR_ECC_WR_REQ | inj . bit_map )
2012-08-09 18:23:53 +02:00
# define SET_NB_DRAM_INJECTION_READ(inj) \
( BIT ( ( ( inj . word ) & 0xF ) + 20 ) | \
BIT ( 16 ) | inj . bit_map )
2009-05-04 19:25:34 +02:00
2011-01-07 16:26:49 +01:00
# define NBCAP 0xE8
# define NBCAP_CHIPKILL BIT(4)
# define NBCAP_SECDED BIT(3)
# define NBCAP_DCT_DUAL BIT(0)
2009-05-04 19:25:34 +02:00
2010-03-09 12:46:00 +01:00
# define EXT_NB_MCA_CFG 0x180
2009-11-03 15:29:26 +01:00
/* MSRs */
2011-01-07 16:26:49 +01:00
# define MSR_MCGCTL_NBE BIT(4)
2009-05-04 19:25:34 +02:00
2016-11-17 17:57:37 -05:00
/* F17h */
/* F0: */
# define DF_DHAR 0x104
2016-11-17 17:57:34 -05:00
/* UMC CH register offsets */
2016-11-17 17:57:37 -05:00
# define UMCCH_BASE_ADDR 0x0
2019-08-22 00:00:01 +00:00
# define UMCCH_BASE_ADDR_SEC 0x10
2016-11-17 17:57:37 -05:00
# define UMCCH_ADDR_MASK 0x20
2019-08-22 00:00:01 +00:00
# define UMCCH_ADDR_MASK_SEC 0x28
2016-11-28 08:50:21 -06:00
# define UMCCH_ADDR_CFG 0x30
2016-11-17 17:57:37 -05:00
# define UMCCH_DIMM_CFG 0x80
2016-11-28 08:50:21 -06:00
# define UMCCH_UMC_CFG 0x100
2016-11-17 17:57:34 -05:00
# define UMCCH_SDP_CTRL 0x104
2016-11-17 17:57:37 -05:00
# define UMCCH_ECC_CTRL 0x14C
2016-11-28 08:50:21 -06:00
# define UMCCH_ECC_BAD_SYMBOL 0xD90
# define UMCCH_UMC_CAP 0xDF0
2016-11-17 17:57:34 -05:00
# define UMCCH_UMC_CAP_HI 0xDF4
/* UMC CH bitfields */
2016-11-17 17:57:37 -05:00
# define UMC_ECC_CHIPKILL_CAP BIT(31)
2016-11-17 17:57:34 -05:00
# define UMC_ECC_ENABLED BIT(30)
2016-11-17 17:57:37 -05:00
2016-11-17 17:57:34 -05:00
# define UMC_SDP_INIT BIT(31)
2010-10-08 18:32:29 +02:00
enum amd_families {
2009-05-04 19:25:34 +02:00
K8_CPUS = 0 ,
F10_CPUS ,
2010-10-08 18:32:29 +02:00
F15_CPUS ,
2013-08-09 11:54:49 -05:00
F15_M30H_CPUS ,
2014-10-30 12:16:09 +01:00
F15_M60H_CPUS ,
2013-04-17 14:57:13 -05:00
F16_CPUS ,
2014-02-20 10:28:46 -06:00
F16_M30H_CPUS ,
2016-11-17 17:57:35 -05:00
F17_CPUS ,
2018-08-16 15:28:40 -04:00
F17_M10H_CPUS ,
2019-02-28 15:36:09 +00:00
F17_M30H_CPUS ,
2020-05-10 20:48:42 +00:00
F17_M60H_CPUS ,
2019-09-06 23:21:38 +00:00
F17_M70H_CPUS ,
2020-01-10 01:56:50 +00:00
F19_CPUS ,
2010-10-08 18:32:29 +02:00
NUM_FAMILIES ,
2009-05-04 19:25:34 +02:00
} ;
/* Error injection control structure */
struct error_injection {
2012-08-09 18:41:07 +02:00
u32 section ;
u32 word ;
u32 bit_map ;
2009-05-04 19:25:34 +02:00
} ;
2010-10-21 18:52:53 +02:00
/* low and high part of PCI config space regs */
struct reg_pair {
u32 lo , hi ;
} ;
/*
* See F1x [ 1 , 0 ] [ 7 C : 40 ] DRAM Base / Limit Registers
*/
struct dram_range {
struct reg_pair base ;
struct reg_pair lim ;
} ;
2010-11-29 19:49:02 +01:00
/* A DCT chip selects collection */
struct chip_select {
u32 csbases [ NUM_CHIPSELECTS ] ;
2019-08-22 00:00:01 +00:00
u32 csbases_sec [ NUM_CHIPSELECTS ] ;
2010-11-29 19:49:02 +01:00
u8 b_cnt ;
u32 csmasks [ NUM_CHIPSELECTS ] ;
2019-08-22 00:00:01 +00:00
u32 csmasks_sec [ NUM_CHIPSELECTS ] ;
2010-11-29 19:49:02 +01:00
u8 m_cnt ;
} ;
2016-11-17 17:57:35 -05:00
struct amd64_umc {
2016-11-17 17:57:37 -05:00
u32 dimm_cfg ; /* DIMM Configuration reg */
2016-11-28 08:50:21 -06:00
u32 umc_cfg ; /* Configuration reg */
2016-11-17 17:57:35 -05:00
u32 sdp_ctrl ; /* SDP Control reg */
2016-11-17 17:57:37 -05:00
u32 ecc_ctrl ; /* DRAM ECC Control reg */
2016-11-28 08:50:21 -06:00
u32 umc_cap_hi ; /* Capabilities High reg */
2016-11-17 17:57:35 -05:00
} ;
2009-05-04 19:25:34 +02:00
struct amd64_pvt {
2010-10-01 19:35:38 +02:00
struct low_ops * ops ;
2009-05-04 19:25:34 +02:00
/* pci_device handles which we utilize */
2016-11-17 17:57:36 -05:00
struct pci_dev * F0 , * F1 , * F2 , * F3 , * F6 ;
2009-05-04 19:25:34 +02:00
2012-11-30 16:44:20 +08:00
u16 mc_node_id ; /* MC index of this MC node */
2013-08-09 11:54:49 -05:00
u8 fam ; /* CPU family */
2013-08-10 13:54:48 +02:00
u8 model ; /* ... model */
u8 stepping ; /* ... stepping */
2009-05-04 19:25:34 +02:00
int ext_model ; /* extended model value of this node */
int channel_count ;
/* Raw registers */
u32 dclr0 ; /* DRAM Configuration Low DCT0 reg */
u32 dclr1 ; /* DRAM Configuration Low DCT1 reg */
u32 dchr0 ; /* DRAM Configuration High DCT0 reg */
u32 dchr1 ; /* DRAM Configuration High DCT1 reg */
u32 nbcap ; /* North Bridge Capabilities */
u32 nbcfg ; /* F10 North Bridge Configuration */
u32 ext_nbcfg ; /* Extended F10 North Bridge Configuration */
u32 dhar ; /* DRAM Hoist reg */
u32 dbam0 ; /* DRAM Base Address Mapping reg for DCT0 */
u32 dbam1 ; /* DRAM Base Address Mapping reg for DCT1 */
2019-08-21 23:59:55 +00:00
/* one for each DCT/UMC */
struct chip_select csels [ NUM_CONTROLLERS ] ;
2009-05-04 19:25:34 +02:00
2010-10-21 18:52:53 +02:00
/* DRAM base and limit pairs F1x[78,70,68,60,58,50,48,40] */
struct dram_range ranges [ DRAM_RANGES ] ;
2009-05-04 19:25:34 +02:00
u64 top_mem ; /* top of memory below 4GB */
u64 top_mem2 ; /* top of memory above 4GB */
2010-12-22 19:31:45 +01:00
u32 dct_sel_lo ; /* DRAM Controller Select Low */
u32 dct_sel_hi ; /* DRAM Controller Select High */
2010-10-08 18:32:29 +02:00
u32 online_spare ; /* On-Line spare Reg */
2009-05-04 19:25:34 +02:00
2019-02-28 15:36:11 +00:00
/* x4, x8, or x16 syndromes in use */
2011-01-19 20:35:12 +01:00
u8 ecc_sym_sz ;
2010-03-09 12:46:00 +01:00
2009-05-04 19:25:34 +02:00
/* place to store error injection parameters prior to issue */
struct error_injection injection ;
2014-10-30 12:16:09 +01:00
/* cache the dram_type */
enum mem_type dram_type ;
2016-11-17 17:57:35 -05:00
struct amd64_umc * umc ; /* UMC registers */
2010-10-14 16:01:30 +02:00
} ;
2012-08-30 18:01:36 +02:00
enum err_codes {
DECODE_OK = 0 ,
ERR_NODE = - 1 ,
ERR_CSROW = - 2 ,
ERR_CHANNEL = - 3 ,
2016-11-28 12:59:53 -06:00
ERR_SYND = - 4 ,
ERR_NORM_ADDR = - 5 ,
2012-08-30 18:01:36 +02:00
} ;
struct err_info {
int err_code ;
struct mem_ctl_info * src_mci ;
int csrow ;
int channel ;
u16 syndrome ;
u32 page ;
u32 offset ;
} ;
2016-11-17 17:57:34 -05:00
static inline u32 get_umc_base ( u8 channel )
{
2019-02-28 15:36:10 +00:00
/* chY: 0xY50000 */
return 0x50000 + ( channel < < 20 ) ;
2016-11-17 17:57:34 -05:00
}
2012-11-30 16:44:20 +08:00
static inline u64 get_dram_base ( struct amd64_pvt * pvt , u8 i )
2010-10-21 18:52:53 +02:00
{
u64 addr = ( ( u64 ) pvt - > ranges [ i ] . base . lo & 0xffff0000 ) < < 8 ;
if ( boot_cpu_data . x86 = = 0xf )
return addr ;
return ( ( ( u64 ) pvt - > ranges [ i ] . base . hi & 0x000000ff ) < < 40 ) | addr ;
}
2012-11-30 16:44:20 +08:00
static inline u64 get_dram_limit ( struct amd64_pvt * pvt , u8 i )
2010-10-21 18:52:53 +02:00
{
u64 lim = ( ( ( u64 ) pvt - > ranges [ i ] . lim . lo & 0xffff0000 ) < < 8 ) | 0x00ffffff ;
if ( boot_cpu_data . x86 = = 0xf )
return lim ;
return ( ( ( u64 ) pvt - > ranges [ i ] . lim . hi & 0x000000ff ) < < 40 ) | lim ;
}
2011-01-10 14:24:32 +01:00
static inline u16 extract_syndrome ( u64 status )
{
return ( ( status > > 47 ) & 0xff ) | ( ( status > > 16 ) & 0xff00 ) ;
}
2013-08-09 11:54:49 -05:00
static inline u8 dct_sel_interleave_addr ( struct amd64_pvt * pvt )
{
if ( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 )
return ( ( ( pvt - > dct_sel_hi > > 9 ) & 0x1 ) < < 2 ) |
( ( pvt - > dct_sel_lo > > 6 ) & 0x3 ) ;
return ( ( pvt ) - > dct_sel_lo > > 6 ) & 0x3 ;
}
2010-10-14 16:01:30 +02:00
/*
* per - node ECC settings descriptor
*/
struct ecc_settings {
u32 old_nbctl ;
bool nbctl_valid ;
2009-05-04 19:25:34 +02:00
struct flags {
2010-02-24 14:49:47 +01:00
unsigned long nb_mce_enable : 1 ;
unsigned long nb_ecc_prev : 1 ;
2009-05-04 19:25:34 +02:00
} flags ;
} ;
/*
* Each of the PCI Device IDs types have their own set of hardware accessor
* functions and per device encoding / decoding logic .
*/
struct low_ops {
2009-10-21 13:44:36 +02:00
int ( * early_channel_count ) ( struct amd64_pvt * pvt ) ;
2011-01-10 14:24:32 +01:00
void ( * map_sysaddr_to_csrow ) ( struct mem_ctl_info * mci , u64 sys_addr ,
2012-08-30 18:01:36 +02:00
struct err_info * ) ;
2014-10-30 12:16:09 +01:00
int ( * dbam_to_cs ) ( struct amd64_pvt * pvt , u8 dct ,
unsigned cs_mode , int cs_mask_nr ) ;
2009-05-04 19:25:34 +02:00
} ;
struct amd64_family_type {
const char * ctl_name ;
2016-11-17 17:57:35 -05:00
u16 f0_id , f1_id , f2_id , f6_id ;
2019-10-22 20:35:11 +00:00
/* Maximum number of memory controllers per die/node. */
u8 max_mcs ;
2009-05-04 19:25:34 +02:00
struct low_ops ops ;
} ;
2012-08-09 18:41:07 +02:00
int __amd64_read_pci_cfg_dword ( struct pci_dev * pdev , int offset ,
u32 * val , const char * func ) ;
2010-10-08 18:32:29 +02:00
int __amd64_write_pci_cfg_dword ( struct pci_dev * pdev , int offset ,
u32 val , const char * func ) ;
2009-10-13 19:26:55 +02:00
2010-10-08 18:32:29 +02:00
# define amd64_read_pci_cfg(pdev, offset, val) \
__amd64_read_pci_cfg_dword ( pdev , offset , val , __func__ )
2009-10-13 19:26:55 +02:00
2010-10-08 18:32:29 +02:00
# define amd64_write_pci_cfg(pdev, offset, val) \
__amd64_write_pci_cfg_dword ( pdev , offset , val , __func__ )
2009-10-13 19:26:55 +02:00
2012-03-21 14:00:44 -03:00
# define to_mci(k) container_of(k, struct mem_ctl_info, dev)
2012-08-09 18:41:07 +02:00
/* Injection helpers */
static inline void disable_caches ( void * dummy )
{
write_cr0 ( read_cr0 ( ) | X86_CR0_CD ) ;
wbinvd ( ) ;
}
static inline void enable_caches ( void * dummy )
{
write_cr0 ( read_cr0 ( ) & ~ X86_CR0_CD ) ;
}
2013-08-09 11:54:49 -05:00
static inline u8 dram_intlv_en ( struct amd64_pvt * pvt , unsigned int i )
{
if ( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 ) {
u32 tmp ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_CONT_LIMIT , & tmp ) ;
return ( u8 ) tmp & 0xF ;
}
return ( u8 ) ( pvt - > ranges [ i ] . base . lo > > 8 ) & 0x7 ;
}
static inline u8 dhar_valid ( struct amd64_pvt * pvt )
{
if ( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 ) {
u32 tmp ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_CONT_BASE , & tmp ) ;
return ( tmp > > 1 ) & BIT ( 0 ) ;
}
return ( pvt ) - > dhar & BIT ( 0 ) ;
}
static inline u32 dct_sel_baseaddr ( struct amd64_pvt * pvt )
{
if ( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 ) {
u32 tmp ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_CONT_BASE , & tmp ) ;
return ( tmp > > 11 ) & 0x1FFF ;
}
return ( pvt ) - > dct_sel_lo & 0xFFFFF800 ;
}