2009-05-04 19:25:34 +02:00
/*
* AMD64 class Memory Controller kernel module
*
* Copyright ( c ) 2009 SoftwareBitMaker .
* Copyright ( c ) 2009 Advanced Micro Devices , Inc .
*
* This file may be distributed under the terms of the
* GNU General Public License .
*
* Originally Written by Thayne Harbaugh
*
* Changes by Douglas " norsk " Thompson < dougthompson @ xmission . com > :
* - K8 CPU Revision D and greater support
*
* Changes by Dave Peterson < dsp @ llnl . gov > < dave_peterson @ pobox . com > :
* - Module largely rewritten , with new ( and hopefully correct )
* code for dealing with node and chip select interleaving ,
* various code cleanup , and bug fixes
* - Added support for memory hoisting using DRAM hole address
* register
*
* Changes by Douglas " norsk " Thompson < dougthompson @ xmission . com > :
* - K8 Rev ( 1207 ) revision support added , required Revision
* specific mini - driver code to support Rev F as well as
* prior revisions
*
* Changes by Douglas " norsk " Thompson < dougthompson @ xmission . com > :
* - Family 10 h revision support added . New PCI Device IDs ,
* indicating new changes . Actual registers modified
* were slight , less than the Rev E to Rev F transition
* but changing the PCI Device ID was the proper thing to
* do , as it provides for almost automactic family
* detection . The mods to Rev F required more family
* information detection .
*
* Changes / Fixes by Borislav Petkov < borislav . petkov @ amd . com > :
* - misc fixes and code cleanups
*
* This module is based on the following documents
* ( available from http : //www.amd.com/):
*
* Title : BIOS and Kernel Developer ' s Guide for AMD Athlon 64 and AMD
* Opteron Processors
* AMD publication # : 26094
* ` Revision : 3.26
*
* Title : BIOS and Kernel Developer ' s Guide for AMD NPT Family 0F h
* Processors
* AMD publication # : 32559
* Revision : 3.00
* Issue Date : May 2006
*
* Title : BIOS and Kernel Developer ' s Guide ( BKDG ) For AMD Family 10 h
* Processors
* AMD publication # : 31116
* Revision : 3.00
* Issue Date : September 07 , 2007
*
* Sections in the first 2 documents are no longer in sync with each other .
* The Family 10 h BKDG was totally re - written from scratch with a new
* presentation model .
* Therefore , comments that refer to a Document section might be off .
*/
# 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>
2009-04-27 19:46:08 +02:00
# include <asm/msr.h>
2009-05-04 19:25:34 +02:00
# include "edac_core.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_debug(fmt, arg...) \
edac_printk ( KERN_DEBUG , " amd64 " , fmt , # # arg )
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_notice(fmt, arg...) \
edac_printk ( KERN_NOTICE , " amd64 " , fmt , # # arg )
# define amd64_warn(fmt, arg...) \
edac_printk ( KERN_WARNING , " amd64 " , fmt , # # arg )
# define amd64_err(fmt, arg...) \
edac_printk ( KERN_ERR , " amd64 " , fmt , # # arg )
# 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 .
*/
2011-01-19 18:15:10 +01:00
# define EDAC_AMD64_VERSION "3.4.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
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
2010-11-29 19:49:02 +01:00
/*
* Create a contiguous bitmask starting at bit position @ lo and ending at
* position @ hi . For example
*
* GENMASK ( 21 , 39 ) gives us the 64 bit vector 0x000000ffffe00000 .
*/
# define GENMASK(lo, hi) (((1ULL << ((hi) - (lo) + 1)) - 1) << (lo))
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
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
2011-02-21 19:33:10 +01:00
# define dram_intlv_en(pvt, i) ((u8)((pvt->ranges[i].base.lo >> 8) & 0x7))
# 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_valid(pvt) ((pvt)->dhar & BIT(0))
# 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
2010-11-29 19:49:02 +01:00
# define csrow_enabled(i, dct, pvt) ((pvt)->csels[(dct)].csbases[(i)] & DCSB_CS_ENABLE)
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 */
# define DBAM_DIMM(i, reg) ((((reg) >> (4*i))) & 0xF)
# 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_sel_baseaddr(pvt) ((pvt)->dct_sel_lo & 0xFFFFF800)
# define dct_sel_interleave_addr(pvt) (((pvt)->dct_sel_lo >> 6) & 0x3)
# 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
/*
* 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
2011-01-07 16:26:49 +01:00
# define F10_NB_ARRAY_DRAM_ECC 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 */
# define SET_NB_ARRAY_ADDRESS(section) (((section) & 0x3) << 1)
# define F10_NB_ARRAY_DATA 0xBC
# define SET_NB_DRAM_INJECTION_WRITE(word, bits) \
( BIT ( ( ( word ) & 0xF ) + 20 ) | \
2009-09-24 11:05:30 +02:00
BIT ( 17 ) | bits )
2009-05-04 19:25:34 +02:00
# define SET_NB_DRAM_INJECTION_READ(word, bits) \
( BIT ( ( ( word ) & 0xF ) + 20 ) | \
2009-09-24 11:05:30 +02:00
BIT ( 16 ) | bits )
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
/* AMD sets the first MC device at device ID 0x18. */
2011-02-21 18:55:00 +01:00
static inline u8 get_node_id ( struct pci_dev * pdev )
2009-05-04 19:25:34 +02:00
{
return PCI_SLOT ( pdev - > devfn ) - 0x18 ;
}
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 ,
NUM_FAMILIES ,
2009-05-04 19:25:34 +02:00
} ;
/* Error injection control structure */
struct error_injection {
u32 section ;
u32 word ;
u32 bit_map ;
} ;
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 ] ;
u8 b_cnt ;
u32 csmasks [ NUM_CHIPSELECTS ] ;
u8 m_cnt ;
} ;
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 */
2010-10-01 20:11:07 +02:00
struct pci_dev * F1 , * F2 , * F3 ;
2009-05-04 19:25:34 +02:00
2011-02-21 18:55:00 +01:00
unsigned mc_node_id ; /* MC index of this MC node */
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 */
2010-11-29 19:49:02 +01:00
/* one for each DCT */
struct chip_select csels [ 2 ] ;
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
2010-03-09 12:46:00 +01:00
/* x4 or x8 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 ;
2010-10-14 16:01:30 +02:00
} ;
2010-10-21 18:52:53 +02:00
static inline u64 get_dram_base ( struct amd64_pvt * pvt , unsigned i )
{
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 ;
}
static inline u64 get_dram_limit ( struct amd64_pvt * pvt , unsigned i )
{
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 ) ;
}
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 ;
} ;
2009-04-27 20:01:01 +02:00
# ifdef CONFIG_EDAC_DEBUG
2010-09-02 18:33:24 +02:00
# define NUM_DBG_ATTRS 5
2009-04-27 20:01:01 +02:00
# else
# define NUM_DBG_ATTRS 0
# endif
# ifdef CONFIG_EDAC_AMD64_ERROR_INJECTION
# define NUM_INJ_ATTRS 5
# else
# define NUM_INJ_ATTRS 0
# endif
extern struct mcidev_sysfs_attribute amd64_dbg_attrs [ NUM_DBG_ATTRS ] ,
amd64_inj_attrs [ NUM_INJ_ATTRS ] ;
2009-05-04 19:25:34 +02:00
/*
* 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 ,
u16 syndrome ) ;
2011-01-18 19:16:08 +01:00
int ( * dbam_to_cs ) ( struct amd64_pvt * pvt , u8 dct , unsigned cs_mode ) ;
2010-10-08 18:32:29 +02:00
int ( * read_dct_pci_cfg ) ( struct amd64_pvt * pvt , int offset ,
u32 * val , const char * func ) ;
2009-05-04 19:25:34 +02:00
} ;
struct amd64_family_type {
const char * ctl_name ;
2010-10-01 20:11:07 +02:00
u16 f1_id , f3_id ;
2009-05-04 19:25:34 +02:00
struct low_ops ops ;
} ;
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
2010-10-08 18:32:29 +02:00
# define amd64_read_dct_pci_cfg(pvt, offset, val) \
pvt - > ops - > read_dct_pci_cfg ( pvt , offset , val , __func__ )
2009-10-13 19:26:55 +02:00
2009-05-04 19:25:34 +02:00
int amd64_get_dram_hole_info ( struct mem_ctl_info * mci , u64 * hole_base ,
u64 * hole_offset , u64 * hole_size ) ;