2019-05-19 15:08:20 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2009-05-04 22:11:14 +04:00
# include "amd64_edac.h"
2010-09-17 20:03:43 +04:00
# include <asm/amd_nb.h>
2009-05-04 22:11:14 +04:00
2013-12-15 20:54:27 +04:00
static struct edac_pci_ctl_info * pci_ctl ;
2009-05-04 22:11:14 +04:00
/*
* Set by command line parameter . If BIOS has enabled the ECC , this override is
* cleared to prevent re - enabling the hardware by this driver .
*/
static int ecc_enable_override ;
module_param ( ecc_enable_override , int , 0644 ) ;
2010-02-02 08:39:15 +03:00
static struct msr __percpu * msrs ;
2009-12-11 20:14:40 +03:00
2019-10-22 23:35:09 +03:00
static struct amd64_family_type * fam_type ;
2015-02-17 12:58:34 +03:00
/* Per-node stuff */
2010-10-14 18:01:30 +04:00
static struct ecc_settings * * ecc_stngs ;
2009-05-04 22:11:14 +04:00
2020-11-22 17:57:21 +03:00
/* Device for the PCI component */
static struct device * pci_ctl_dev ;
2009-06-25 21:32:38 +04:00
/*
* Valid scrub rates for the K8 hardware memory scrubber . We map the scrubbing
* bandwidth to a valid bit pattern . The ' set ' operation finds the ' matching -
* or higher value ' .
*
* FIXME : Produce a better mapping / linearisation .
*/
2012-11-30 12:44:20 +04:00
static const struct scrubrate {
2010-11-24 21:52:09 +03:00
u32 scrubval ; /* bit pattern for scrub rate */
u32 bandwidth ; /* bandwidth consumed (bytes/sec) */
} scrubrates [ ] = {
2009-06-25 21:32:38 +04:00
{ 0x01 , 1600000000UL } ,
{ 0x02 , 800000000UL } ,
{ 0x03 , 400000000UL } ,
{ 0x04 , 200000000UL } ,
{ 0x05 , 100000000UL } ,
{ 0x06 , 50000000UL } ,
{ 0x07 , 25000000UL } ,
{ 0x08 , 12284069UL } ,
{ 0x09 , 6274509UL } ,
{ 0x0A , 3121951UL } ,
{ 0x0B , 1560975UL } ,
{ 0x0C , 781440UL } ,
{ 0x0D , 390720UL } ,
{ 0x0E , 195300UL } ,
{ 0x0F , 97650UL } ,
{ 0x10 , 48854UL } ,
{ 0x11 , 24427UL } ,
{ 0x12 , 12213UL } ,
{ 0x13 , 6101UL } ,
{ 0x14 , 3051UL } ,
{ 0x15 , 1523UL } ,
{ 0x16 , 761UL } ,
{ 0x00 , 0UL } , /* scrubbing off */
} ;
2012-08-09 20:41:07 +04:00
int __amd64_read_pci_cfg_dword ( struct pci_dev * pdev , int offset ,
u32 * val , const char * func )
2010-10-08 20:32:29 +04:00
{
int err = 0 ;
err = pci_read_config_dword ( pdev , offset , val ) ;
if ( err )
amd64_warn ( " %s: error reading F%dx%03x. \n " ,
func , PCI_FUNC ( pdev - > devfn ) , offset ) ;
return err ;
}
int __amd64_write_pci_cfg_dword ( struct pci_dev * pdev , int offset ,
u32 val , const char * func )
{
int err = 0 ;
err = pci_write_config_dword ( pdev , offset , val ) ;
if ( err )
amd64_warn ( " %s: error writing to F%dx%03x. \n " ,
func , PCI_FUNC ( pdev - > devfn ) , offset ) ;
return err ;
}
2014-09-15 20:37:38 +04:00
/*
* Select DCT to which PCI cfg accesses are routed
*/
static void f15h_select_dct ( struct amd64_pvt * pvt , u8 dct )
{
u32 reg = 0 ;
amd64_read_pci_cfg ( pvt - > F1 , DCT_CFG_SEL , & reg ) ;
reg & = ( pvt - > model = = 0x30 ) ? ~ 3 : ~ 1 ;
reg | = dct ;
amd64_write_pci_cfg ( pvt - > F1 , DCT_CFG_SEL , reg ) ;
}
2010-10-08 20:32:29 +04:00
/*
*
* Depending on the family , F2 DCT reads need special handling :
*
2014-09-15 20:37:38 +04:00
* K8 : has a single DCT only and no address offsets > = 0x100
2010-10-08 20:32:29 +04:00
*
* F10h : each DCT has its own set of regs
* DCT0 - > F2x040 . .
* DCT1 - > F2x140 . .
*
2013-04-17 23:57:13 +04:00
* F16h : has only 1 DCT
2014-09-15 20:37:38 +04:00
*
* F15h : we select which DCT we access using F1x10C [ DctCfgSel ]
2010-10-08 20:32:29 +04:00
*/
2014-09-15 20:37:38 +04:00
static inline int amd64_read_dct_pci_cfg ( struct amd64_pvt * pvt , u8 dct ,
int offset , u32 * val )
2010-10-08 20:32:29 +04:00
{
2014-09-15 20:37:38 +04:00
switch ( pvt - > fam ) {
case 0xf :
if ( dct | | offset > = 0x100 )
return - EINVAL ;
break ;
2010-10-08 20:32:29 +04:00
2014-09-15 20:37:38 +04:00
case 0x10 :
if ( dct ) {
/*
* Note : If ganging is enabled , barring the regs
* F2x [ 1 , 0 ] 98 and F2x [ 1 , 0 ] 9 C ; reads reads to F2x1xx
* return 0. ( cf . Section 2.8 .1 F10h BKDG )
*/
if ( dct_ganging_enabled ( pvt ) )
return 0 ;
2010-10-08 20:32:29 +04:00
2014-09-15 20:37:38 +04:00
offset + = 0x100 ;
}
break ;
2011-09-19 19:34:45 +04:00
2014-09-15 20:37:38 +04:00
case 0x15 :
/*
* F15h : F2x1xx addresses do not map explicitly to DCT1 .
* We should select which DCT we access using F1x10C [ DctCfgSel ]
*/
dct = ( dct & & pvt - > model = = 0x30 ) ? 3 : dct ;
f15h_select_dct ( pvt , dct ) ;
break ;
2011-09-19 19:34:45 +04:00
2014-09-15 20:37:38 +04:00
case 0x16 :
if ( dct )
return - EINVAL ;
break ;
2010-10-08 20:32:29 +04:00
2014-09-15 20:37:38 +04:00
default :
break ;
2010-10-08 20:32:29 +04:00
}
2014-09-15 20:37:38 +04:00
return amd64_read_pci_cfg ( pvt - > F2 , offset , val ) ;
2010-10-08 20:32:29 +04:00
}
2009-05-04 22:11:14 +04:00
/*
* Memory scrubber control interface . For K8 , memory scrubbing is handled by
* hardware and can involve L2 cache , dcache as well as the main memory . With
* F10 , this is extended to L3 cache scrubbing on CPU models sporting that
* functionality .
*
* This causes the " units " for the scrubbing speed to vary from 64 byte blocks
* ( dram ) over to cache lines . This is nasty , so we will use bandwidth in
* bytes / sec for the setting .
*
* Currently , we only do dram scrubbing . If the scrubbing is done in software on
* other archs , we might not have access to the caches directly .
*/
2016-11-18 01:57:42 +03:00
static inline void __f17h_set_scrubval ( struct amd64_pvt * pvt , u32 scrubval )
{
/*
* Fam17h supports scrub values between 0x5 and 0x14 . Also , the values
* are shifted down by 0x5 , so scrubval 0x5 is written to the register
* as 0x0 , scrubval 0x6 as 0x1 , etc .
*/
if ( scrubval > = 0x5 & & scrubval < = 0x14 ) {
scrubval - = 0x5 ;
pci_write_bits32 ( pvt - > F6 , F17H_SCR_LIMIT_ADDR , scrubval , 0xF ) ;
pci_write_bits32 ( pvt - > F6 , F17H_SCR_BASE_ADDR , 1 , 0x1 ) ;
} else {
pci_write_bits32 ( pvt - > F6 , F17H_SCR_BASE_ADDR , 0 , 0x1 ) ;
}
}
2009-05-04 22:11:14 +04:00
/*
2016-11-18 01:57:42 +03:00
* Scan the scrub rate mapping table for a close or matching bandwidth value to
2009-05-04 22:11:14 +04:00
* issue . If requested is too big , then use last maximum value found .
*/
2015-09-28 14:43:12 +03:00
static int __set_scrub_rate ( struct amd64_pvt * pvt , u32 new_bw , u32 min_rate )
2009-05-04 22:11:14 +04:00
{
u32 scrubval ;
int i ;
/*
* map the configured rate ( new_bw ) to a value specific to the AMD64
* memory controller and apply to register . Search for the first
* bandwidth entry that is greater or equal than the setting requested
* and program that . If at last entry , turn off DRAM scrubbing .
2012-10-24 01:09:39 +04:00
*
* If no suitable bandwidth is found , turn off DRAM scrubbing entirely
* by falling back to the last element in scrubrates [ ] .
2009-05-04 22:11:14 +04:00
*/
2012-10-24 01:09:39 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( scrubrates ) - 1 ; i + + ) {
2009-05-04 22:11:14 +04:00
/*
* skip scrub rates which aren ' t recommended
* ( see F10 BKDG , F3x58 )
*/
2010-10-01 20:38:19 +04:00
if ( scrubrates [ i ] . scrubval < min_rate )
2009-05-04 22:11:14 +04:00
continue ;
if ( scrubrates [ i ] . bandwidth < = new_bw )
break ;
}
scrubval = scrubrates [ i ] . scrubval ;
2020-01-10 04:56:51 +03:00
if ( pvt - > umc ) {
2016-11-18 01:57:42 +03:00
__f17h_set_scrubval ( pvt , scrubval ) ;
} else if ( pvt - > fam = = 0x15 & & pvt - > model = = 0x60 ) {
2015-09-28 14:43:12 +03:00
f15h_select_dct ( pvt , 0 ) ;
pci_write_bits32 ( pvt - > F2 , F15H_M60H_SCRCTRL , scrubval , 0x001F ) ;
f15h_select_dct ( pvt , 1 ) ;
pci_write_bits32 ( pvt - > F2 , F15H_M60H_SCRCTRL , scrubval , 0x001F ) ;
} else {
pci_write_bits32 ( pvt - > F3 , SCRCTRL , scrubval , 0x001F ) ;
}
2009-05-04 22:11:14 +04:00
2010-11-24 21:52:09 +03:00
if ( scrubval )
return scrubrates [ i ] . bandwidth ;
2009-05-04 22:11:14 +04:00
return 0 ;
}
2013-12-15 20:54:27 +04:00
static int set_scrub_rate ( struct mem_ctl_info * mci , u32 bw )
2009-05-04 22:11:14 +04:00
{
struct amd64_pvt * pvt = mci - > pvt_info ;
2011-01-19 22:02:38 +03:00
u32 min_scrubrate = 0x5 ;
2009-05-04 22:11:14 +04:00
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0xf )
2011-01-19 22:02:38 +03:00
min_scrubrate = 0x0 ;
2015-09-28 14:43:12 +03:00
if ( pvt - > fam = = 0x15 ) {
/* Erratum #505 */
if ( pvt - > model < 0x10 )
f15h_select_dct ( pvt , 0 ) ;
2011-09-19 19:34:45 +04:00
2015-09-28 14:43:12 +03:00
if ( pvt - > model = = 0x60 )
min_scrubrate = 0x6 ;
}
return __set_scrub_rate ( pvt , bw , min_scrubrate ) ;
2009-05-04 22:11:14 +04:00
}
2013-12-15 20:54:27 +04:00
static int get_scrub_rate ( struct mem_ctl_info * mci )
2009-05-04 22:11:14 +04:00
{
struct amd64_pvt * pvt = mci - > pvt_info ;
2010-11-24 21:52:09 +03:00
int i , retval = - EINVAL ;
2016-11-18 01:57:42 +03:00
u32 scrubval = 0 ;
2009-05-04 22:11:14 +04:00
2020-01-10 04:56:51 +03:00
if ( pvt - > umc ) {
2016-11-18 01:57:42 +03:00
amd64_read_pci_cfg ( pvt - > F6 , F17H_SCR_BASE_ADDR , & scrubval ) ;
if ( scrubval & BIT ( 0 ) ) {
amd64_read_pci_cfg ( pvt - > F6 , F17H_SCR_LIMIT_ADDR , & scrubval ) ;
scrubval & = 0xF ;
scrubval + = 0x5 ;
} else {
scrubval = 0 ;
}
2020-01-10 04:56:51 +03:00
} else if ( pvt - > fam = = 0x15 ) {
/* Erratum #505 */
if ( pvt - > model < 0x10 )
f15h_select_dct ( pvt , 0 ) ;
2016-11-18 01:57:42 +03:00
2020-01-10 04:56:51 +03:00
if ( pvt - > model = = 0x60 )
amd64_read_pci_cfg ( pvt - > F2 , F15H_M60H_SCRCTRL , & scrubval ) ;
2020-06-18 21:25:25 +03:00
else
amd64_read_pci_cfg ( pvt - > F3 , SCRCTRL , & scrubval ) ;
2020-01-10 04:56:51 +03:00
} else {
2015-09-28 14:43:12 +03:00
amd64_read_pci_cfg ( pvt - > F3 , SCRCTRL , & scrubval ) ;
2016-11-18 01:57:42 +03:00
}
2009-05-04 22:11:14 +04:00
scrubval = scrubval & 0x001F ;
2010-01-11 22:58:21 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( scrubrates ) ; i + + ) {
2009-05-04 22:11:14 +04:00
if ( scrubrates [ i ] . scrubval = = scrubval ) {
2010-11-24 21:52:09 +03:00
retval = scrubrates [ i ] . bandwidth ;
2009-05-04 22:11:14 +04:00
break ;
}
}
2010-11-24 21:52:09 +03:00
return retval ;
2009-05-04 22:11:14 +04:00
}
2009-04-27 17:53:22 +04:00
/*
2010-10-21 20:52:53 +04:00
* returns true if the SysAddr given by sys_addr matches the
* DRAM base / limit associated with node_id
2009-04-27 17:53:22 +04:00
*/
2013-12-15 20:54:27 +04:00
static bool base_limit_match ( struct amd64_pvt * pvt , u64 sys_addr , u8 nid )
2009-04-27 17:53:22 +04:00
{
2010-10-21 20:52:53 +04:00
u64 addr ;
2009-04-27 17:53:22 +04:00
/* The K8 treats this as a 40-bit value. However, bits 63-40 will be
* all ones if the most significant implemented address bit is 1.
* Here we discard bits 63 - 40. See section 3.4 .2 of AMD publication
* 24592 : AMD x86 - 64 Architecture Programmer ' s Manual Volume 1
* Application Programming .
*/
addr = sys_addr & 0x000000ffffffffffull ;
2010-10-21 20:52:53 +04:00
return ( ( addr > = get_dram_base ( pvt , nid ) ) & &
( addr < = get_dram_limit ( pvt , nid ) ) ) ;
2009-04-27 17:53:22 +04:00
}
/*
* Attempt to map a SysAddr to a node . On success , return a pointer to the
* mem_ctl_info structure for the node that the SysAddr maps to .
*
* On failure , return NULL .
*/
static struct mem_ctl_info * find_mc_by_sys_addr ( struct mem_ctl_info * mci ,
u64 sys_addr )
{
struct amd64_pvt * pvt ;
2012-11-30 12:44:20 +04:00
u8 node_id ;
2009-04-27 17:53:22 +04:00
u32 intlv_en , bits ;
/*
* Here we use the DRAM Base ( section 3.4 .4 .1 ) and DRAM Limit ( section
* 3.4 .4 .2 ) registers to map the SysAddr to a node ID .
*/
pvt = mci - > pvt_info ;
/*
* The value of this field should be the same for all DRAM Base
* registers . Therefore we arbitrarily choose to read it from the
* register for node 0.
*/
2010-10-21 20:52:53 +04:00
intlv_en = dram_intlv_en ( pvt , 0 ) ;
2009-04-27 17:53:22 +04:00
if ( intlv_en = = 0 ) {
2010-10-21 20:52:53 +04:00
for ( node_id = 0 ; node_id < DRAM_RANGES ; node_id + + ) {
2013-12-15 20:54:27 +04:00
if ( base_limit_match ( pvt , sys_addr , node_id ) )
2009-09-18 14:39:19 +04:00
goto found ;
2009-04-27 17:53:22 +04:00
}
2009-09-18 14:39:19 +04:00
goto err_no_match ;
2009-04-27 17:53:22 +04:00
}
2009-09-18 14:27:27 +04:00
if ( unlikely ( ( intlv_en ! = 0x01 ) & &
( intlv_en ! = 0x03 ) & &
( intlv_en ! = 0x07 ) ) ) {
2010-10-07 20:29:15 +04:00
amd64_warn ( " DRAM Base[IntlvEn] junk value: 0x%x, BIOS bug? \n " , intlv_en ) ;
2009-04-27 17:53:22 +04:00
return NULL ;
}
bits = ( ( ( u32 ) sys_addr ) > > 12 ) & intlv_en ;
for ( node_id = 0 ; ; ) {
2010-10-21 20:52:53 +04:00
if ( ( dram_intlv_sel ( pvt , node_id ) & intlv_en ) = = bits )
2009-04-27 17:53:22 +04:00
break ; /* intlv_sel field matches */
2010-10-21 20:52:53 +04:00
if ( + + node_id > = DRAM_RANGES )
2009-04-27 17:53:22 +04:00
goto err_no_match ;
}
/* sanity test for sys_addr */
2013-12-15 20:54:27 +04:00
if ( unlikely ( ! base_limit_match ( pvt , sys_addr , node_id ) ) ) {
2010-10-07 20:29:15 +04:00
amd64_warn ( " %s: sys_addr 0x%llx falls outside base/limit address "
" range for node %d with node interleaving enabled. \n " ,
__func__ , sys_addr , node_id ) ;
2009-04-27 17:53:22 +04:00
return NULL ;
}
found :
2011-02-21 20:55:00 +03:00
return edac_mc_find ( ( int ) node_id ) ;
2009-04-27 17:53:22 +04:00
err_no_match :
2012-04-30 00:08:39 +04:00
edac_dbg ( 2 , " sys_addr 0x%lx doesn't match any node \n " ,
( unsigned long ) sys_addr ) ;
2009-04-27 17:53:22 +04:00
return NULL ;
}
2009-04-27 17:57:12 +04:00
/*
2010-11-29 21:49:02 +03:00
* compute the CS base address of the @ csrow on the DRAM controller @ dct .
* For details see F2x [ 5 C : 40 ] in the processor ' s BKDG
2009-04-27 17:57:12 +04:00
*/
2010-11-29 21:49:02 +03:00
static void get_cs_base_and_mask ( struct amd64_pvt * pvt , int csrow , u8 dct ,
u64 * base , u64 * mask )
2009-04-27 17:57:12 +04:00
{
2010-11-29 21:49:02 +03:00
u64 csbase , csmask , base_bits , mask_bits ;
u8 addr_shift ;
2009-04-27 17:57:12 +04:00
2013-08-09 20:54:49 +04:00
if ( pvt - > fam = = 0xf & & pvt - > ext_model < K8_REV_F ) {
2010-11-29 21:49:02 +03:00
csbase = pvt - > csels [ dct ] . csbases [ csrow ] ;
csmask = pvt - > csels [ dct ] . csmasks [ csrow ] ;
2013-10-19 01:29:07 +04:00
base_bits = GENMASK_ULL ( 31 , 21 ) | GENMASK_ULL ( 15 , 9 ) ;
mask_bits = GENMASK_ULL ( 29 , 21 ) | GENMASK_ULL ( 15 , 9 ) ;
2010-11-29 21:49:02 +03:00
addr_shift = 4 ;
2013-04-17 23:57:13 +04:00
/*
2013-08-09 20:54:49 +04:00
* F16h and F15h , models 30 h and later need two addr_shift values :
* 8 for high and 6 for low ( cf . F16h BKDG ) .
*/
} else if ( pvt - > fam = = 0x16 | |
( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 ) ) {
2013-04-17 23:57:13 +04:00
csbase = pvt - > csels [ dct ] . csbases [ csrow ] ;
csmask = pvt - > csels [ dct ] . csmasks [ csrow > > 1 ] ;
2013-10-19 01:29:07 +04:00
* base = ( csbase & GENMASK_ULL ( 15 , 5 ) ) < < 6 ;
* base | = ( csbase & GENMASK_ULL ( 30 , 19 ) ) < < 8 ;
2013-04-17 23:57:13 +04:00
* mask = ~ 0ULL ;
/* poke holes for the csmask */
2013-10-19 01:29:07 +04:00
* mask & = ~ ( ( GENMASK_ULL ( 15 , 5 ) < < 6 ) |
( GENMASK_ULL ( 30 , 19 ) < < 8 ) ) ;
2013-04-17 23:57:13 +04:00
2013-10-19 01:29:07 +04:00
* mask | = ( csmask & GENMASK_ULL ( 15 , 5 ) ) < < 6 ;
* mask | = ( csmask & GENMASK_ULL ( 30 , 19 ) ) < < 8 ;
2013-04-17 23:57:13 +04:00
return ;
2010-11-29 21:49:02 +03:00
} else {
csbase = pvt - > csels [ dct ] . csbases [ csrow ] ;
csmask = pvt - > csels [ dct ] . csmasks [ csrow > > 1 ] ;
addr_shift = 8 ;
2009-04-27 17:57:12 +04:00
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0x15 )
2013-10-19 01:29:07 +04:00
base_bits = mask_bits =
GENMASK_ULL ( 30 , 19 ) | GENMASK_ULL ( 13 , 5 ) ;
2010-11-29 21:49:02 +03:00
else
2013-10-19 01:29:07 +04:00
base_bits = mask_bits =
GENMASK_ULL ( 28 , 19 ) | GENMASK_ULL ( 13 , 5 ) ;
2010-11-29 21:49:02 +03:00
}
2009-04-27 17:57:12 +04:00
2010-11-29 21:49:02 +03:00
* base = ( csbase & base_bits ) < < addr_shift ;
2009-04-27 17:57:12 +04:00
2010-11-29 21:49:02 +03:00
* mask = ~ 0ULL ;
/* poke holes for the csmask */
* mask & = ~ ( mask_bits < < addr_shift ) ;
/* OR them in */
* mask | = ( csmask & mask_bits ) < < addr_shift ;
2009-04-27 17:57:12 +04:00
}
2010-11-29 21:49:02 +03:00
# define for_each_chip_select(i, dct, pvt) \
for ( i = 0 ; i < pvt - > csels [ dct ] . b_cnt ; i + + )
2011-01-13 20:02:22 +03:00
# define chip_select_base(i, dct, pvt) \
pvt - > csels [ dct ] . csbases [ i ]
2010-11-29 21:49:02 +03:00
# define for_each_chip_select_mask(i, dct, pvt) \
for ( i = 0 ; i < pvt - > csels [ dct ] . m_cnt ; i + + )
2019-02-28 18:36:10 +03:00
# define for_each_umc(i) \
2019-10-22 23:35:11 +03:00
for ( i = 0 ; i < fam_type - > max_mcs ; i + + )
2019-02-28 18:36:10 +03:00
2009-04-27 17:57:12 +04:00
/*
* @ input_addr is an InputAddr associated with the node given by mci . Return the
* csrow that input_addr maps to , or - 1 on failure ( no csrow claims input_addr ) .
*/
static int input_addr_to_csrow ( struct mem_ctl_info * mci , u64 input_addr )
{
struct amd64_pvt * pvt ;
int csrow ;
u64 base , mask ;
pvt = mci - > pvt_info ;
2010-11-29 21:49:02 +03:00
for_each_chip_select ( csrow , 0 , pvt ) {
if ( ! csrow_enabled ( csrow , 0 , pvt ) )
2009-04-27 17:57:12 +04:00
continue ;
2010-11-29 21:49:02 +03:00
get_cs_base_and_mask ( pvt , csrow , 0 , & base , & mask ) ;
mask = ~ mask ;
2009-04-27 17:57:12 +04:00
if ( ( input_addr & mask ) = = ( base & mask ) ) {
2012-04-30 00:08:39 +04:00
edac_dbg ( 2 , " InputAddr 0x%lx matches csrow %d (node %d) \n " ,
( unsigned long ) input_addr , csrow ,
pvt - > mc_node_id ) ;
2009-04-27 17:57:12 +04:00
return csrow ;
}
}
2012-04-30 00:08:39 +04:00
edac_dbg ( 2 , " no matching csrow for InputAddr 0x%lx (MC node %d) \n " ,
( unsigned long ) input_addr , pvt - > mc_node_id ) ;
2009-04-27 17:57:12 +04:00
return - 1 ;
}
/*
* Obtain info from the DRAM Hole Address Register ( section 3.4 .8 , pub # 26094 )
* for the node represented by mci . Info is passed back in * hole_base ,
* * hole_offset , and * hole_size . Function returns 0 if info is valid or 1 if
* info is invalid . Info may be invalid for either of the following reasons :
*
* - The revision of the node is not E or greater . In this case , the DRAM Hole
* Address Register does not exist .
*
* - The DramHoleValid bit is cleared in the DRAM Hole Address Register ,
* indicating that its contents are not valid .
*
* The values passed back in * hole_base , * hole_offset , and * hole_size are
* complete 32 - bit values despite the fact that the bitfields in the DHAR
* only represent bits 31 - 24 of the base and offset values .
*/
2020-12-14 22:47:11 +03:00
static int get_dram_hole_info ( struct mem_ctl_info * mci , u64 * hole_base ,
u64 * hole_offset , u64 * hole_size )
2009-04-27 17:57:12 +04:00
{
struct amd64_pvt * pvt = mci - > pvt_info ;
/* only revE and later have the DRAM Hole Address Register */
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0xf & & pvt - > ext_model < K8_REV_E ) {
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " revision %d for node %d does not support DHAR \n " ,
pvt - > ext_model , pvt - > mc_node_id ) ;
2009-04-27 17:57:12 +04:00
return 1 ;
}
2010-11-11 19:29:13 +03:00
/* valid for Fam10h and above */
2013-08-10 15:54:48 +04:00
if ( pvt - > fam > = 0x10 & & ! dhar_mem_hoist_valid ( pvt ) ) {
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " Dram Memory Hoisting is DISABLED on this system \n " ) ;
2009-04-27 17:57:12 +04:00
return 1 ;
}
2010-12-10 21:49:19 +03:00
if ( ! dhar_valid ( pvt ) ) {
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " Dram Memory Hoisting is DISABLED on this node %d \n " ,
pvt - > mc_node_id ) ;
2009-04-27 17:57:12 +04:00
return 1 ;
}
/* This node has Memory Hoisting */
/* +------------------+--------------------+--------------------+-----
* | memory | DRAM hole | relocated |
* | [ 0 , ( x - 1 ) ] | [ x , 0xffffffff ] | addresses from |
* | | | DRAM hole |
* | | | [ 0x100000000 , |
* | | | ( 0x100000000 + |
* | | | ( 0xffffffff - x ) ) ] |
* + - - - - - - - - - - - - - - - - - - + - - - - - - - - - - - - - - - - - - - - + - - - - - - - - - - - - - - - - - - - - + - - - - -
*
* Above is a diagram of physical memory showing the DRAM hole and the
* relocated addresses from the DRAM hole . As shown , the DRAM hole
* starts at address x ( the base address ) and extends through address
* 0xffffffff . The DRAM Hole Address Register ( DHAR ) relocates the
* addresses in the hole so that they start at 0x100000000 .
*/
2012-08-10 14:50:50 +04:00
* hole_base = dhar_base ( pvt ) ;
* hole_size = ( 1ULL < < 32 ) - * hole_base ;
2009-04-27 17:57:12 +04:00
2013-08-10 15:54:48 +04:00
* hole_offset = ( pvt - > fam > 0xf ) ? f10_dhar_offset ( pvt )
: k8_dhar_offset ( pvt ) ;
2009-04-27 17:57:12 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " DHAR info for node %d base 0x%lx offset 0x%lx size 0x%lx \n " ,
pvt - > mc_node_id , ( unsigned long ) * hole_base ,
( unsigned long ) * hole_offset , ( unsigned long ) * hole_size ) ;
2009-04-27 17:57:12 +04:00
return 0 ;
}
2020-12-14 22:47:11 +03:00
# ifdef CONFIG_EDAC_DEBUG
# define EDAC_DCT_ATTR_SHOW(reg) \
static ssize_t reg # # _show ( struct device * dev , \
struct device_attribute * mattr , char * data ) \
{ \
struct mem_ctl_info * mci = to_mci ( dev ) ; \
struct amd64_pvt * pvt = mci - > pvt_info ; \
\
return sprintf ( data , " 0x%016llx \n " , ( u64 ) pvt - > reg ) ; \
}
EDAC_DCT_ATTR_SHOW ( dhar ) ;
EDAC_DCT_ATTR_SHOW ( dbam0 ) ;
EDAC_DCT_ATTR_SHOW ( top_mem ) ;
EDAC_DCT_ATTR_SHOW ( top_mem2 ) ;
static ssize_t hole_show ( struct device * dev , struct device_attribute * mattr ,
char * data )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
u64 hole_base = 0 ;
u64 hole_offset = 0 ;
u64 hole_size = 0 ;
get_dram_hole_info ( mci , & hole_base , & hole_offset , & hole_size ) ;
return sprintf ( data , " %llx %llx %llx \n " , hole_base , hole_offset ,
hole_size ) ;
}
/*
* update NUM_DBG_ATTRS in case you add new members
*/
static DEVICE_ATTR ( dhar , S_IRUGO , dhar_show , NULL ) ;
static DEVICE_ATTR ( dbam , S_IRUGO , dbam0_show , NULL ) ;
static DEVICE_ATTR ( topmem , S_IRUGO , top_mem_show , NULL ) ;
static DEVICE_ATTR ( topmem2 , S_IRUGO , top_mem2_show , NULL ) ;
static DEVICE_ATTR ( dram_hole , S_IRUGO , hole_show , NULL ) ;
static struct attribute * dbg_attrs [ ] = {
& dev_attr_dhar . attr ,
& dev_attr_dbam . attr ,
& dev_attr_topmem . attr ,
& dev_attr_topmem2 . attr ,
& dev_attr_dram_hole . attr ,
NULL
} ;
static const struct attribute_group dbg_group = {
. attrs = dbg_attrs ,
} ;
2020-12-15 11:18:44 +03:00
static ssize_t inject_section_show ( struct device * dev ,
struct device_attribute * mattr , char * buf )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
return sprintf ( buf , " 0x%x \n " , pvt - > injection . section ) ;
}
/*
* store error injection section value which refers to one of 4 16 - byte sections
* within a 64 - byte cacheline
*
* range : 0. .3
*/
static ssize_t inject_section_store ( struct device * dev ,
struct device_attribute * mattr ,
const char * data , size_t count )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
unsigned long value ;
int ret ;
ret = kstrtoul ( data , 10 , & value ) ;
if ( ret < 0 )
return ret ;
if ( value > 3 ) {
amd64_warn ( " %s: invalid section 0x%lx \n " , __func__ , value ) ;
return - EINVAL ;
}
pvt - > injection . section = ( u32 ) value ;
return count ;
}
static ssize_t inject_word_show ( struct device * dev ,
struct device_attribute * mattr , char * buf )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
return sprintf ( buf , " 0x%x \n " , pvt - > injection . word ) ;
}
/*
* store error injection word value which refers to one of 9 16 - bit word of the
* 16 - byte ( 128 - bit + ECC bits ) section
*
* range : 0. .8
*/
static ssize_t inject_word_store ( struct device * dev ,
struct device_attribute * mattr ,
const char * data , size_t count )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
unsigned long value ;
int ret ;
ret = kstrtoul ( data , 10 , & value ) ;
if ( ret < 0 )
return ret ;
if ( value > 8 ) {
amd64_warn ( " %s: invalid word 0x%lx \n " , __func__ , value ) ;
return - EINVAL ;
}
pvt - > injection . word = ( u32 ) value ;
return count ;
}
static ssize_t inject_ecc_vector_show ( struct device * dev ,
struct device_attribute * mattr ,
char * buf )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
return sprintf ( buf , " 0x%x \n " , pvt - > injection . bit_map ) ;
}
/*
* store 16 bit error injection vector which enables injecting errors to the
* corresponding bit within the error injection word above . When used during a
* DRAM ECC read , it holds the contents of the of the DRAM ECC bits .
*/
static ssize_t inject_ecc_vector_store ( struct device * dev ,
struct device_attribute * mattr ,
const char * data , size_t count )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
unsigned long value ;
int ret ;
ret = kstrtoul ( data , 16 , & value ) ;
if ( ret < 0 )
return ret ;
if ( value & 0xFFFF0000 ) {
amd64_warn ( " %s: invalid EccVector: 0x%lx \n " , __func__ , value ) ;
return - EINVAL ;
}
pvt - > injection . bit_map = ( u32 ) value ;
return count ;
}
/*
* Do a DRAM ECC read . Assemble staged values in the pvt area , format into
* fields needed by the injection registers and read the NB Array Data Port .
*/
static ssize_t inject_read_store ( struct device * dev ,
struct device_attribute * mattr ,
const char * data , size_t count )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
unsigned long value ;
u32 section , word_bits ;
int ret ;
ret = kstrtoul ( data , 10 , & value ) ;
if ( ret < 0 )
return ret ;
/* Form value to choose 16-byte section of cacheline */
section = F10_NB_ARRAY_DRAM | SET_NB_ARRAY_ADDR ( pvt - > injection . section ) ;
amd64_write_pci_cfg ( pvt - > F3 , F10_NB_ARRAY_ADDR , section ) ;
word_bits = SET_NB_DRAM_INJECTION_READ ( pvt - > injection ) ;
/* Issue 'word' and 'bit' along with the READ request */
amd64_write_pci_cfg ( pvt - > F3 , F10_NB_ARRAY_DATA , word_bits ) ;
edac_dbg ( 0 , " section=0x%x word_bits=0x%x \n " , section , word_bits ) ;
return count ;
}
/*
* Do a DRAM ECC write . Assemble staged values in the pvt area and format into
* fields needed by the injection registers .
*/
static ssize_t inject_write_store ( struct device * dev ,
struct device_attribute * mattr ,
const char * data , size_t count )
{
struct mem_ctl_info * mci = to_mci ( dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
u32 section , word_bits , tmp ;
unsigned long value ;
int ret ;
ret = kstrtoul ( data , 10 , & value ) ;
if ( ret < 0 )
return ret ;
/* Form value to choose 16-byte section of cacheline */
section = F10_NB_ARRAY_DRAM | SET_NB_ARRAY_ADDR ( pvt - > injection . section ) ;
amd64_write_pci_cfg ( pvt - > F3 , F10_NB_ARRAY_ADDR , section ) ;
word_bits = SET_NB_DRAM_INJECTION_WRITE ( pvt - > injection ) ;
pr_notice_once ( " Don't forget to decrease MCE polling interval in \n "
" /sys/bus/machinecheck/devices/machinecheck<CPUNUM>/check_interval \n "
" so that you can get the error report faster. \n " ) ;
on_each_cpu ( disable_caches , NULL , 1 ) ;
/* Issue 'word' and 'bit' along with the READ request */
amd64_write_pci_cfg ( pvt - > F3 , F10_NB_ARRAY_DATA , word_bits ) ;
retry :
/* wait until injection happens */
amd64_read_pci_cfg ( pvt - > F3 , F10_NB_ARRAY_DATA , & tmp ) ;
if ( tmp & F10_NB_ARR_ECC_WR_REQ ) {
cpu_relax ( ) ;
goto retry ;
}
on_each_cpu ( enable_caches , NULL , 1 ) ;
edac_dbg ( 0 , " section=0x%x word_bits=0x%x \n " , section , word_bits ) ;
return count ;
}
/*
* update NUM_INJ_ATTRS in case you add new members
*/
static DEVICE_ATTR ( inject_section , S_IRUGO | S_IWUSR ,
inject_section_show , inject_section_store ) ;
static DEVICE_ATTR ( inject_word , S_IRUGO | S_IWUSR ,
inject_word_show , inject_word_store ) ;
static DEVICE_ATTR ( inject_ecc_vector , S_IRUGO | S_IWUSR ,
inject_ecc_vector_show , inject_ecc_vector_store ) ;
static DEVICE_ATTR ( inject_write , S_IWUSR ,
NULL , inject_write_store ) ;
static DEVICE_ATTR ( inject_read , S_IWUSR ,
NULL , inject_read_store ) ;
static struct attribute * inj_attrs [ ] = {
& dev_attr_inject_section . attr ,
& dev_attr_inject_word . attr ,
& dev_attr_inject_ecc_vector . attr ,
& dev_attr_inject_write . attr ,
& dev_attr_inject_read . attr ,
NULL
} ;
static umode_t inj_is_visible ( struct kobject * kobj , struct attribute * attr , int idx )
{
struct device * dev = kobj_to_dev ( kobj ) ;
struct mem_ctl_info * mci = container_of ( dev , struct mem_ctl_info , dev ) ;
struct amd64_pvt * pvt = mci - > pvt_info ;
2020-12-22 20:55:06 +03:00
/* Families which have that injection hw */
if ( pvt - > fam > = 0x10 & & pvt - > fam < = 0x16 )
return attr - > mode ;
return 0 ;
2020-12-15 11:18:44 +03:00
}
static const struct attribute_group inj_group = {
. attrs = inj_attrs ,
. is_visible = inj_is_visible ,
} ;
# endif /* CONFIG_EDAC_DEBUG */
2009-04-27 17:57:12 +04:00
2009-05-04 22:46:50 +04:00
/*
* Return the DramAddr that the SysAddr given by @ sys_addr maps to . It is
* assumed that sys_addr maps to the node given by mci .
*
* The first part of section 3.4 .4 ( p . 70 ) shows how the DRAM Base ( section
* 3.4 .4 .1 ) and DRAM Limit ( section 3.4 .4 .2 ) registers are used to translate a
* SysAddr to a DramAddr . If the DRAM Hole Address Register ( DHAR ) is enabled ,
* then it is also involved in translating a SysAddr to a DramAddr . Sections
* 3.4 .8 and 3.5 .8 .2 describe the DHAR and how it is used for memory hoisting .
* These parts of the documentation are unclear . I interpret them as follows :
*
* When node n receives a SysAddr , it processes the SysAddr as follows :
*
* 1. It extracts the DRAMBase and DRAMLimit values from the DRAM Base and DRAM
* Limit registers for node n . If the SysAddr is not within the range
* specified by the base and limit values , then node n ignores the Sysaddr
* ( since it does not map to node n ) . Otherwise continue to step 2 below .
*
* 2. If the DramHoleValid bit of the DHAR for node n is clear , the DHAR is
* disabled so skip to step 3 below . Otherwise see if the SysAddr is within
* the range of relocated addresses ( starting at 0x100000000 ) from the DRAM
* hole . If not , skip to step 3 below . Else get the value of the
* DramHoleOffset field from the DHAR . To obtain the DramAddr , subtract the
* offset defined by this value from the SysAddr .
*
* 3. Obtain the base address for node n from the DRAMBase field of the DRAM
* Base register for node n . To obtain the DramAddr , subtract the base
* address from the SysAddr , as shown near the start of section 3.4 .4 ( p .70 ) .
*/
static u64 sys_addr_to_dram_addr ( struct mem_ctl_info * mci , u64 sys_addr )
{
2010-10-21 20:52:53 +04:00
struct amd64_pvt * pvt = mci - > pvt_info ;
2009-05-04 22:46:50 +04:00
u64 dram_base , hole_base , hole_offset , hole_size , dram_addr ;
2012-08-10 14:50:50 +04:00
int ret ;
2009-05-04 22:46:50 +04:00
2010-10-21 20:52:53 +04:00
dram_base = get_dram_base ( pvt , pvt - > mc_node_id ) ;
2009-05-04 22:46:50 +04:00
2020-12-14 22:47:11 +03:00
ret = get_dram_hole_info ( mci , & hole_base , & hole_offset , & hole_size ) ;
2009-05-04 22:46:50 +04:00
if ( ! ret ) {
2012-08-10 14:50:50 +04:00
if ( ( sys_addr > = ( 1ULL < < 32 ) ) & &
( sys_addr < ( ( 1ULL < < 32 ) + hole_size ) ) ) {
2009-05-04 22:46:50 +04:00
/* use DHAR to translate SysAddr to DramAddr */
dram_addr = sys_addr - hole_offset ;
2012-04-30 00:08:39 +04:00
edac_dbg ( 2 , " using DHAR to translate SysAddr 0x%lx to DramAddr 0x%lx \n " ,
( unsigned long ) sys_addr ,
( unsigned long ) dram_addr ) ;
2009-05-04 22:46:50 +04:00
return dram_addr ;
}
}
/*
* Translate the SysAddr to a DramAddr as shown near the start of
* section 3.4 .4 ( p . 70 ) . Although sys_addr is a 64 - bit value , the k8
* only deals with 40 - bit values . Therefore we discard bits 63 - 40 of
* sys_addr below . If bit 39 of sys_addr is 1 then the bits we
* discard are all 1 s . Otherwise the bits we discard are all 0 s . See
* section 3.4 .2 of AMD publication 24592 : AMD x86 - 64 Architecture
* Programmer ' s Manual Volume 1 Application Programming .
*/
2013-10-19 01:29:07 +04:00
dram_addr = ( sys_addr & GENMASK_ULL ( 39 , 0 ) ) - dram_base ;
2009-05-04 22:46:50 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 2 , " using DRAM Base register to translate SysAddr 0x%lx to DramAddr 0x%lx \n " ,
( unsigned long ) sys_addr , ( unsigned long ) dram_addr ) ;
2009-05-04 22:46:50 +04:00
return dram_addr ;
}
/*
* @ intlv_en is the value of the IntlvEn field from a DRAM Base register
* ( section 3.4 .4 .1 ) . Return the number of bits from a SysAddr that are used
* for node interleaving .
*/
static int num_node_interleave_bits ( unsigned intlv_en )
{
static const int intlv_shift_table [ ] = { 0 , 1 , 0 , 2 , 0 , 0 , 0 , 3 } ;
int n ;
BUG_ON ( intlv_en > 7 ) ;
n = intlv_shift_table [ intlv_en ] ;
return n ;
}
/* Translate the DramAddr given by @dram_addr to an InputAddr. */
static u64 dram_addr_to_input_addr ( struct mem_ctl_info * mci , u64 dram_addr )
{
struct amd64_pvt * pvt ;
int intlv_shift ;
u64 input_addr ;
pvt = mci - > pvt_info ;
/*
* See the start of section 3.4 .4 ( p . 70 , BKDG # 26094 , K8 , revA - E )
* concerning translating a DramAddr to an InputAddr .
*/
2010-10-21 20:52:53 +04:00
intlv_shift = num_node_interleave_bits ( dram_intlv_en ( pvt , 0 ) ) ;
2013-10-19 01:29:07 +04:00
input_addr = ( ( dram_addr > > intlv_shift ) & GENMASK_ULL ( 35 , 12 ) ) +
2010-12-13 21:21:07 +03:00
( dram_addr & 0xfff ) ;
2009-05-04 22:46:50 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 2 , " Intlv Shift=%d DramAddr=0x%lx maps to InputAddr=0x%lx \n " ,
intlv_shift , ( unsigned long ) dram_addr ,
( unsigned long ) input_addr ) ;
2009-05-04 22:46:50 +04:00
return input_addr ;
}
/*
* Translate the SysAddr represented by @ sys_addr to an InputAddr . It is
* assumed that @ sys_addr maps to the node given by mci .
*/
static u64 sys_addr_to_input_addr ( struct mem_ctl_info * mci , u64 sys_addr )
{
u64 input_addr ;
input_addr =
dram_addr_to_input_addr ( mci , sys_addr_to_dram_addr ( mci , sys_addr ) ) ;
2016-02-08 14:53:12 +03:00
edac_dbg ( 2 , " SysAddr 0x%lx translates to InputAddr 0x%lx \n " ,
2012-04-30 00:08:39 +04:00
( unsigned long ) sys_addr , ( unsigned long ) input_addr ) ;
2009-05-04 22:46:50 +04:00
return input_addr ;
}
/* Map the Error address to a PAGE and PAGE OFFSET. */
static inline void error_address_to_page_and_offset ( u64 error_address ,
2012-08-30 20:01:36 +04:00
struct err_info * err )
2009-05-04 22:46:50 +04:00
{
2012-08-30 20:01:36 +04:00
err - > page = ( u32 ) ( error_address > > PAGE_SHIFT ) ;
err - > offset = ( ( u32 ) error_address ) & ~ PAGE_MASK ;
2009-05-04 22:46:50 +04:00
}
/*
* @ sys_addr is an error address ( a SysAddr ) extracted from the MCA NB Address
* Low ( section 3.6 .4 .5 ) and MCA NB Address High ( section 3.6 .4 .6 ) registers
* of a node that detected an ECC memory error . mci represents the node that
* the error address maps to ( possibly different from the node that detected
* the error ) . Return the number of the csrow that sys_addr maps to , or - 1 on
* error .
*/
static int sys_addr_to_csrow ( struct mem_ctl_info * mci , u64 sys_addr )
{
int csrow ;
csrow = input_addr_to_csrow ( mci , sys_addr_to_input_addr ( mci , sys_addr ) ) ;
if ( csrow = = - 1 )
2010-10-07 20:29:15 +04:00
amd64_mc_err ( mci , " Failed to translate InputAddr to csrow for "
" address 0x%lx \n " , ( unsigned long ) sys_addr ) ;
2009-05-04 22:46:50 +04:00
return csrow ;
}
2009-04-27 17:57:12 +04:00
2009-11-12 21:05:07 +03:00
static int get_channel_from_ecc_syndrome ( struct mem_ctl_info * , u16 ) ;
2009-04-27 18:09:09 +04:00
/*
* Determine if the DIMMs have ECC enabled . ECC is enabled ONLY if all the DIMMs
* are ECC capable .
*/
2013-12-15 20:54:27 +04:00
static unsigned long determine_edac_cap ( struct amd64_pvt * pvt )
2009-04-27 18:09:09 +04:00
{
2011-10-06 10:30:25 +04:00
unsigned long edac_cap = EDAC_FLAG_NONE ;
2016-11-18 01:57:40 +03:00
u8 bit ;
if ( pvt - > umc ) {
u8 i , umc_en_mask = 0 , dimm_ecc_en_mask = 0 ;
2009-04-27 18:09:09 +04:00
2019-02-28 18:36:10 +03:00
for_each_umc ( i ) {
2016-11-18 01:57:40 +03:00
if ( ! ( pvt - > umc [ i ] . sdp_ctrl & UMC_SDP_INIT ) )
continue ;
2009-04-27 18:09:09 +04:00
2016-11-18 01:57:40 +03:00
umc_en_mask | = BIT ( i ) ;
/* UMC Configuration bit 12 (DimmEccEn) */
if ( pvt - > umc [ i ] . umc_cfg & BIT ( 12 ) )
dimm_ecc_en_mask | = BIT ( i ) ;
}
if ( umc_en_mask = = dimm_ecc_en_mask )
edac_cap = EDAC_FLAG_SECDED ;
} else {
bit = ( pvt - > fam > 0xf | | pvt - > ext_model > = K8_REV_F )
? 19
: 17 ;
if ( pvt - > dclr0 & BIT ( bit ) )
edac_cap = EDAC_FLAG_SECDED ;
}
2009-04-27 18:09:09 +04:00
return edac_cap ;
}
2013-12-15 20:54:27 +04:00
static void debug_display_dimm_sizes ( struct amd64_pvt * , u8 ) ;
2009-04-27 18:09:09 +04:00
2013-12-15 20:54:27 +04:00
static void debug_dump_dramcfg_low ( struct amd64_pvt * pvt , u32 dclr , int chan )
2009-11-03 18:18:33 +03:00
{
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " F2x%d90 (DRAM Cfg Low): 0x%08x \n " , chan , dclr ) ;
2009-11-03 18:18:33 +03:00
2014-10-30 14:16:09 +03:00
if ( pvt - > dram_type = = MEM_LRDDR3 ) {
u32 dcsm = pvt - > csels [ chan ] . csmasks [ 0 ] ;
/*
* It ' s assumed all LRDIMMs in a DCT are going to be of
* same ' type ' until proven otherwise . So , use a cs
* value of ' 0 ' here to get dcsm value .
*/
edac_dbg ( 1 , " LRDIMM %dx rank multiply \n " , ( dcsm & 0x3 ) ) ;
}
edac_dbg ( 1 , " All DIMMs support ECC:%s \n " ,
( dclr & BIT ( 19 ) ) ? " yes " : " no " ) ;
2009-11-03 18:18:33 +03:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " PAR/ERR parity: %s \n " ,
( dclr & BIT ( 8 ) ) ? " enabled " : " disabled " ) ;
2009-11-03 18:18:33 +03:00
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0x10 )
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " DCT 128bit mode width: %s \n " ,
( dclr & BIT ( 11 ) ) ? " 128b " : " 64b " ) ;
2009-11-03 18:18:33 +03:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " x4 logical DIMMs present: L0: %s L1: %s L2: %s L3: %s \n " ,
( dclr & BIT ( 12 ) ) ? " yes " : " no " ,
( dclr & BIT ( 13 ) ) ? " yes " : " no " ,
( dclr & BIT ( 14 ) ) ? " yes " : " no " ,
( dclr & BIT ( 15 ) ) ? " yes " : " no " ) ;
2009-11-03 18:18:33 +03:00
}
2019-08-22 02:59:59 +03:00
# define CS_EVEN_PRIMARY BIT(0)
# define CS_ODD_PRIMARY BIT(1)
2019-08-22 03:00:02 +03:00
# define CS_EVEN_SECONDARY BIT(2)
# define CS_ODD_SECONDARY BIT(3)
2019-08-22 02:59:59 +03:00
2019-08-22 03:00:02 +03:00
# define CS_EVEN (CS_EVEN_PRIMARY | CS_EVEN_SECONDARY)
# define CS_ODD (CS_ODD_PRIMARY | CS_ODD_SECONDARY)
2019-08-22 02:59:59 +03:00
static int f17_get_cs_mode ( int dimm , u8 ctrl , struct amd64_pvt * pvt )
2019-02-28 18:36:12 +03:00
{
2019-08-22 02:59:59 +03:00
int cs_mode = 0 ;
2019-02-28 18:36:12 +03:00
2019-08-22 02:59:59 +03:00
if ( csrow_enabled ( 2 * dimm , ctrl , pvt ) )
cs_mode | = CS_EVEN_PRIMARY ;
2019-02-28 18:36:12 +03:00
2019-08-22 02:59:59 +03:00
if ( csrow_enabled ( 2 * dimm + 1 , ctrl , pvt ) )
cs_mode | = CS_ODD_PRIMARY ;
2019-08-22 03:00:02 +03:00
/* Asymmetric dual-rank DIMM support. */
if ( csrow_sec_enabled ( 2 * dimm + 1 , ctrl , pvt ) )
cs_mode | = CS_ODD_SECONDARY ;
2019-08-22 02:59:59 +03:00
return cs_mode ;
2019-02-28 18:36:12 +03:00
}
2016-11-28 17:50:21 +03:00
static void debug_display_dimm_sizes_df ( struct amd64_pvt * pvt , u8 ctrl )
{
2019-08-22 02:59:59 +03:00
int dimm , size0 , size1 , cs0 , cs1 , cs_mode ;
2016-11-28 17:50:21 +03:00
edac_printk ( KERN_DEBUG , EDAC_MC , " UMC%d chip selects: \n " , ctrl ) ;
2019-08-22 02:59:55 +03:00
for ( dimm = 0 ; dimm < 2 ; dimm + + ) {
2017-04-27 20:11:54 +03:00
cs0 = dimm * 2 ;
cs1 = dimm * 2 + 1 ;
2019-08-22 02:59:59 +03:00
cs_mode = f17_get_cs_mode ( dimm , ctrl , pvt ) ;
size0 = pvt - > ops - > dbam_to_cs ( pvt , ctrl , cs_mode , cs0 ) ;
size1 = pvt - > ops - > dbam_to_cs ( pvt , ctrl , cs_mode , cs1 ) ;
2016-11-28 17:50:21 +03:00
amd64_info ( EDAC_MC " : %d: %5dMB %d: %5dMB \n " ,
2017-04-27 20:11:54 +03:00
cs0 , size0 ,
cs1 , size1 ) ;
2016-11-28 17:50:21 +03:00
}
}
static void __dump_misc_regs_df ( struct amd64_pvt * pvt )
{
struct amd64_umc * umc ;
u32 i , tmp , umc_base ;
2019-02-28 18:36:10 +03:00
for_each_umc ( i ) {
2016-11-28 17:50:21 +03:00
umc_base = get_umc_base ( i ) ;
umc = & pvt - > umc [ i ] ;
edac_dbg ( 1 , " UMC%d DIMM cfg: 0x%x \n " , i , umc - > dimm_cfg ) ;
edac_dbg ( 1 , " UMC%d UMC cfg: 0x%x \n " , i , umc - > umc_cfg ) ;
edac_dbg ( 1 , " UMC%d SDP ctrl: 0x%x \n " , i , umc - > sdp_ctrl ) ;
edac_dbg ( 1 , " UMC%d ECC ctrl: 0x%x \n " , i , umc - > ecc_ctrl ) ;
amd_smn_read ( pvt - > mc_node_id , umc_base + UMCCH_ECC_BAD_SYMBOL , & tmp ) ;
edac_dbg ( 1 , " UMC%d ECC bad symbol: 0x%x \n " , i , tmp ) ;
amd_smn_read ( pvt - > mc_node_id , umc_base + UMCCH_UMC_CAP , & tmp ) ;
edac_dbg ( 1 , " UMC%d UMC cap: 0x%x \n " , i , tmp ) ;
edac_dbg ( 1 , " UMC%d UMC cap high: 0x%x \n " , i , umc - > umc_cap_hi ) ;
edac_dbg ( 1 , " UMC%d ECC capable: %s, ChipKill ECC capable: %s \n " ,
i , ( umc - > umc_cap_hi & BIT ( 30 ) ) ? " yes " : " no " ,
( umc - > umc_cap_hi & BIT ( 31 ) ) ? " yes " : " no " ) ;
edac_dbg ( 1 , " UMC%d All DIMMs support ECC: %s \n " ,
i , ( umc - > umc_cfg & BIT ( 12 ) ) ? " yes " : " no " ) ;
edac_dbg ( 1 , " UMC%d x4 DIMMs present: %s \n " ,
i , ( umc - > dimm_cfg & BIT ( 6 ) ) ? " yes " : " no " ) ;
edac_dbg ( 1 , " UMC%d x16 DIMMs present: %s \n " ,
i , ( umc - > dimm_cfg & BIT ( 7 ) ) ? " yes " : " no " ) ;
if ( pvt - > dram_type = = MEM_LRDDR4 ) {
amd_smn_read ( pvt - > mc_node_id , umc_base + UMCCH_ADDR_CFG , & tmp ) ;
edac_dbg ( 1 , " UMC%d LRDIMM %dx rank multiply \n " ,
i , 1 < < ( ( tmp > > 4 ) & 0x3 ) ) ;
}
debug_display_dimm_sizes_df ( pvt , i ) ;
}
edac_dbg ( 1 , " F0x104 (DRAM Hole Address): 0x%08x, base: 0x%08x \n " ,
pvt - > dhar , dhar_base ( pvt ) ) ;
}
2009-04-27 18:09:09 +04:00
/* Display and decode various NB registers for debug purposes. */
2016-11-28 17:50:21 +03:00
static void __dump_misc_regs ( struct amd64_pvt * pvt )
2009-04-27 18:09:09 +04:00
{
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " F3xE8 (NB Cap): 0x%08x \n " , pvt - > nbcap ) ;
2009-11-03 18:18:33 +03:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " NB two channel DRAM capable: %s \n " ,
( pvt - > nbcap & NBCAP_DCT_DUAL ) ? " yes " : " no " ) ;
2009-04-27 18:09:09 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " ECC capable: %s, ChipKill ECC capable: %s \n " ,
( pvt - > nbcap & NBCAP_SECDED ) ? " yes " : " no " ,
( pvt - > nbcap & NBCAP_CHIPKILL ) ? " yes " : " no " ) ;
2009-11-03 18:18:33 +03:00
2013-12-15 20:54:27 +04:00
debug_dump_dramcfg_low ( pvt , pvt - > dclr0 , 0 ) ;
2009-04-27 18:09:09 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " F3xB0 (Online Spare): 0x%08x \n " , pvt - > online_spare ) ;
2009-04-27 18:09:09 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " F1xF0 (DRAM Hole Address): 0x%08x, base: 0x%08x, offset: 0x%08x \n " ,
pvt - > dhar , dhar_base ( pvt ) ,
2013-08-10 15:54:48 +04:00
( pvt - > fam = = 0xf ) ? k8_dhar_offset ( pvt )
: f10_dhar_offset ( pvt ) ) ;
2009-04-27 18:09:09 +04:00
2013-12-15 20:54:27 +04:00
debug_display_dimm_sizes ( pvt , 0 ) ;
2011-02-03 17:59:57 +03:00
2009-10-16 15:39:30 +04:00
/* everything below this point is Fam10h and above */
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0xf )
2009-04-27 18:09:09 +04:00
return ;
2011-02-03 17:59:57 +03:00
2013-12-15 20:54:27 +04:00
debug_display_dimm_sizes ( pvt , 1 ) ;
2009-04-27 18:09:09 +04:00
2009-10-16 15:39:30 +04:00
/* Only if NOT ganged does dclr1 have valid info */
2009-11-03 18:18:33 +03:00
if ( ! dct_ganging_enabled ( pvt ) )
2013-12-15 20:54:27 +04:00
debug_dump_dramcfg_low ( pvt , pvt - > dclr1 , 1 ) ;
2009-04-27 18:09:09 +04:00
}
2016-11-28 17:50:21 +03:00
/* Display and decode various NB registers for debug purposes. */
static void dump_misc_regs ( struct amd64_pvt * pvt )
{
if ( pvt - > umc )
__dump_misc_regs_df ( pvt ) ;
else
__dump_misc_regs ( pvt ) ;
edac_dbg ( 1 , " DramHoleValid: %s \n " , dhar_valid ( pvt ) ? " yes " : " no " ) ;
2019-02-28 18:36:11 +03:00
amd64_info ( " using x%u syndromes. \n " , pvt - > ecc_sym_sz ) ;
2016-11-28 17:50:21 +03:00
}
2009-04-27 18:12:00 +04:00
/*
2013-08-09 20:54:49 +04:00
* See BKDG , F2x [ 1 , 0 ] [ 5 C : 40 ] , F2 [ 1 , 0 ] [ 6 C : 60 ]
2009-04-27 18:12:00 +04:00
*/
2010-11-29 21:49:02 +03:00
static void prep_chip_selects ( struct amd64_pvt * pvt )
2009-04-27 18:12:00 +04:00
{
2013-08-09 20:54:49 +04:00
if ( pvt - > fam = = 0xf & & pvt - > ext_model < K8_REV_F ) {
2010-11-29 21:49:02 +03:00
pvt - > csels [ 0 ] . b_cnt = pvt - > csels [ 1 ] . b_cnt = 8 ;
pvt - > csels [ 0 ] . m_cnt = pvt - > csels [ 1 ] . m_cnt = 8 ;
2014-10-30 14:16:09 +03:00
} else if ( pvt - > fam = = 0x15 & & pvt - > model = = 0x30 ) {
2013-08-09 20:54:49 +04:00
pvt - > csels [ 0 ] . b_cnt = pvt - > csels [ 1 ] . b_cnt = 4 ;
pvt - > csels [ 0 ] . m_cnt = pvt - > csels [ 1 ] . m_cnt = 2 ;
2019-08-22 02:59:55 +03:00
} else if ( pvt - > fam > = 0x17 ) {
int umc ;
for_each_umc ( umc ) {
pvt - > csels [ umc ] . b_cnt = 4 ;
pvt - > csels [ umc ] . m_cnt = 2 ;
}
2009-09-21 16:35:51 +04:00
} else {
2010-11-29 21:49:02 +03:00
pvt - > csels [ 0 ] . b_cnt = pvt - > csels [ 1 ] . b_cnt = 8 ;
pvt - > csels [ 0 ] . m_cnt = pvt - > csels [ 1 ] . m_cnt = 4 ;
2009-04-27 18:12:00 +04:00
}
}
2019-08-22 02:59:55 +03:00
static void read_umc_base_mask ( struct amd64_pvt * pvt )
{
2019-08-22 03:00:01 +03:00
u32 umc_base_reg , umc_base_reg_sec ;
u32 umc_mask_reg , umc_mask_reg_sec ;
u32 base_reg , base_reg_sec ;
u32 mask_reg , mask_reg_sec ;
u32 * base , * base_sec ;
u32 * mask , * mask_sec ;
2019-08-22 02:59:55 +03:00
int cs , umc ;
for_each_umc ( umc ) {
umc_base_reg = get_umc_base ( umc ) + UMCCH_BASE_ADDR ;
2019-08-22 03:00:01 +03:00
umc_base_reg_sec = get_umc_base ( umc ) + UMCCH_BASE_ADDR_SEC ;
2019-08-22 02:59:55 +03:00
for_each_chip_select ( cs , umc , pvt ) {
base = & pvt - > csels [ umc ] . csbases [ cs ] ;
2019-08-22 03:00:01 +03:00
base_sec = & pvt - > csels [ umc ] . csbases_sec [ cs ] ;
2019-08-22 02:59:55 +03:00
base_reg = umc_base_reg + ( cs * 4 ) ;
2019-08-22 03:00:01 +03:00
base_reg_sec = umc_base_reg_sec + ( cs * 4 ) ;
2019-08-22 02:59:55 +03:00
if ( ! amd_smn_read ( pvt - > mc_node_id , base_reg , base ) )
edac_dbg ( 0 , " DCSB%d[%d]=0x%08x reg: 0x%x \n " ,
umc , cs , * base , base_reg ) ;
2019-08-22 03:00:01 +03:00
if ( ! amd_smn_read ( pvt - > mc_node_id , base_reg_sec , base_sec ) )
edac_dbg ( 0 , " DCSB_SEC%d[%d]=0x%08x reg: 0x%x \n " ,
umc , cs , * base_sec , base_reg_sec ) ;
2019-08-22 02:59:55 +03:00
}
umc_mask_reg = get_umc_base ( umc ) + UMCCH_ADDR_MASK ;
2019-08-22 03:00:01 +03:00
umc_mask_reg_sec = get_umc_base ( umc ) + UMCCH_ADDR_MASK_SEC ;
2019-08-22 02:59:55 +03:00
for_each_chip_select_mask ( cs , umc , pvt ) {
mask = & pvt - > csels [ umc ] . csmasks [ cs ] ;
2019-08-22 03:00:01 +03:00
mask_sec = & pvt - > csels [ umc ] . csmasks_sec [ cs ] ;
2019-08-22 02:59:55 +03:00
mask_reg = umc_mask_reg + ( cs * 4 ) ;
2019-08-22 03:00:01 +03:00
mask_reg_sec = umc_mask_reg_sec + ( cs * 4 ) ;
2019-08-22 02:59:55 +03:00
if ( ! amd_smn_read ( pvt - > mc_node_id , mask_reg , mask ) )
edac_dbg ( 0 , " DCSM%d[%d]=0x%08x reg: 0x%x \n " ,
umc , cs , * mask , mask_reg ) ;
2019-08-22 03:00:01 +03:00
if ( ! amd_smn_read ( pvt - > mc_node_id , mask_reg_sec , mask_sec ) )
edac_dbg ( 0 , " DCSM_SEC%d[%d]=0x%08x reg: 0x%x \n " ,
umc , cs , * mask_sec , mask_reg_sec ) ;
2019-08-22 02:59:55 +03:00
}
}
}
2009-04-27 18:12:00 +04:00
/*
2010-11-29 21:49:02 +03:00
* Function 2 Offset F10_DCSB0 ; read in the DCS Base and DCS Mask registers
2009-04-27 18:12:00 +04:00
*/
2010-10-08 20:32:29 +04:00
static void read_dct_base_mask ( struct amd64_pvt * pvt )
2009-04-27 18:12:00 +04:00
{
2019-08-22 02:59:55 +03:00
int cs ;
2009-04-27 18:12:00 +04:00
2010-11-29 21:49:02 +03:00
prep_chip_selects ( pvt ) ;
2009-04-27 18:12:00 +04:00
2019-08-22 02:59:55 +03:00
if ( pvt - > umc )
return read_umc_base_mask ( pvt ) ;
2016-11-18 01:57:37 +03:00
2010-11-29 21:49:02 +03:00
for_each_chip_select ( cs , 0 , pvt ) {
2019-08-22 02:59:55 +03:00
int reg0 = DCSB0 + ( cs * 4 ) ;
int reg1 = DCSB1 + ( cs * 4 ) ;
2010-11-29 21:49:02 +03:00
u32 * base0 = & pvt - > csels [ 0 ] . csbases [ cs ] ;
u32 * base1 = & pvt - > csels [ 1 ] . csbases [ cs ] ;
2010-10-08 20:32:29 +04:00
2019-08-22 02:59:55 +03:00
if ( ! amd64_read_dct_pci_cfg ( pvt , 0 , reg0 , base0 ) )
edac_dbg ( 0 , " DCSB0[%d]=0x%08x reg: F2x%x \n " ,
cs , * base0 , reg0 ) ;
2019-04-25 17:30:34 +03:00
2019-08-22 02:59:55 +03:00
if ( pvt - > fam = = 0xf )
continue ;
2016-11-18 01:57:37 +03:00
2019-08-22 02:59:55 +03:00
if ( ! amd64_read_dct_pci_cfg ( pvt , 1 , reg0 , base1 ) )
edac_dbg ( 0 , " DCSB1[%d]=0x%08x reg: F2x%x \n " ,
cs , * base1 , ( pvt - > fam = = 0x10 ) ? reg1
: reg0 ) ;
2009-04-27 18:12:00 +04:00
}
2010-11-29 21:49:02 +03:00
for_each_chip_select_mask ( cs , 0 , pvt ) {
2019-08-22 02:59:55 +03:00
int reg0 = DCSM0 + ( cs * 4 ) ;
int reg1 = DCSM1 + ( cs * 4 ) ;
2010-11-29 21:49:02 +03:00
u32 * mask0 = & pvt - > csels [ 0 ] . csmasks [ cs ] ;
u32 * mask1 = & pvt - > csels [ 1 ] . csmasks [ cs ] ;
2010-10-08 20:32:29 +04:00
2019-08-22 02:59:55 +03:00
if ( ! amd64_read_dct_pci_cfg ( pvt , 0 , reg0 , mask0 ) )
edac_dbg ( 0 , " DCSM0[%d]=0x%08x reg: F2x%x \n " ,
cs , * mask0 , reg0 ) ;
2016-11-18 01:57:37 +03:00
2019-08-22 02:59:55 +03:00
if ( pvt - > fam = = 0xf )
continue ;
2019-04-25 17:30:34 +03:00
2019-08-22 02:59:55 +03:00
if ( ! amd64_read_dct_pci_cfg ( pvt , 1 , reg0 , mask1 ) )
edac_dbg ( 0 , " DCSM1[%d]=0x%08x reg: F2x%x \n " ,
cs , * mask1 , ( pvt - > fam = = 0x10 ) ? reg1
: reg0 ) ;
2009-04-27 18:12:00 +04:00
}
}
2014-10-30 14:16:09 +03:00
static void determine_memory_type ( struct amd64_pvt * pvt )
2009-04-27 18:12:00 +04:00
{
2014-10-30 14:16:09 +03:00
u32 dram_ctrl , dcsm ;
2009-04-27 18:12:00 +04:00
2020-01-10 04:56:51 +03:00
if ( pvt - > umc ) {
if ( ( pvt - > umc [ 0 ] . dimm_cfg | pvt - > umc [ 1 ] . dimm_cfg ) & BIT ( 5 ) )
pvt - > dram_type = MEM_LRDDR4 ;
else if ( ( pvt - > umc [ 0 ] . dimm_cfg | pvt - > umc [ 1 ] . dimm_cfg ) & BIT ( 4 ) )
pvt - > dram_type = MEM_RDDR4 ;
else
pvt - > dram_type = MEM_DDR4 ;
return ;
}
2014-10-30 14:16:09 +03:00
switch ( pvt - > fam ) {
case 0xf :
if ( pvt - > ext_model > = K8_REV_F )
goto ddr3 ;
pvt - > dram_type = ( pvt - > dclr0 & BIT ( 18 ) ) ? MEM_DDR : MEM_RDDR ;
return ;
case 0x10 :
2009-11-12 17:37:57 +03:00
if ( pvt - > dchr0 & DDR3_MODE )
2014-10-30 14:16:09 +03:00
goto ddr3 ;
pvt - > dram_type = ( pvt - > dclr0 & BIT ( 16 ) ) ? MEM_DDR2 : MEM_RDDR2 ;
return ;
case 0x15 :
if ( pvt - > model < 0x60 )
goto ddr3 ;
/*
* Model 0x60 h needs special handling :
*
* We use a Chip Select value of ' 0 ' to obtain dcsm .
* Theoretically , it is possible to populate LRDIMMs of different
* ' Rank ' value on a DCT . But this is not the common case . So ,
* it ' s reasonable to assume all DIMMs are going to be of same
* ' type ' until proven otherwise .
*/
amd64_read_dct_pci_cfg ( pvt , 0 , DRAM_CONTROL , & dram_ctrl ) ;
dcsm = pvt - > csels [ 0 ] . csmasks [ 0 ] ;
if ( ( ( dram_ctrl > > 8 ) & 0x7 ) = = 0x2 )
pvt - > dram_type = MEM_DDR4 ;
else if ( pvt - > dclr0 & BIT ( 16 ) )
pvt - > dram_type = MEM_DDR3 ;
else if ( dcsm & 0x3 )
pvt - > dram_type = MEM_LRDDR3 ;
2009-11-12 17:37:57 +03:00
else
2014-10-30 14:16:09 +03:00
pvt - > dram_type = MEM_RDDR3 ;
2009-04-27 18:12:00 +04:00
2014-10-30 14:16:09 +03:00
return ;
case 0x16 :
goto ddr3 ;
default :
WARN ( 1 , KERN_ERR " %s: Family??? 0x%x \n " , __func__ , pvt - > fam ) ;
pvt - > dram_type = MEM_EMPTY ;
}
return ;
2009-04-27 18:12:00 +04:00
2014-10-30 14:16:09 +03:00
ddr3 :
pvt - > dram_type = ( pvt - > dclr0 & BIT ( 16 ) ) ? MEM_DDR3 : MEM_RDDR3 ;
2009-04-27 18:12:00 +04:00
}
2010-12-22 16:28:24 +03:00
/* Get the number of DCT channels the memory controller is using. */
2009-04-27 18:14:52 +04:00
static int k8_early_channel_count ( struct amd64_pvt * pvt )
{
2010-12-22 16:28:24 +03:00
int flag ;
2009-04-27 18:14:52 +04:00
2010-10-01 21:44:53 +04:00
if ( pvt - > ext_model > = K8_REV_F )
2009-04-27 18:14:52 +04:00
/* RevF (NPT) and later */
2011-01-18 21:16:08 +03:00
flag = pvt - > dclr0 & WIDTH_128 ;
2010-10-01 21:44:53 +04:00
else
2009-04-27 18:14:52 +04:00
/* RevE and earlier */
flag = pvt - > dclr0 & REVE_WIDTH_128 ;
/* not used */
pvt - > dclr1 = 0 ;
return ( flag ) ? 2 : 1 ;
}
2011-01-10 16:37:27 +03:00
/* On F10h and later ErrAddr is MC4_ADDR[47:1] */
2013-08-10 15:54:48 +04:00
static u64 get_error_address ( struct amd64_pvt * pvt , struct mce * m )
2009-04-27 18:14:52 +04:00
{
2020-11-10 00:06:57 +03:00
u16 mce_nid = topology_die_id ( m - > extcpu ) ;
2015-02-17 12:58:34 +03:00
struct mem_ctl_info * mci ;
2011-01-10 16:37:27 +03:00
u8 start_bit = 1 ;
u8 end_bit = 47 ;
2015-02-17 12:58:34 +03:00
u64 addr ;
mci = edac_mc_find ( mce_nid ) ;
if ( ! mci )
return 0 ;
pvt = mci - > pvt_info ;
2011-01-10 16:37:27 +03:00
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0xf ) {
2011-01-10 16:37:27 +03:00
start_bit = 3 ;
end_bit = 39 ;
}
2013-10-19 01:29:07 +04:00
addr = m - > addr & GENMASK_ULL ( end_bit , start_bit ) ;
2011-03-30 17:42:10 +04:00
/*
* Erratum 637 workaround
*/
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0x15 ) {
2011-03-30 17:42:10 +04:00
u64 cc6_base , tmp_addr ;
u32 tmp ;
2012-11-27 10:32:10 +04:00
u8 intlv_en ;
2011-03-30 17:42:10 +04:00
2013-10-19 01:29:07 +04:00
if ( ( addr & GENMASK_ULL ( 47 , 24 ) ) > > 24 ! = 0x00fdf7 )
2011-03-30 17:42:10 +04:00
return addr ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_LOCAL_NODE_LIM , & tmp ) ;
intlv_en = tmp > > 21 & 0x7 ;
/* add [47:27] + 3 trailing bits */
2013-10-19 01:29:07 +04:00
cc6_base = ( tmp & GENMASK_ULL ( 20 , 0 ) ) < < 3 ;
2011-03-30 17:42:10 +04:00
/* reverse and add DramIntlvEn */
cc6_base | = intlv_en ^ 0x7 ;
/* pin at [47:24] */
cc6_base < < = 24 ;
if ( ! intlv_en )
2013-10-19 01:29:07 +04:00
return cc6_base | ( addr & GENMASK_ULL ( 23 , 0 ) ) ;
2011-03-30 17:42:10 +04:00
amd64_read_pci_cfg ( pvt - > F1 , DRAM_LOCAL_NODE_BASE , & tmp ) ;
/* faster log2 */
2013-10-19 01:29:07 +04:00
tmp_addr = ( addr & GENMASK_ULL ( 23 , 12 ) ) < < __fls ( intlv_en + 1 ) ;
2011-03-30 17:42:10 +04:00
/* OR DramIntlvSel into bits [14:12] */
2013-10-19 01:29:07 +04:00
tmp_addr | = ( tmp & GENMASK_ULL ( 23 , 21 ) ) > > 9 ;
2011-03-30 17:42:10 +04:00
/* add remaining [11:0] bits from original MC4_ADDR */
2013-10-19 01:29:07 +04:00
tmp_addr | = addr & GENMASK_ULL ( 11 , 0 ) ;
2011-03-30 17:42:10 +04:00
return cc6_base | tmp_addr ;
}
return addr ;
2009-04-27 18:14:52 +04:00
}
2012-11-30 12:44:19 +04:00
static struct pci_dev * pci_get_related_function ( unsigned int vendor ,
unsigned int device ,
struct pci_dev * related )
{
struct pci_dev * dev = NULL ;
while ( ( dev = pci_get_device ( vendor , device , dev ) ) ) {
if ( pci_domain_nr ( dev - > bus ) = = pci_domain_nr ( related - > bus ) & &
( dev - > bus - > number = = related - > bus - > number ) & &
( PCI_SLOT ( dev - > devfn ) = = PCI_SLOT ( related - > devfn ) ) )
break ;
}
return dev ;
}
2010-10-21 20:52:53 +04:00
static void read_dram_base_limit_regs ( struct amd64_pvt * pvt , unsigned range )
2009-04-27 18:14:52 +04:00
{
2012-11-30 12:44:19 +04:00
struct amd_northbridge * nb ;
2013-08-09 20:54:49 +04:00
struct pci_dev * f1 = NULL ;
unsigned int pci_func ;
2011-02-21 21:37:24 +03:00
int off = range < < 3 ;
2012-11-30 12:44:19 +04:00
u32 llim ;
2009-04-27 18:14:52 +04:00
2010-10-21 20:52:53 +04:00
amd64_read_pci_cfg ( pvt - > F1 , DRAM_BASE_LO + off , & pvt - > ranges [ range ] . base . lo ) ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_LIMIT_LO + off , & pvt - > ranges [ range ] . lim . lo ) ;
2009-04-27 18:14:52 +04:00
2013-08-09 20:54:49 +04:00
if ( pvt - > fam = = 0xf )
2010-10-21 20:52:53 +04:00
return ;
2009-04-27 18:14:52 +04:00
2010-10-21 20:52:53 +04:00
if ( ! dram_rw ( pvt , range ) )
return ;
2009-04-27 18:14:52 +04:00
2010-10-21 20:52:53 +04:00
amd64_read_pci_cfg ( pvt - > F1 , DRAM_BASE_HI + off , & pvt - > ranges [ range ] . base . hi ) ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_LIMIT_HI + off , & pvt - > ranges [ range ] . lim . hi ) ;
2011-03-21 22:45:06 +03:00
2012-11-30 12:44:19 +04:00
/* F15h: factor in CC6 save area by reading dst node's limit reg */
2013-08-09 20:54:49 +04:00
if ( pvt - > fam ! = 0x15 )
2012-11-30 12:44:19 +04:00
return ;
2011-03-21 22:45:06 +03:00
2012-11-30 12:44:19 +04:00
nb = node_to_amd_nb ( dram_dst_node ( pvt , range ) ) ;
if ( WARN_ON ( ! nb ) )
return ;
2011-03-21 22:45:06 +03:00
2014-10-30 14:16:09 +03:00
if ( pvt - > model = = 0x60 )
pci_func = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1 ;
else if ( pvt - > model = = 0x30 )
pci_func = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1 ;
else
pci_func = PCI_DEVICE_ID_AMD_15H_NB_F1 ;
2013-08-09 20:54:49 +04:00
f1 = pci_get_related_function ( nb - > misc - > vendor , pci_func , nb - > misc ) ;
2012-11-30 12:44:19 +04:00
if ( WARN_ON ( ! f1 ) )
return ;
2011-03-21 22:45:06 +03:00
2012-11-30 12:44:19 +04:00
amd64_read_pci_cfg ( f1 , DRAM_LOCAL_NODE_LIM , & llim ) ;
2011-03-21 22:45:06 +03:00
2013-10-19 01:29:07 +04:00
pvt - > ranges [ range ] . lim . lo & = GENMASK_ULL ( 15 , 0 ) ;
2011-03-21 22:45:06 +03:00
2012-11-30 12:44:19 +04:00
/* {[39:27],111b} */
pvt - > ranges [ range ] . lim . lo | = ( ( llim & 0x1fff ) < < 3 | 0x7 ) < < 16 ;
2011-03-21 22:45:06 +03:00
2013-10-19 01:29:07 +04:00
pvt - > ranges [ range ] . lim . hi & = GENMASK_ULL ( 7 , 0 ) ;
2011-03-21 22:45:06 +03:00
2012-11-30 12:44:19 +04:00
/* [47:40] */
pvt - > ranges [ range ] . lim . hi | = llim > > 13 ;
pci_dev_put ( f1 ) ;
2009-04-27 18:14:52 +04:00
}
2011-01-10 16:24:32 +03:00
static void k8_map_sysaddr_to_csrow ( struct mem_ctl_info * mci , u64 sys_addr ,
2012-08-30 20:01:36 +04:00
struct err_info * err )
2009-04-27 18:14:52 +04:00
{
2011-01-10 16:24:32 +03:00
struct amd64_pvt * pvt = mci - > pvt_info ;
2009-04-27 18:14:52 +04:00
2012-08-30 20:01:36 +04:00
error_address_to_page_and_offset ( sys_addr , err ) ;
2012-04-16 22:03:50 +04:00
/*
* Find out which node the error address belongs to . This may be
* different from the node that detected the error .
*/
2012-08-30 20:01:36 +04:00
err - > src_mci = find_mc_by_sys_addr ( mci , sys_addr ) ;
if ( ! err - > src_mci ) {
2012-04-16 22:03:50 +04:00
amd64_mc_err ( mci , " failed to map error addr 0x%lx to a node \n " ,
( unsigned long ) sys_addr ) ;
2012-08-30 20:01:36 +04:00
err - > err_code = ERR_NODE ;
2012-04-16 22:03:50 +04:00
return ;
}
/* Now map the sys_addr to a CSROW */
2012-08-30 20:01:36 +04:00
err - > csrow = sys_addr_to_csrow ( err - > src_mci , sys_addr ) ;
if ( err - > csrow < 0 ) {
err - > err_code = ERR_CSROW ;
2012-04-16 22:03:50 +04:00
return ;
}
2009-04-27 18:14:52 +04:00
/* CHIPKILL enabled */
2011-01-10 16:24:32 +03:00
if ( pvt - > nbcfg & NBCFG_CHIPKILL ) {
2012-08-30 20:01:36 +04:00
err - > channel = get_channel_from_ecc_syndrome ( mci , err - > syndrome ) ;
if ( err - > channel < 0 ) {
2009-04-27 18:14:52 +04:00
/*
* Syndrome didn ' t map , so we don ' t know which of the
* 2 DIMMs is in error . So we need to ID ' both ' of them
* as suspect .
*/
2012-08-30 20:01:36 +04:00
amd64_mc_warn ( err - > src_mci , " unknown syndrome 0x%04x - "
2012-04-16 22:03:50 +04:00
" possible error reporting race \n " ,
2012-08-30 20:01:36 +04:00
err - > syndrome ) ;
err - > err_code = ERR_CHANNEL ;
2009-04-27 18:14:52 +04:00
return ;
}
} else {
/*
* non - chipkill ecc mode
*
* The k8 documentation is unclear about how to determine the
* channel number when using non - chipkill memory . This method
* was obtained from email communication with someone at AMD .
* ( Wish the email was placed in this comment - norsk )
*/
2012-08-30 20:01:36 +04:00
err - > channel = ( ( sys_addr & BIT ( 3 ) ) ! = 0 ) ;
2009-04-27 18:14:52 +04:00
}
}
2011-01-18 21:16:08 +03:00
static int ddr2_cs_size ( unsigned i , bool dct_width )
2009-04-27 18:14:52 +04:00
{
2011-01-18 21:16:08 +03:00
unsigned shift = 0 ;
2009-04-27 18:14:52 +04:00
2011-01-18 21:16:08 +03:00
if ( i < = 2 )
shift = i ;
else if ( ! ( i & 0x1 ) )
shift = i > > 1 ;
2009-10-21 15:44:36 +04:00
else
2011-01-18 21:16:08 +03:00
shift = ( i + 1 ) > > 1 ;
2009-04-27 18:14:52 +04:00
2011-01-18 21:16:08 +03:00
return 128 < < ( shift + ! ! dct_width ) ;
}
static int k8_dbam_to_chip_select ( struct amd64_pvt * pvt , u8 dct ,
2014-10-30 14:16:09 +03:00
unsigned cs_mode , int cs_mask_nr )
2011-01-18 21:16:08 +03:00
{
u32 dclr = dct ? pvt - > dclr1 : pvt - > dclr0 ;
if ( pvt - > ext_model > = K8_REV_F ) {
WARN_ON ( cs_mode > 11 ) ;
return ddr2_cs_size ( cs_mode , dclr & WIDTH_128 ) ;
}
else if ( pvt - > ext_model > = K8_REV_D ) {
2011-11-10 00:28:43 +04:00
unsigned diff ;
2011-01-18 21:16:08 +03:00
WARN_ON ( cs_mode > 10 ) ;
2011-11-10 00:28:43 +04:00
/*
* the below calculation , besides trying to win an obfuscated C
* contest , maps cs_mode values to DIMM chip select sizes . The
* mappings are :
*
* cs_mode CS size ( mb )
* = = = = = = = = = = = = = = = = = = =
* 0 32
* 1 64
* 2 128
* 3 128
* 4 256
* 5 512
* 6 256
* 7 512
* 8 1024
* 9 1024
* 10 2048
*
* Basically , it calculates a value with which to shift the
* smallest CS size of 32 MB .
*
* ddr [ 23 ] _cs_size have a similar purpose .
*/
diff = cs_mode / 3 + ( unsigned ) ( cs_mode > 5 ) ;
return 32 < < ( cs_mode - diff ) ;
2011-01-18 21:16:08 +03:00
}
else {
WARN_ON ( cs_mode > 6 ) ;
return 32 < < cs_mode ;
}
2009-04-27 18:14:52 +04:00
}
2009-04-27 18:16:50 +04:00
/*
* Get the number of DCT channels in use .
*
* Return :
* number of Memory Channels in operation
* Pass back :
* contents of the DCL0_LOW register
*/
2011-01-07 19:58:04 +03:00
static int f1x_early_channel_count ( struct amd64_pvt * pvt )
2009-04-27 18:16:50 +04:00
{
2009-10-13 21:26:55 +04:00
int i , j , channels = 0 ;
2009-04-27 18:16:50 +04:00
2011-01-07 19:58:04 +03:00
/* On F10h, if we are in 128 bit mode, then we are using 2 channels */
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0x10 & & ( pvt - > dclr0 & WIDTH_128 ) )
2011-01-07 19:58:04 +03:00
return 2 ;
2009-04-27 18:16:50 +04:00
/*
2009-10-16 21:55:49 +04:00
* Need to check if in unganged mode : In such , there are 2 channels ,
* but they are not in 128 bit mode and thus the above ' dclr0 ' status
* bit will be OFF .
2009-04-27 18:16:50 +04:00
*
* Need to check DCT0 [ 0 ] and DCT1 [ 0 ] to see if only one of them has
* their CSEnable bit on . If so , then SINGLE DIMM case .
*/
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " Data width is not 128 bits - need more decoding \n " ) ;
2009-04-27 18:14:52 +04:00
2009-04-27 18:16:50 +04:00
/*
* Check DRAM Bank Address Mapping values for each DIMM to see if there
* is more than just one DIMM present in unganged mode . Need to check
* both controllers since DIMMs can be placed in either one .
*/
2010-12-21 17:53:27 +03:00
for ( i = 0 ; i < 2 ; i + + ) {
u32 dbam = ( i ? pvt - > dbam1 : pvt - > dbam0 ) ;
2009-04-27 18:16:50 +04:00
2009-08-07 19:04:49 +04:00
for ( j = 0 ; j < 4 ; j + + ) {
if ( DBAM_DIMM ( j , dbam ) > 0 ) {
channels + + ;
break ;
}
}
2009-04-27 18:16:50 +04:00
}
2009-10-16 21:55:49 +04:00
if ( channels > 2 )
channels = 2 ;
2010-10-07 20:29:15 +04:00
amd64_info ( " MCT channel count: %d \n " , channels ) ;
2009-04-27 18:16:50 +04:00
return channels ;
}
2016-11-18 01:57:35 +03:00
static int f17_early_channel_count ( struct amd64_pvt * pvt )
{
int i , channels = 0 ;
/* SDP Control bit 31 (SdpInit) is clear for unused UMC channels */
2019-02-28 18:36:10 +03:00
for_each_umc ( i )
2016-11-18 01:57:35 +03:00
channels + = ! ! ( pvt - > umc [ i ] . sdp_ctrl & UMC_SDP_INIT ) ;
amd64_info ( " MCT channel count: %d \n " , channels ) ;
return channels ;
}
2011-01-18 21:16:08 +03:00
static int ddr3_cs_size ( unsigned i , bool dct_width )
2009-04-27 18:16:50 +04:00
{
2011-01-18 21:16:08 +03:00
unsigned shift = 0 ;
int cs_size = 0 ;
if ( i = = 0 | | i = = 3 | | i = = 4 )
cs_size = - 1 ;
else if ( i < = 2 )
shift = i ;
else if ( i = = 12 )
shift = 7 ;
else if ( ! ( i & 0x1 ) )
shift = i > > 1 ;
else
shift = ( i + 1 ) > > 1 ;
if ( cs_size ! = - 1 )
cs_size = ( 128 * ( 1 < < ! ! dct_width ) ) < < shift ;
return cs_size ;
}
2014-10-30 14:16:09 +03:00
static int ddr3_lrdimm_cs_size ( unsigned i , unsigned rank_multiply )
{
unsigned shift = 0 ;
int cs_size = 0 ;
if ( i < 4 | | i = = 6 )
cs_size = - 1 ;
else if ( i = = 12 )
shift = 7 ;
else if ( ! ( i & 0x1 ) )
shift = i > > 1 ;
else
shift = ( i + 1 ) > > 1 ;
if ( cs_size ! = - 1 )
cs_size = rank_multiply * ( 128 < < shift ) ;
return cs_size ;
}
static int ddr4_cs_size ( unsigned i )
{
int cs_size = 0 ;
if ( i = = 0 )
cs_size = - 1 ;
else if ( i = = 1 )
cs_size = 1024 ;
else
/* Min cs_size = 1G */
cs_size = 1024 * ( 1 < < ( i > > 1 ) ) ;
return cs_size ;
}
2011-01-18 21:16:08 +03:00
static int f10_dbam_to_chip_select ( struct amd64_pvt * pvt , u8 dct ,
2014-10-30 14:16:09 +03:00
unsigned cs_mode , int cs_mask_nr )
2011-01-18 21:16:08 +03:00
{
u32 dclr = dct ? pvt - > dclr1 : pvt - > dclr0 ;
WARN_ON ( cs_mode > 11 ) ;
2009-10-21 15:44:36 +04:00
if ( pvt - > dchr0 & DDR3_MODE | | pvt - > dchr1 & DDR3_MODE )
2011-01-18 21:16:08 +03:00
return ddr3_cs_size ( cs_mode , dclr & WIDTH_128 ) ;
2009-10-21 15:44:36 +04:00
else
2011-01-18 21:16:08 +03:00
return ddr2_cs_size ( cs_mode , dclr & WIDTH_128 ) ;
}
/*
* F15h supports only 64 bit DCT interfaces
*/
static int f15_dbam_to_chip_select ( struct amd64_pvt * pvt , u8 dct ,
2014-10-30 14:16:09 +03:00
unsigned cs_mode , int cs_mask_nr )
2011-01-18 21:16:08 +03:00
{
WARN_ON ( cs_mode > 12 ) ;
2009-10-21 15:44:36 +04:00
2011-01-18 21:16:08 +03:00
return ddr3_cs_size ( cs_mode , false ) ;
2009-04-27 18:16:50 +04:00
}
2014-10-30 14:16:09 +03:00
/* F15h M60h supports DDR4 mapping as well.. */
static int f15_m60h_dbam_to_chip_select ( struct amd64_pvt * pvt , u8 dct ,
unsigned cs_mode , int cs_mask_nr )
{
int cs_size ;
u32 dcsm = pvt - > csels [ dct ] . csmasks [ cs_mask_nr ] ;
WARN_ON ( cs_mode > 12 ) ;
if ( pvt - > dram_type = = MEM_DDR4 ) {
if ( cs_mode > 9 )
return - 1 ;
cs_size = ddr4_cs_size ( cs_mode ) ;
} else if ( pvt - > dram_type = = MEM_LRDDR3 ) {
unsigned rank_multiply = dcsm & 0xf ;
if ( rank_multiply = = 3 )
rank_multiply = 4 ;
cs_size = ddr3_lrdimm_cs_size ( cs_mode , rank_multiply ) ;
} else {
/* Minimum cs size is 512mb for F15hM60h*/
if ( cs_mode = = 0x1 )
return - 1 ;
cs_size = ddr3_cs_size ( cs_mode , false ) ;
}
return cs_size ;
}
2013-04-17 23:57:13 +04:00
/*
2013-08-09 20:54:49 +04:00
* F16h and F15h model 30 h have only limited cs_modes .
2013-04-17 23:57:13 +04:00
*/
static int f16_dbam_to_chip_select ( struct amd64_pvt * pvt , u8 dct ,
2014-10-30 14:16:09 +03:00
unsigned cs_mode , int cs_mask_nr )
2013-04-17 23:57:13 +04:00
{
WARN_ON ( cs_mode > 12 ) ;
if ( cs_mode = = 6 | | cs_mode = = 8 | |
cs_mode = = 9 | | cs_mode = = 12 )
return - 1 ;
else
return ddr3_cs_size ( cs_mode , false ) ;
}
2019-08-22 02:59:59 +03:00
static int f17_addr_mask_to_cs_size ( struct amd64_pvt * pvt , u8 umc ,
2016-11-18 01:57:35 +03:00
unsigned int cs_mode , int csrow_nr )
{
2019-08-22 02:59:59 +03:00
u32 addr_mask_orig , addr_mask_deinterleaved ;
u32 msb , weight , num_zero_bits ;
int dimm , size = 0 ;
2016-11-18 01:57:35 +03:00
2019-08-22 02:59:59 +03:00
/* No Chip Selects are enabled. */
if ( ! cs_mode )
return size ;
2016-11-18 01:57:35 +03:00
2019-08-22 02:59:59 +03:00
/* Requested size of an even CS but none are enabled. */
if ( ! ( cs_mode & CS_EVEN ) & & ! ( csrow_nr & 1 ) )
return size ;
2016-11-18 01:57:35 +03:00
2019-08-22 02:59:59 +03:00
/* Requested size of an odd CS but none are enabled. */
if ( ! ( cs_mode & CS_ODD ) & & ( csrow_nr & 1 ) )
return size ;
/*
* There is one mask per DIMM , and two Chip Selects per DIMM .
* CS0 and CS1 - > DIMM0
* CS2 and CS3 - > DIMM1
*/
dimm = csrow_nr > > 1 ;
2019-08-22 03:00:02 +03:00
/* Asymmetric dual-rank DIMM support. */
if ( ( csrow_nr & 1 ) & & ( cs_mode & CS_ODD_SECONDARY ) )
addr_mask_orig = pvt - > csels [ umc ] . csmasks_sec [ dimm ] ;
else
addr_mask_orig = pvt - > csels [ umc ] . csmasks [ dimm ] ;
2019-08-22 02:59:59 +03:00
/*
* The number of zero bits in the mask is equal to the number of bits
* in a full mask minus the number of bits in the current mask .
*
* The MSB is the number of bits in the full mask because BIT [ 0 ] is
* always 0.
*/
msb = fls ( addr_mask_orig ) - 1 ;
weight = hweight_long ( addr_mask_orig ) ;
num_zero_bits = msb - weight ;
/* Take the number of zero bits off from the top of the mask. */
addr_mask_deinterleaved = GENMASK_ULL ( msb - num_zero_bits , 1 ) ;
edac_dbg ( 1 , " CS%d DIMM%d AddrMasks: \n " , csrow_nr , dimm ) ;
edac_dbg ( 1 , " Original AddrMask: 0x%x \n " , addr_mask_orig ) ;
edac_dbg ( 1 , " Deinterleaved AddrMask: 0x%x \n " , addr_mask_deinterleaved ) ;
/* Register [31:1] = Address [39:9]. Size is in kBs here. */
size = ( addr_mask_deinterleaved > > 2 ) + 1 ;
2016-11-18 01:57:35 +03:00
/* Return size in MBs. */
return size > > 10 ;
}
2011-01-17 19:52:57 +03:00
static void read_dram_ctl_register ( struct amd64_pvt * pvt )
2009-04-27 18:20:17 +04:00
{
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0xf )
2011-01-17 19:52:57 +03:00
return ;
2014-09-15 20:37:38 +04:00
if ( ! amd64_read_pci_cfg ( pvt - > F2 , DCT_SEL_LO , & pvt - > dct_sel_lo ) ) {
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " F2x110 (DCTSelLow): 0x%08x, High range addrs at: 0x%x \n " ,
pvt - > dct_sel_lo , dct_sel_baseaddr ( pvt ) ) ;
2009-10-09 21:14:43 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " DCTs operate in %s mode \n " ,
( dct_ganging_enabled ( pvt ) ? " ganged " : " unganged " ) ) ;
2009-10-09 21:14:43 +04:00
if ( ! dct_ganging_enabled ( pvt ) )
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " Address range split per DCT: %s \n " ,
( dct_high_range_enabled ( pvt ) ? " yes " : " no " ) ) ;
2009-10-09 21:14:43 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " data interleave for ECC: %s, DRAM cleared since last warm reset: %s \n " ,
( dct_data_intlv_enabled ( pvt ) ? " enabled " : " disabled " ) ,
( dct_memory_cleared ( pvt ) ? " yes " : " no " ) ) ;
2009-10-09 21:14:43 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " channel interleave: %s, "
" interleave bits selector: 0x%x \n " ,
( dct_interleave_enabled ( pvt ) ? " enabled " : " disabled " ) ,
dct_sel_interleave_addr ( pvt ) ) ;
2009-04-27 18:20:17 +04:00
}
2014-09-15 20:37:38 +04:00
amd64_read_pci_cfg ( pvt - > F2 , DCT_SEL_HI , & pvt - > dct_sel_hi ) ;
2009-04-27 18:20:17 +04:00
}
2013-08-09 20:54:49 +04:00
/*
* Determine channel ( DCT ) based on the interleaving mode ( see F15h M30h BKDG ,
* 2.10 .12 Memory Interleaving Modes ) .
*/
static u8 f15_m30h_determine_channel ( struct amd64_pvt * pvt , u64 sys_addr ,
u8 intlv_en , int num_dcts_intlv ,
u32 dct_sel )
{
u8 channel = 0 ;
u8 select ;
if ( ! ( intlv_en ) )
return ( u8 ) ( dct_sel ) ;
if ( num_dcts_intlv = = 2 ) {
select = ( sys_addr > > 8 ) & 0x3 ;
channel = select ? 0x3 : 0 ;
2014-01-22 01:03:36 +04:00
} else if ( num_dcts_intlv = = 4 ) {
u8 intlv_addr = dct_sel_interleave_addr ( pvt ) ;
switch ( intlv_addr ) {
case 0x4 :
channel = ( sys_addr > > 8 ) & 0x3 ;
break ;
case 0x5 :
channel = ( sys_addr > > 9 ) & 0x3 ;
break ;
}
}
2013-08-09 20:54:49 +04:00
return channel ;
}
2009-04-27 18:22:43 +04:00
/*
2010-12-09 20:57:54 +03:00
* Determine channel ( DCT ) based on the interleaving mode : F10h BKDG , 2.8 .9 Memory
2009-04-27 18:22:43 +04:00
* Interleaving Modes .
*/
2011-01-17 17:59:58 +03:00
static u8 f1x_determine_channel ( struct amd64_pvt * pvt , u64 sys_addr ,
2010-12-09 20:57:54 +03:00
bool hi_range_sel , u8 intlv_en )
2009-04-27 18:20:17 +04:00
{
2011-02-21 21:33:10 +03:00
u8 dct_sel_high = ( pvt - > dct_sel_lo > > 1 ) & 1 ;
2009-04-27 18:20:17 +04:00
if ( dct_ganging_enabled ( pvt ) )
2010-12-09 20:57:54 +03:00
return 0 ;
2009-04-27 18:20:17 +04:00
2010-12-09 20:57:54 +03:00
if ( hi_range_sel )
return dct_sel_high ;
2009-04-27 18:20:17 +04:00
2010-12-09 20:57:54 +03:00
/*
* see F2x110 [ DctSelIntLvAddr ] - channel interleave mode
*/
if ( dct_interleave_enabled ( pvt ) ) {
u8 intlv_addr = dct_sel_interleave_addr ( pvt ) ;
/* return DCT select function: 0=DCT0, 1=DCT1 */
if ( ! intlv_addr )
return sys_addr > > 6 & 1 ;
if ( intlv_addr & 0x2 ) {
u8 shift = intlv_addr & 0x1 ? 9 : 6 ;
2016-08-03 17:59:15 +03:00
u32 temp = hweight_long ( ( u32 ) ( ( sys_addr > > 16 ) & 0x1F ) ) & 1 ;
2010-12-09 20:57:54 +03:00
return ( ( sys_addr > > shift ) & 1 ) ^ temp ;
}
2016-08-03 17:59:15 +03:00
if ( intlv_addr & 0x4 ) {
u8 shift = intlv_addr & 0x1 ? 9 : 8 ;
return ( sys_addr > > shift ) & 1 ;
}
2010-12-09 20:57:54 +03:00
return ( sys_addr > > ( 12 + hweight8 ( intlv_en ) ) ) & 1 ;
}
if ( dct_high_range_enabled ( pvt ) )
return ~ dct_sel_high & 1 ;
2009-04-27 18:20:17 +04:00
return 0 ;
}
2010-12-10 21:49:19 +03:00
/* Convert the sys_addr to the normalized DCT address */
2012-11-30 12:44:20 +04:00
static u64 f1x_get_norm_dct_addr ( struct amd64_pvt * pvt , u8 range ,
2010-12-10 21:49:19 +03:00
u64 sys_addr , bool hi_rng ,
u32 dct_sel_base_addr )
2009-04-27 18:20:17 +04:00
{
u64 chan_off ;
2010-12-10 21:49:19 +03:00
u64 dram_base = get_dram_base ( pvt , range ) ;
u64 hole_off = f10_dhar_offset ( pvt ) ;
2016-01-20 12:54:51 +03:00
u64 dct_sel_base_off = ( u64 ) ( pvt - > dct_sel_hi & 0xFFFFFC00 ) < < 16 ;
2009-04-27 18:20:17 +04:00
2010-12-10 21:49:19 +03:00
if ( hi_rng ) {
/*
* if
* base address of high range is below 4 Gb
* ( bits [ 47 : 27 ] at [ 31 : 11 ] )
* DRAM address space on this DCT is hoisted above 4 Gb & &
* sys_addr > 4 Gb
*
* remove hole offset from sys_addr
* else
* remove high range offset from sys_addr
*/
if ( ( ! ( dct_sel_base_addr > > 16 ) | |
dct_sel_base_addr < dhar_base ( pvt ) ) & &
2011-02-21 21:43:02 +03:00
dhar_valid ( pvt ) & &
2010-12-10 21:49:19 +03:00
( sys_addr > = BIT_64 ( 32 ) ) )
2010-11-11 19:29:13 +03:00
chan_off = hole_off ;
2009-04-27 18:20:17 +04:00
else
chan_off = dct_sel_base_off ;
} else {
2010-12-10 21:49:19 +03:00
/*
* if
* we have a valid hole & &
* sys_addr > 4 Gb
*
* remove hole
* else
* remove dram base to normalize to DCT address
*/
2011-02-21 21:43:02 +03:00
if ( dhar_valid ( pvt ) & & ( sys_addr > = BIT_64 ( 32 ) ) )
2010-11-11 19:29:13 +03:00
chan_off = hole_off ;
2009-04-27 18:20:17 +04:00
else
2010-12-10 21:49:19 +03:00
chan_off = dram_base ;
2009-04-27 18:20:17 +04:00
}
2013-10-19 01:29:07 +04:00
return ( sys_addr & GENMASK_ULL ( 47 , 6 ) ) - ( chan_off & GENMASK_ULL ( 47 , 23 ) ) ;
2009-04-27 18:20:17 +04:00
}
/*
* checks if the csrow passed in is marked as SPARED , if so returns the new
* spare row
*/
2010-11-29 21:49:02 +03:00
static int f10_process_possible_spare ( struct amd64_pvt * pvt , u8 dct , int csrow )
2009-04-27 18:20:17 +04:00
{
2011-01-13 20:02:22 +03:00
int tmp_cs ;
if ( online_spare_swap_done ( pvt , dct ) & &
csrow = = online_spare_bad_dramcs ( pvt , dct ) ) {
for_each_chip_select ( tmp_cs , dct , pvt ) {
if ( chip_select_base ( tmp_cs , dct , pvt ) & 0x2 ) {
csrow = tmp_cs ;
break ;
}
}
2009-04-27 18:20:17 +04:00
}
return csrow ;
}
/*
* Iterate over the DRAM DCT " base " and " mask " registers looking for a
* SystemAddr match on the specified ' ChannelSelect ' and ' NodeID '
*
* Return :
* - EINVAL : NOT FOUND
* 0. . csrow = Chip - Select Row
*/
2012-11-30 12:44:20 +04:00
static int f1x_lookup_addr_in_dct ( u64 in_addr , u8 nid , u8 dct )
2009-04-27 18:20:17 +04:00
{
struct mem_ctl_info * mci ;
struct amd64_pvt * pvt ;
2010-11-29 21:49:02 +03:00
u64 cs_base , cs_mask ;
2009-04-27 18:20:17 +04:00
int cs_found = - EINVAL ;
int csrow ;
2015-02-17 12:58:34 +03:00
mci = edac_mc_find ( nid ) ;
2009-04-27 18:20:17 +04:00
if ( ! mci )
return cs_found ;
pvt = mci - > pvt_info ;
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " input addr: 0x%llx, DCT: %d \n " , in_addr , dct ) ;
2009-04-27 18:20:17 +04:00
2010-11-29 21:49:02 +03:00
for_each_chip_select ( csrow , dct , pvt ) {
if ( ! csrow_enabled ( csrow , dct , pvt ) )
2009-04-27 18:20:17 +04:00
continue ;
2010-11-29 21:49:02 +03:00
get_cs_base_and_mask ( pvt , csrow , dct , & cs_base , & cs_mask ) ;
2009-04-27 18:20:17 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " CSROW=%d CSBase=0x%llx CSMask=0x%llx \n " ,
csrow , cs_base , cs_mask ) ;
2009-04-27 18:20:17 +04:00
2010-11-29 21:49:02 +03:00
cs_mask = ~ cs_mask ;
2009-04-27 18:20:17 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " (InputAddr & ~CSMask)=0x%llx (CSBase & ~CSMask)=0x%llx \n " ,
( in_addr & cs_mask ) , ( cs_base & cs_mask ) ) ;
2009-04-27 18:20:17 +04:00
2010-11-29 21:49:02 +03:00
if ( ( in_addr & cs_mask ) = = ( cs_base & cs_mask ) ) {
2013-08-09 20:54:49 +04:00
if ( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 ) {
cs_found = csrow ;
break ;
}
2010-11-29 21:49:02 +03:00
cs_found = f10_process_possible_spare ( pvt , dct , csrow ) ;
2009-04-27 18:20:17 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " MATCH csrow=%d \n " , cs_found ) ;
2009-04-27 18:20:17 +04:00
break ;
}
}
return cs_found ;
}
2011-01-12 00:08:07 +03:00
/*
* See F2x10C . Non - interleaved graphics framebuffer memory under the 16 G is
* swapped with a region located at the bottom of memory so that the GPU can use
* the interleaved region and thus two channels .
*/
2011-01-17 17:59:58 +03:00
static u64 f1x_swap_interleaved_region ( struct amd64_pvt * pvt , u64 sys_addr )
2011-01-12 00:08:07 +03:00
{
u32 swap_reg , swap_base , swap_limit , rgn_size , tmp_addr ;
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0x10 ) {
2011-01-12 00:08:07 +03:00
/* only revC3 and revE have that feature */
2013-08-10 15:54:48 +04:00
if ( pvt - > model < 4 | | ( pvt - > model < 0xa & & pvt - > stepping < 3 ) )
2011-01-12 00:08:07 +03:00
return sys_addr ;
}
2014-09-15 20:37:38 +04:00
amd64_read_pci_cfg ( pvt - > F2 , SWAP_INTLV_REG , & swap_reg ) ;
2011-01-12 00:08:07 +03:00
if ( ! ( swap_reg & 0x1 ) )
return sys_addr ;
swap_base = ( swap_reg > > 3 ) & 0x7f ;
swap_limit = ( swap_reg > > 11 ) & 0x7f ;
rgn_size = ( swap_reg > > 20 ) & 0x7f ;
tmp_addr = sys_addr > > 27 ;
if ( ! ( sys_addr > > 34 ) & &
( ( ( tmp_addr > = swap_base ) & &
( tmp_addr < = swap_limit ) ) | |
( tmp_addr < rgn_size ) ) )
return sys_addr ^ ( u64 ) swap_base < < 27 ;
return sys_addr ;
}
2009-04-27 18:22:43 +04:00
/* For a given @dram_range, check if @sys_addr falls within it. */
2011-02-21 21:49:01 +03:00
static int f1x_match_to_this_node ( struct amd64_pvt * pvt , unsigned range ,
2012-08-30 20:01:36 +04:00
u64 sys_addr , int * chan_sel )
2009-04-27 18:22:43 +04:00
{
2010-12-09 20:57:54 +03:00
int cs_found = - EINVAL ;
2010-12-10 21:49:19 +03:00
u64 chan_addr ;
2011-01-13 18:01:13 +03:00
u32 dct_sel_base ;
2010-11-29 21:49:02 +03:00
u8 channel ;
2010-12-09 20:57:54 +03:00
bool high_range = false ;
2009-04-27 18:22:43 +04:00
2010-10-21 20:52:53 +04:00
u8 node_id = dram_dst_node ( pvt , range ) ;
2010-12-09 20:57:54 +03:00
u8 intlv_en = dram_intlv_en ( pvt , range ) ;
2010-10-21 20:52:53 +04:00
u32 intlv_sel = dram_intlv_sel ( pvt , range ) ;
2009-04-27 18:22:43 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " (range %d) SystemAddr= 0x%llx Limit=0x%llx \n " ,
range , sys_addr , get_dram_limit ( pvt , range ) ) ;
2009-04-27 18:22:43 +04:00
2011-01-17 15:03:26 +03:00
if ( dhar_valid ( pvt ) & &
dhar_base ( pvt ) < = sys_addr & &
sys_addr < BIT_64 ( 32 ) ) {
amd64_warn ( " Huh? Address is in the MMIO hole: 0x%016llx \n " ,
sys_addr ) ;
return - EINVAL ;
}
2011-04-08 17:05:21 +04:00
if ( intlv_en & & ( intlv_sel ! = ( ( sys_addr > > 12 ) & intlv_en ) ) )
2009-04-27 18:22:43 +04:00
return - EINVAL ;
2011-01-17 17:59:58 +03:00
sys_addr = f1x_swap_interleaved_region ( pvt , sys_addr ) ;
2011-01-12 00:08:07 +03:00
2009-04-27 18:22:43 +04:00
dct_sel_base = dct_sel_baseaddr ( pvt ) ;
/*
* check whether addresses > = DctSelBaseAddr [ 47 : 27 ] are to be used to
* select between DCT0 and DCT1 .
*/
if ( dct_high_range_enabled ( pvt ) & &
! dct_ganging_enabled ( pvt ) & &
( ( sys_addr > > 27 ) > = ( dct_sel_base > > 11 ) ) )
2010-12-09 20:57:54 +03:00
high_range = true ;
2009-04-27 18:22:43 +04:00
2011-01-17 17:59:58 +03:00
channel = f1x_determine_channel ( pvt , sys_addr , high_range , intlv_en ) ;
2009-04-27 18:22:43 +04:00
2011-01-17 17:59:58 +03:00
chan_addr = f1x_get_norm_dct_addr ( pvt , range , sys_addr ,
2010-12-10 21:49:19 +03:00
high_range , dct_sel_base ) ;
2009-04-27 18:22:43 +04:00
2011-01-13 16:57:34 +03:00
/* Remove node interleaving, see F1x120 */
if ( intlv_en )
chan_addr = ( ( chan_addr > > ( 12 + hweight8 ( intlv_en ) ) ) < < 12 ) |
( chan_addr & 0xfff ) ;
2009-04-27 18:22:43 +04:00
2011-01-13 18:01:13 +03:00
/* remove channel interleave */
2009-04-27 18:22:43 +04:00
if ( dct_interleave_enabled ( pvt ) & &
! dct_high_range_enabled ( pvt ) & &
! dct_ganging_enabled ( pvt ) ) {
2011-01-13 18:01:13 +03:00
if ( dct_sel_interleave_addr ( pvt ) ! = 1 ) {
if ( dct_sel_interleave_addr ( pvt ) = = 0x3 )
/* hash 9 */
chan_addr = ( ( chan_addr > > 10 ) < < 9 ) |
( chan_addr & 0x1ff ) ;
else
/* A[6] or hash 6 */
chan_addr = ( ( chan_addr > > 7 ) < < 6 ) |
( chan_addr & 0x3f ) ;
} else
/* A[12] */
chan_addr = ( ( chan_addr > > 13 ) < < 12 ) |
( chan_addr & 0xfff ) ;
2009-04-27 18:22:43 +04:00
}
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " Normalized DCT addr: 0x%llx \n " , chan_addr ) ;
2009-04-27 18:22:43 +04:00
2011-01-17 17:59:58 +03:00
cs_found = f1x_lookup_addr_in_dct ( chan_addr , node_id , channel ) ;
2009-04-27 18:22:43 +04:00
2012-08-30 20:01:36 +04:00
if ( cs_found > = 0 )
2009-04-27 18:22:43 +04:00
* chan_sel = channel ;
2012-08-30 20:01:36 +04:00
2009-04-27 18:22:43 +04:00
return cs_found ;
}
2013-08-09 20:54:49 +04:00
static int f15_m30h_match_to_this_node ( struct amd64_pvt * pvt , unsigned range ,
u64 sys_addr , int * chan_sel )
{
int cs_found = - EINVAL ;
int num_dcts_intlv = 0 ;
u64 chan_addr , chan_offset ;
u64 dct_base , dct_limit ;
u32 dct_cont_base_reg , dct_cont_limit_reg , tmp ;
u8 channel , alias_channel , leg_mmio_hole , dct_sel , dct_offset_en ;
u64 dhar_offset = f10_dhar_offset ( pvt ) ;
u8 intlv_addr = dct_sel_interleave_addr ( pvt ) ;
u8 node_id = dram_dst_node ( pvt , range ) ;
u8 intlv_en = dram_intlv_en ( pvt , range ) ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_CONT_BASE , & dct_cont_base_reg ) ;
amd64_read_pci_cfg ( pvt - > F1 , DRAM_CONT_LIMIT , & dct_cont_limit_reg ) ;
dct_offset_en = ( u8 ) ( ( dct_cont_base_reg > > 3 ) & BIT ( 0 ) ) ;
dct_sel = ( u8 ) ( ( dct_cont_base_reg > > 4 ) & 0x7 ) ;
edac_dbg ( 1 , " (range %d) SystemAddr= 0x%llx Limit=0x%llx \n " ,
range , sys_addr , get_dram_limit ( pvt , range ) ) ;
if ( ! ( get_dram_base ( pvt , range ) < = sys_addr ) & &
! ( get_dram_limit ( pvt , range ) > = sys_addr ) )
return - EINVAL ;
if ( dhar_valid ( pvt ) & &
dhar_base ( pvt ) < = sys_addr & &
sys_addr < BIT_64 ( 32 ) ) {
amd64_warn ( " Huh? Address is in the MMIO hole: 0x%016llx \n " ,
sys_addr ) ;
return - EINVAL ;
}
/* Verify sys_addr is within DCT Range. */
2013-08-24 19:47:48 +04:00
dct_base = ( u64 ) dct_sel_baseaddr ( pvt ) ;
dct_limit = ( dct_cont_limit_reg > > 11 ) & 0x1FFF ;
2013-08-09 20:54:49 +04:00
if ( ! ( dct_cont_base_reg & BIT ( 0 ) ) & &
2013-08-24 19:47:48 +04:00
! ( dct_base < = ( sys_addr > > 27 ) & &
dct_limit > = ( sys_addr > > 27 ) ) )
2013-08-09 20:54:49 +04:00
return - EINVAL ;
/* Verify number of dct's that participate in channel interleaving. */
num_dcts_intlv = ( int ) hweight8 ( intlv_en ) ;
if ( ! ( num_dcts_intlv % 2 = = 0 ) | | ( num_dcts_intlv > 4 ) )
return - EINVAL ;
2016-08-03 17:59:15 +03:00
if ( pvt - > model > = 0x60 )
channel = f1x_determine_channel ( pvt , sys_addr , false , intlv_en ) ;
else
channel = f15_m30h_determine_channel ( pvt , sys_addr , intlv_en ,
num_dcts_intlv , dct_sel ) ;
2013-08-09 20:54:49 +04:00
/* Verify we stay within the MAX number of channels allowed */
2013-12-04 21:40:11 +04:00
if ( channel > 3 )
2013-08-09 20:54:49 +04:00
return - EINVAL ;
leg_mmio_hole = ( u8 ) ( dct_cont_base_reg > > 1 & BIT ( 0 ) ) ;
/* Get normalized DCT addr */
if ( leg_mmio_hole & & ( sys_addr > = BIT_64 ( 32 ) ) )
chan_offset = dhar_offset ;
else
2013-08-24 19:47:48 +04:00
chan_offset = dct_base < < 27 ;
2013-08-09 20:54:49 +04:00
chan_addr = sys_addr - chan_offset ;
/* remove channel interleave */
if ( num_dcts_intlv = = 2 ) {
if ( intlv_addr = = 0x4 )
chan_addr = ( ( chan_addr > > 9 ) < < 8 ) |
( chan_addr & 0xff ) ;
else if ( intlv_addr = = 0x5 )
chan_addr = ( ( chan_addr > > 10 ) < < 9 ) |
( chan_addr & 0x1ff ) ;
else
return - EINVAL ;
} else if ( num_dcts_intlv = = 4 ) {
if ( intlv_addr = = 0x4 )
chan_addr = ( ( chan_addr > > 10 ) < < 8 ) |
( chan_addr & 0xff ) ;
else if ( intlv_addr = = 0x5 )
chan_addr = ( ( chan_addr > > 11 ) < < 9 ) |
( chan_addr & 0x1ff ) ;
else
return - EINVAL ;
}
if ( dct_offset_en ) {
amd64_read_pci_cfg ( pvt - > F1 ,
DRAM_CONT_HIGH_OFF + ( int ) channel * 4 ,
& tmp ) ;
2013-08-24 19:47:48 +04:00
chan_addr + = ( u64 ) ( ( tmp > > 11 ) & 0xfff ) < < 27 ;
2013-08-09 20:54:49 +04:00
}
f15h_select_dct ( pvt , channel ) ;
edac_dbg ( 1 , " Normalized DCT addr: 0x%llx \n " , chan_addr ) ;
/*
* Find Chip select :
* if channel = 3 , then alias it to 1. This is because , in F15 M30h ,
* there is support for 4 DCT ' s , but only 2 are currently functional .
* They are DCT0 and DCT3 . But we have read all registers of DCT3 into
* pvt - > csels [ 1 ] . So we need to use ' 1 ' here to get correct info .
* Refer F15 M30h BKDG Section 2.10 and 2.10 .3 for clarifications .
*/
alias_channel = ( channel = = 3 ) ? 1 : channel ;
cs_found = f1x_lookup_addr_in_dct ( chan_addr , node_id , alias_channel ) ;
if ( cs_found > = 0 )
* chan_sel = alias_channel ;
return cs_found ;
}
static int f1x_translate_sysaddr_to_cs ( struct amd64_pvt * pvt ,
u64 sys_addr ,
int * chan_sel )
2009-04-27 18:22:43 +04:00
{
2011-02-21 21:49:01 +03:00
int cs_found = - EINVAL ;
unsigned range ;
2009-04-27 18:22:43 +04:00
2010-10-21 20:52:53 +04:00
for ( range = 0 ; range < DRAM_RANGES ; range + + ) {
if ( ! dram_rw ( pvt , range ) )
2009-04-27 18:22:43 +04:00
continue ;
2013-08-09 20:54:49 +04:00
if ( pvt - > fam = = 0x15 & & pvt - > model > = 0x30 )
cs_found = f15_m30h_match_to_this_node ( pvt , range ,
sys_addr ,
chan_sel ) ;
2009-04-27 18:22:43 +04:00
2013-08-09 20:54:49 +04:00
else if ( ( get_dram_base ( pvt , range ) < = sys_addr ) & &
( get_dram_limit ( pvt , range ) > = sys_addr ) ) {
2011-01-17 17:59:58 +03:00
cs_found = f1x_match_to_this_node ( pvt , range ,
2012-08-30 20:01:36 +04:00
sys_addr , chan_sel ) ;
2009-04-27 18:22:43 +04:00
if ( cs_found > = 0 )
break ;
}
}
return cs_found ;
}
/*
2009-11-13 17:10:43 +03:00
* For reference see " 2.8.5 Routing DRAM Requests " in F10 BKDG . This code maps
* a @ sys_addr to NodeID , DCT ( channel ) and chip select ( CSROW ) .
2009-04-27 18:22:43 +04:00
*
2009-11-13 17:10:43 +03:00
* The @ sys_addr is usually an error address received from the hardware
* ( MCX_ADDR ) .
2009-04-27 18:22:43 +04:00
*/
2011-01-17 17:59:58 +03:00
static void f1x_map_sysaddr_to_csrow ( struct mem_ctl_info * mci , u64 sys_addr ,
2012-08-30 20:01:36 +04:00
struct err_info * err )
2009-04-27 18:22:43 +04:00
{
struct amd64_pvt * pvt = mci - > pvt_info ;
2012-08-30 20:01:36 +04:00
error_address_to_page_and_offset ( sys_addr , err ) ;
2012-04-16 22:03:50 +04:00
2012-08-30 20:01:36 +04:00
err - > csrow = f1x_translate_sysaddr_to_cs ( pvt , sys_addr , & err - > channel ) ;
if ( err - > csrow < 0 ) {
err - > err_code = ERR_CSROW ;
2009-11-13 17:10:43 +03:00
return ;
}
/*
* We need the syndromes for channel detection only when we ' re
* ganged . Otherwise @ chan should already contain the channel at
* this point .
*/
2010-12-23 16:07:18 +03:00
if ( dct_ganging_enabled ( pvt ) )
2012-08-30 20:01:36 +04:00
err - > channel = get_channel_from_ecc_syndrome ( mci , err - > syndrome ) ;
2009-04-27 18:22:43 +04:00
}
/*
2009-10-16 15:48:28 +04:00
* debug routine to display the memory sizes of all logical DIMMs and its
2010-12-22 16:28:24 +03:00
* CSROWs
2009-04-27 18:22:43 +04:00
*/
2013-12-15 20:54:27 +04:00
static void debug_display_dimm_sizes ( struct amd64_pvt * pvt , u8 ctrl )
2009-04-27 18:22:43 +04:00
{
2012-09-12 20:06:00 +04:00
int dimm , size0 , size1 ;
2010-12-21 17:53:27 +03:00
u32 * dcsb = ctrl ? pvt - > csels [ 1 ] . csbases : pvt - > csels [ 0 ] . csbases ;
u32 dbam = ctrl ? pvt - > dbam1 : pvt - > dbam0 ;
2009-04-27 18:22:43 +04:00
2013-08-10 15:54:48 +04:00
if ( pvt - > fam = = 0xf ) {
2009-10-16 15:48:28 +04:00
/* K8 families < revF not supported yet */
2009-10-21 15:44:36 +04:00
if ( pvt - > ext_model < K8_REV_F )
2009-10-16 15:48:28 +04:00
return ;
else
WARN_ON ( ctrl ! = 0 ) ;
}
2014-09-15 20:37:38 +04:00
if ( pvt - > fam = = 0x10 ) {
dbam = ( ctrl & & ! dct_ganging_enabled ( pvt ) ) ? pvt - > dbam1
: pvt - > dbam0 ;
dcsb = ( ctrl & & ! dct_ganging_enabled ( pvt ) ) ?
pvt - > csels [ 1 ] . csbases :
pvt - > csels [ 0 ] . csbases ;
} else if ( ctrl ) {
dbam = pvt - > dbam0 ;
dcsb = pvt - > csels [ 1 ] . csbases ;
}
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " F2x%d80 (DRAM Bank Address Mapping): 0x%08x \n " ,
ctrl , dbam ) ;
2009-04-27 18:22:43 +04:00
2009-10-16 15:48:28 +04:00
edac_printk ( KERN_DEBUG , EDAC_MC , " DCT%d chip selects: \n " , ctrl ) ;
2009-04-27 18:22:43 +04:00
/* Dump memory sizes for DIMM and its CSROWs */
for ( dimm = 0 ; dimm < 4 ; dimm + + ) {
size0 = 0 ;
2010-11-29 21:49:02 +03:00
if ( dcsb [ dimm * 2 ] & DCSB_CS_ENABLE )
2016-11-28 17:50:21 +03:00
/*
* For F15m60h , we need multiplier for LRDIMM cs_size
* calculation . We pass dimm value to the dbam_to_cs
2014-10-30 14:16:09 +03:00
* mapper so we can find the multiplier from the
* corresponding DCSM .
*/
2011-01-18 21:16:08 +03:00
size0 = pvt - > ops - > dbam_to_cs ( pvt , ctrl ,
2014-10-30 14:16:09 +03:00
DBAM_DIMM ( dimm , dbam ) ,
dimm ) ;
2009-04-27 18:22:43 +04:00
size1 = 0 ;
2010-11-29 21:49:02 +03:00
if ( dcsb [ dimm * 2 + 1 ] & DCSB_CS_ENABLE )
2011-01-18 21:16:08 +03:00
size1 = pvt - > ops - > dbam_to_cs ( pvt , ctrl ,
2014-10-30 14:16:09 +03:00
DBAM_DIMM ( dimm , dbam ) ,
dimm ) ;
2009-04-27 18:22:43 +04:00
2010-10-07 20:29:15 +04:00
amd64_info ( EDAC_MC " : %d: %5dMB %d: %5dMB \n " ,
2012-09-12 20:06:00 +04:00
dimm * 2 , size0 ,
dimm * 2 + 1 , size1 ) ;
2009-04-27 18:22:43 +04:00
}
}
2013-12-15 20:54:27 +04:00
static struct amd64_family_type family_types [ ] = {
2009-04-27 18:25:05 +04:00
[ K8_CPUS ] = {
2010-10-01 21:20:05 +04:00
. ctl_name = " K8 " ,
2010-10-01 22:11:07 +04:00
. f1_id = PCI_DEVICE_ID_AMD_K8_NB_ADDRMAP ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_K8_NB_MEMCTL ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2009-04-27 18:25:05 +04:00
. ops = {
2009-10-21 15:44:36 +04:00
. early_channel_count = k8_early_channel_count ,
. map_sysaddr_to_csrow = k8_map_sysaddr_to_csrow ,
. dbam_to_cs = k8_dbam_to_chip_select ,
2009-04-27 18:25:05 +04:00
}
} ,
[ F10_CPUS ] = {
2010-10-01 21:20:05 +04:00
. ctl_name = " F10h " ,
2010-10-01 22:11:07 +04:00
. f1_id = PCI_DEVICE_ID_AMD_10H_NB_MAP ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_10H_NB_DRAM ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2009-04-27 18:25:05 +04:00
. ops = {
2011-01-07 19:58:04 +03:00
. early_channel_count = f1x_early_channel_count ,
2011-01-17 17:59:58 +03:00
. map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow ,
2009-10-21 15:44:36 +04:00
. dbam_to_cs = f10_dbam_to_chip_select ,
2010-10-08 20:32:29 +04:00
}
} ,
[ F15_CPUS ] = {
. ctl_name = " F15h " ,
2011-01-19 20:15:10 +03:00
. f1_id = PCI_DEVICE_ID_AMD_15H_NB_F1 ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_15H_NB_F2 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2010-10-08 20:32:29 +04:00
. ops = {
2011-01-07 19:58:04 +03:00
. early_channel_count = f1x_early_channel_count ,
2011-01-17 17:59:58 +03:00
. map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow ,
2011-01-18 21:16:08 +03:00
. dbam_to_cs = f15_dbam_to_chip_select ,
2009-04-27 18:25:05 +04:00
}
} ,
2013-08-09 20:54:49 +04:00
[ F15_M30H_CPUS ] = {
. ctl_name = " F15h_M30h " ,
. f1_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1 ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F2 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2013-08-09 20:54:49 +04:00
. ops = {
. early_channel_count = f1x_early_channel_count ,
. map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow ,
. dbam_to_cs = f16_dbam_to_chip_select ,
}
} ,
2014-10-30 14:16:09 +03:00
[ F15_M60H_CPUS ] = {
. ctl_name = " F15h_M60h " ,
. f1_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1 ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F2 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2014-10-30 14:16:09 +03:00
. ops = {
. early_channel_count = f1x_early_channel_count ,
. map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow ,
. dbam_to_cs = f15_m60h_dbam_to_chip_select ,
}
} ,
2013-04-17 23:57:13 +04:00
[ F16_CPUS ] = {
. ctl_name = " F16h " ,
. f1_id = PCI_DEVICE_ID_AMD_16H_NB_F1 ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_16H_NB_F2 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2013-04-17 23:57:13 +04:00
. ops = {
. early_channel_count = f1x_early_channel_count ,
. map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow ,
. dbam_to_cs = f16_dbam_to_chip_select ,
}
} ,
2014-02-20 20:28:46 +04:00
[ F16_M30H_CPUS ] = {
. ctl_name = " F16h_M30h " ,
. f1_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F1 ,
2016-05-06 20:44:27 +03:00
. f2_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F2 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2014-02-20 20:28:46 +04:00
. ops = {
. early_channel_count = f1x_early_channel_count ,
. map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow ,
. dbam_to_cs = f16_dbam_to_chip_select ,
}
} ,
2016-11-18 01:57:35 +03:00
[ F17_CPUS ] = {
. ctl_name = " F17h " ,
. f0_id = PCI_DEVICE_ID_AMD_17H_DF_F0 ,
. f6_id = PCI_DEVICE_ID_AMD_17H_DF_F6 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2016-11-18 01:57:35 +03:00
. ops = {
. early_channel_count = f17_early_channel_count ,
2019-08-22 02:59:59 +03:00
. dbam_to_cs = f17_addr_mask_to_cs_size ,
2016-11-18 01:57:35 +03:00
}
} ,
2018-08-16 22:28:40 +03:00
[ F17_M10H_CPUS ] = {
. ctl_name = " F17h_M10h " ,
. f0_id = PCI_DEVICE_ID_AMD_17H_M10H_DF_F0 ,
. f6_id = PCI_DEVICE_ID_AMD_17H_M10H_DF_F6 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2018-08-16 22:28:40 +03:00
. ops = {
. early_channel_count = f17_early_channel_count ,
2019-08-22 02:59:59 +03:00
. dbam_to_cs = f17_addr_mask_to_cs_size ,
2018-08-16 22:28:40 +03:00
}
} ,
2019-02-28 18:36:09 +03:00
[ F17_M30H_CPUS ] = {
. ctl_name = " F17h_M30h " ,
. f0_id = PCI_DEVICE_ID_AMD_17H_M30H_DF_F0 ,
. f6_id = PCI_DEVICE_ID_AMD_17H_M30H_DF_F6 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 8 ,
2019-02-28 18:36:09 +03:00
. ops = {
. early_channel_count = f17_early_channel_count ,
2019-08-22 02:59:59 +03:00
. dbam_to_cs = f17_addr_mask_to_cs_size ,
2019-02-28 18:36:09 +03:00
}
} ,
2020-05-10 23:48:42 +03:00
[ F17_M60H_CPUS ] = {
. ctl_name = " F17h_M60h " ,
. f0_id = PCI_DEVICE_ID_AMD_17H_M60H_DF_F0 ,
. f6_id = PCI_DEVICE_ID_AMD_17H_M60H_DF_F6 ,
. max_mcs = 2 ,
. ops = {
. early_channel_count = f17_early_channel_count ,
. dbam_to_cs = f17_addr_mask_to_cs_size ,
}
} ,
2019-09-07 02:21:38 +03:00
[ F17_M70H_CPUS ] = {
. ctl_name = " F17h_M70h " ,
. f0_id = PCI_DEVICE_ID_AMD_17H_M70H_DF_F0 ,
. f6_id = PCI_DEVICE_ID_AMD_17H_M70H_DF_F6 ,
2019-10-22 23:35:11 +03:00
. max_mcs = 2 ,
2019-09-07 02:21:38 +03:00
. ops = {
. early_channel_count = f17_early_channel_count ,
. dbam_to_cs = f17_addr_mask_to_cs_size ,
}
} ,
2020-01-10 04:56:50 +03:00
[ F19_CPUS ] = {
. ctl_name = " F19h " ,
. f0_id = PCI_DEVICE_ID_AMD_19H_DF_F0 ,
. f6_id = PCI_DEVICE_ID_AMD_19H_DF_F6 ,
. max_mcs = 8 ,
. ops = {
. early_channel_count = f17_early_channel_count ,
. dbam_to_cs = f17_addr_mask_to_cs_size ,
}
} ,
2009-04-27 18:25:05 +04:00
} ;
2009-04-27 18:37:05 +04:00
/*
2009-11-12 21:05:07 +03:00
* These are tables of eigenvectors ( one per line ) which can be used for the
* construction of the syndrome tables . The modified syndrome search algorithm
* uses those to find the symbol in error and thus the DIMM .
2009-04-27 18:37:05 +04:00
*
2009-11-12 21:05:07 +03:00
* Algorithm courtesy of Ross LaFetra from AMD .
2009-04-27 18:37:05 +04:00
*/
2012-11-30 12:44:20 +04:00
static const u16 x4_vectors [ ] = {
2009-11-12 21:05:07 +03:00
0x2f57 , 0x1afe , 0x66cc , 0xdd88 ,
0x11eb , 0x3396 , 0x7f4c , 0xeac8 ,
0x0001 , 0x0002 , 0x0004 , 0x0008 ,
0x1013 , 0x3032 , 0x4044 , 0x8088 ,
0x106b , 0x30d6 , 0x70fc , 0xe0a8 ,
0x4857 , 0xc4fe , 0x13cc , 0x3288 ,
0x1ac5 , 0x2f4a , 0x5394 , 0xa1e8 ,
0x1f39 , 0x251e , 0xbd6c , 0x6bd8 ,
0x15c1 , 0x2a42 , 0x89ac , 0x4758 ,
0x2b03 , 0x1602 , 0x4f0c , 0xca08 ,
0x1f07 , 0x3a0e , 0x6b04 , 0xbd08 ,
0x8ba7 , 0x465e , 0x244c , 0x1cc8 ,
0x2b87 , 0x164e , 0x642c , 0xdc18 ,
0x40b9 , 0x80de , 0x1094 , 0x20e8 ,
0x27db , 0x1eb6 , 0x9dac , 0x7b58 ,
0x11c1 , 0x2242 , 0x84ac , 0x4c58 ,
0x1be5 , 0x2d7a , 0x5e34 , 0xa718 ,
0x4b39 , 0x8d1e , 0x14b4 , 0x28d8 ,
0x4c97 , 0xc87e , 0x11fc , 0x33a8 ,
0x8e97 , 0x497e , 0x2ffc , 0x1aa8 ,
0x16b3 , 0x3d62 , 0x4f34 , 0x8518 ,
0x1e2f , 0x391a , 0x5cac , 0xf858 ,
0x1d9f , 0x3b7a , 0x572c , 0xfe18 ,
0x15f5 , 0x2a5a , 0x5264 , 0xa3b8 ,
0x1dbb , 0x3b66 , 0x715c , 0xe3f8 ,
0x4397 , 0xc27e , 0x17fc , 0x3ea8 ,
0x1617 , 0x3d3e , 0x6464 , 0xb8b8 ,
0x23ff , 0x12aa , 0xab6c , 0x56d8 ,
0x2dfb , 0x1ba6 , 0x913c , 0x7328 ,
0x185d , 0x2ca6 , 0x7914 , 0x9e28 ,
0x171b , 0x3e36 , 0x7d7c , 0xebe8 ,
0x4199 , 0x82ee , 0x19f4 , 0x2e58 ,
0x4807 , 0xc40e , 0x130c , 0x3208 ,
0x1905 , 0x2e0a , 0x5804 , 0xac08 ,
0x213f , 0x132a , 0xadfc , 0x5ba8 ,
0x19a9 , 0x2efe , 0xb5cc , 0x6f88 ,
2009-04-27 18:37:05 +04:00
} ;
2012-11-30 12:44:20 +04:00
static const u16 x8_vectors [ ] = {
2009-11-12 21:05:07 +03:00
0x0145 , 0x028a , 0x2374 , 0x43c8 , 0xa1f0 , 0x0520 , 0x0a40 , 0x1480 ,
0x0211 , 0x0422 , 0x0844 , 0x1088 , 0x01b0 , 0x44e0 , 0x23c0 , 0xed80 ,
0x1011 , 0x0116 , 0x022c , 0x0458 , 0x08b0 , 0x8c60 , 0x2740 , 0x4e80 ,
0x0411 , 0x0822 , 0x1044 , 0x0158 , 0x02b0 , 0x2360 , 0x46c0 , 0xab80 ,
0x0811 , 0x1022 , 0x012c , 0x0258 , 0x04b0 , 0x4660 , 0x8cc0 , 0x2780 ,
0x2071 , 0x40e2 , 0xa0c4 , 0x0108 , 0x0210 , 0x0420 , 0x0840 , 0x1080 ,
0x4071 , 0x80e2 , 0x0104 , 0x0208 , 0x0410 , 0x0820 , 0x1040 , 0x2080 ,
0x8071 , 0x0102 , 0x0204 , 0x0408 , 0x0810 , 0x1020 , 0x2040 , 0x4080 ,
0x019d , 0x03d6 , 0x136c , 0x2198 , 0x50b0 , 0xb2e0 , 0x0740 , 0x0e80 ,
0x0189 , 0x03ea , 0x072c , 0x0e58 , 0x1cb0 , 0x56e0 , 0x37c0 , 0xf580 ,
0x01fd , 0x0376 , 0x06ec , 0x0bb8 , 0x1110 , 0x2220 , 0x4440 , 0x8880 ,
0x0163 , 0x02c6 , 0x1104 , 0x0758 , 0x0eb0 , 0x2be0 , 0x6140 , 0xc280 ,
0x02fd , 0x01c6 , 0x0b5c , 0x1108 , 0x07b0 , 0x25a0 , 0x8840 , 0x6180 ,
0x0801 , 0x012e , 0x025c , 0x04b8 , 0x1370 , 0x26e0 , 0x57c0 , 0xb580 ,
0x0401 , 0x0802 , 0x015c , 0x02b8 , 0x22b0 , 0x13e0 , 0x7140 , 0xe280 ,
0x0201 , 0x0402 , 0x0804 , 0x01b8 , 0x11b0 , 0x31a0 , 0x8040 , 0x7180 ,
0x0101 , 0x0202 , 0x0404 , 0x0808 , 0x1010 , 0x2020 , 0x4040 , 0x8080 ,
0x0001 , 0x0002 , 0x0004 , 0x0008 , 0x0010 , 0x0020 , 0x0040 , 0x0080 ,
0x0100 , 0x0200 , 0x0400 , 0x0800 , 0x1000 , 0x2000 , 0x4000 , 0x8000 ,
} ;
2012-11-30 12:44:20 +04:00
static int decode_syndrome ( u16 syndrome , const u16 * vectors , unsigned num_vecs ,
2011-02-23 19:41:50 +03:00
unsigned v_dim )
2009-04-27 18:37:05 +04:00
{
2009-11-12 21:05:07 +03:00
unsigned int i , err_sym ;
for ( err_sym = 0 ; err_sym < num_vecs / v_dim ; err_sym + + ) {
u16 s = syndrome ;
2011-02-23 19:41:50 +03:00
unsigned v_idx = err_sym * v_dim ;
unsigned v_end = ( err_sym + 1 ) * v_dim ;
2009-11-12 21:05:07 +03:00
/* walk over all 16 bits of the syndrome */
for ( i = 1 ; i < ( 1U < < 16 ) ; i < < = 1 ) {
/* if bit is set in that eigenvector... */
if ( v_idx < v_end & & vectors [ v_idx ] & i ) {
u16 ev_comp = vectors [ v_idx + + ] ;
/* ... and bit set in the modified syndrome, */
if ( s & i ) {
/* remove it. */
s ^ = ev_comp ;
2009-04-27 18:25:05 +04:00
2009-11-12 21:05:07 +03:00
if ( ! s )
return err_sym ;
}
2009-04-27 18:37:05 +04:00
2009-11-12 21:05:07 +03:00
} else if ( s & i )
/* can't get to zero, move to next symbol */
break ;
}
2009-04-27 18:37:05 +04:00
}
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " syndrome(%x) not found \n " , syndrome ) ;
2009-04-27 18:37:05 +04:00
return - 1 ;
}
2009-05-06 19:55:27 +04:00
2009-11-12 21:05:07 +03:00
static int map_err_sym_to_channel ( int err_sym , int sym_size )
{
if ( sym_size = = 4 )
switch ( err_sym ) {
case 0x20 :
case 0x21 :
return 0 ;
case 0x22 :
case 0x23 :
return 1 ;
default :
return err_sym > > 4 ;
}
/* x8 symbols */
else
switch ( err_sym ) {
/* imaginary bits not in a DIMM */
case 0x10 :
WARN ( 1 , KERN_ERR " Invalid error symbol: 0x%x \n " ,
err_sym ) ;
return - 1 ;
case 0x11 :
return 0 ;
case 0x12 :
return 1 ;
default :
return err_sym > > 3 ;
}
return - 1 ;
}
static int get_channel_from_ecc_syndrome ( struct mem_ctl_info * mci , u16 syndrome )
{
struct amd64_pvt * pvt = mci - > pvt_info ;
2010-03-09 14:46:00 +03:00
int err_sym = - 1 ;
2011-01-19 22:35:12 +03:00
if ( pvt - > ecc_sym_sz = = 8 )
2010-03-09 14:46:00 +03:00
err_sym = decode_syndrome ( syndrome , x8_vectors ,
ARRAY_SIZE ( x8_vectors ) ,
2011-01-19 22:35:12 +03:00
pvt - > ecc_sym_sz ) ;
else if ( pvt - > ecc_sym_sz = = 4 )
2010-03-09 14:46:00 +03:00
err_sym = decode_syndrome ( syndrome , x4_vectors ,
ARRAY_SIZE ( x4_vectors ) ,
2011-01-19 22:35:12 +03:00
pvt - > ecc_sym_sz ) ;
2010-03-09 14:46:00 +03:00
else {
2011-01-19 22:35:12 +03:00
amd64_warn ( " Illegal syndrome type: %u \n " , pvt - > ecc_sym_sz ) ;
2010-03-09 14:46:00 +03:00
return err_sym ;
2009-11-12 21:05:07 +03:00
}
2010-03-09 14:46:00 +03:00
2011-01-19 22:35:12 +03:00
return map_err_sym_to_channel ( err_sym , pvt - > ecc_sym_sz ) ;
2009-11-12 21:05:07 +03:00
}
2016-11-18 01:57:31 +03:00
static void __log_ecc_error ( struct mem_ctl_info * mci , struct err_info * err ,
2012-08-30 20:01:36 +04:00
u8 ecc_type )
2009-05-06 19:55:27 +04:00
{
2012-08-30 20:01:36 +04:00
enum hw_event_mc_err_type err_type ;
const char * string ;
2009-05-06 19:55:27 +04:00
2012-08-30 20:01:36 +04:00
if ( ecc_type = = 2 )
err_type = HW_EVENT_ERR_CORRECTED ;
else if ( ecc_type = = 1 )
err_type = HW_EVENT_ERR_UNCORRECTED ;
2016-11-18 01:57:32 +03:00
else if ( ecc_type = = 3 )
err_type = HW_EVENT_ERR_DEFERRED ;
2012-08-30 20:01:36 +04:00
else {
WARN ( 1 , " Something is rotten in the state of Denmark. \n " ) ;
2009-05-06 19:55:27 +04:00
return ;
}
2012-08-30 20:01:36 +04:00
switch ( err - > err_code ) {
case DECODE_OK :
string = " " ;
break ;
case ERR_NODE :
string = " Failed to map error addr to a node " ;
break ;
case ERR_CSROW :
string = " Failed to map error addr to a csrow " ;
break ;
case ERR_CHANNEL :
2016-11-28 21:59:53 +03:00
string = " Unknown syndrome - possible error reporting race " ;
break ;
case ERR_SYND :
string = " MCA_SYND not valid - unknown syndrome and csrow " ;
break ;
case ERR_NORM_ADDR :
string = " Cannot decode normalized address " ;
2012-08-30 20:01:36 +04:00
break ;
default :
string = " WTF error " ;
break ;
2009-05-06 19:55:27 +04:00
}
2012-08-30 20:01:36 +04:00
edac_mc_handle_error ( err_type , mci , 1 ,
err - > page , err - > offset , err - > syndrome ,
err - > csrow , err - > channel , - 1 ,
string , " " ) ;
2009-05-06 19:55:27 +04:00
}
2013-12-15 20:29:44 +04:00
static inline void decode_bus_error ( int node_id , struct mce * m )
2009-05-06 19:55:27 +04:00
{
2015-02-17 06:34:38 +03:00
struct mem_ctl_info * mci ;
struct amd64_pvt * pvt ;
2011-01-10 16:24:32 +03:00
u8 ecc_type = ( m - > status > > 45 ) & 0x3 ;
2012-08-09 20:41:07 +04:00
u8 xec = XEC ( m - > status , 0x1f ) ;
u16 ec = EC ( m - > status ) ;
2012-08-30 20:01:36 +04:00
u64 sys_addr ;
struct err_info err ;
2009-05-06 19:55:27 +04:00
2015-02-17 06:34:38 +03:00
mci = edac_mc_find ( node_id ) ;
if ( ! mci )
return ;
pvt = mci - > pvt_info ;
2012-08-09 20:41:07 +04:00
/* Bail out early if this was an 'observed' error */
2011-01-07 18:26:49 +03:00
if ( PP ( ec ) = = NBSL_PP_OBS )
2009-06-25 21:32:38 +04:00
return ;
2009-05-06 19:55:27 +04:00
2009-07-23 18:32:01 +04:00
/* Do only ECC errors */
if ( xec & & xec ! = F10_NBSL_EXT_ERR_ECC )
2009-05-06 19:55:27 +04:00
return ;
2012-08-30 20:01:36 +04:00
memset ( & err , 0 , sizeof ( err ) ) ;
2013-08-10 15:54:48 +04:00
sys_addr = get_error_address ( pvt , m ) ;
2012-08-30 20:01:36 +04:00
2009-07-23 18:32:01 +04:00
if ( ecc_type = = 2 )
2012-08-30 20:01:36 +04:00
err . syndrome = extract_syndrome ( m - > status ) ;
pvt - > ops - > map_sysaddr_to_csrow ( mci , sys_addr , & err ) ;
2016-11-18 01:57:31 +03:00
__log_ecc_error ( mci , & err , ecc_type ) ;
2009-05-06 19:55:27 +04:00
}
2016-11-28 21:59:53 +03:00
/*
* To find the UMC channel represented by this bank we need to match on its
* instance_id . The instance_id of a bank is held in the lower 32 bits of its
* IPID .
2019-02-28 18:36:10 +03:00
*
* Currently , we can derive the channel number by looking at the 6 th nibble in
* the instance_id . For example , instance_id = 0 xYXXXXX where Y is the channel
* number .
2016-11-28 21:59:53 +03:00
*/
2019-02-28 18:36:10 +03:00
static int find_umc_channel ( struct mce * m )
2016-11-28 21:59:53 +03:00
{
2019-02-28 18:36:10 +03:00
return ( m - > ipid & GENMASK ( 31 , 0 ) ) > > 20 ;
2016-11-28 21:59:53 +03:00
}
static void decode_umc_error ( int node_id , struct mce * m )
{
u8 ecc_type = ( m - > status > > 45 ) & 0x3 ;
struct mem_ctl_info * mci ;
struct amd64_pvt * pvt ;
struct err_info err ;
u64 sys_addr ;
mci = edac_mc_find ( node_id ) ;
if ( ! mci )
return ;
pvt = mci - > pvt_info ;
memset ( & err , 0 , sizeof ( err ) ) ;
if ( m - > status & MCI_STATUS_DEFERRED )
ecc_type = 3 ;
2019-02-28 18:36:10 +03:00
err . channel = find_umc_channel ( m ) ;
2016-11-28 21:59:53 +03:00
if ( ! ( m - > status & MCI_STATUS_SYNDV ) ) {
err . err_code = ERR_SYND ;
goto log_error ;
}
if ( ecc_type = = 2 ) {
u8 length = ( m - > synd > > 18 ) & 0x3f ;
if ( length )
err . syndrome = ( m - > synd > > 32 ) & GENMASK ( length - 1 , 0 ) ;
else
err . err_code = ERR_CHANNEL ;
}
err . csrow = m - > synd & 0x7 ;
2019-08-22 03:00:00 +03:00
if ( umc_normaddr_to_sysaddr ( m - > addr , pvt - > mc_node_id , err . channel , & sys_addr ) ) {
err . err_code = ERR_NORM_ADDR ;
goto log_error ;
}
error_address_to_page_and_offset ( sys_addr , & err ) ;
2016-11-28 21:59:53 +03:00
log_error :
__log_ecc_error ( mci , & err , ecc_type ) ;
}
2009-04-27 21:41:25 +04:00
/*
2016-05-06 20:44:27 +03:00
* Use pvt - > F3 which contains the F3 CPU PCI device to get the related
* F1 ( AddrMap ) and F2 ( Dct ) devices . Return negative value on error .
2016-11-18 01:57:36 +03:00
* Reserve F0 and F6 on systems with a UMC .
2009-04-27 21:41:25 +04:00
*/
2016-11-18 01:57:36 +03:00
static int
reserve_mc_sibling_devs ( struct amd64_pvt * pvt , u16 pci_id1 , u16 pci_id2 )
{
if ( pvt - > umc ) {
pvt - > F0 = pci_get_related_function ( pvt - > F3 - > vendor , pci_id1 , pvt - > F3 ) ;
if ( ! pvt - > F0 ) {
2020-12-15 20:01:31 +03:00
edac_dbg ( 1 , " F0 not found, device 0x%x \n " , pci_id1 ) ;
2016-11-18 01:57:36 +03:00
return - ENODEV ;
}
pvt - > F6 = pci_get_related_function ( pvt - > F3 - > vendor , pci_id2 , pvt - > F3 ) ;
if ( ! pvt - > F6 ) {
pci_dev_put ( pvt - > F0 ) ;
pvt - > F0 = NULL ;
2020-12-15 20:01:31 +03:00
edac_dbg ( 1 , " F6 not found: device 0x%x \n " , pci_id2 ) ;
2016-11-18 01:57:36 +03:00
return - ENODEV ;
}
2016-12-01 13:35:07 +03:00
2020-11-22 17:57:21 +03:00
if ( ! pci_ctl_dev )
pci_ctl_dev = & pvt - > F0 - > dev ;
2016-11-18 01:57:36 +03:00
edac_dbg ( 1 , " F0: %s \n " , pci_name ( pvt - > F0 ) ) ;
edac_dbg ( 1 , " F3: %s \n " , pci_name ( pvt - > F3 ) ) ;
edac_dbg ( 1 , " F6: %s \n " , pci_name ( pvt - > F6 ) ) ;
return 0 ;
}
2009-04-27 21:41:25 +04:00
/* Reserve the ADDRESS MAP Device */
2016-11-18 01:57:36 +03:00
pvt - > F1 = pci_get_related_function ( pvt - > F3 - > vendor , pci_id1 , pvt - > F3 ) ;
2010-10-01 22:11:07 +04:00
if ( ! pvt - > F1 ) {
2020-12-15 20:01:31 +03:00
edac_dbg ( 1 , " F1 not found: device 0x%x \n " , pci_id1 ) ;
2010-10-01 21:27:58 +04:00
return - ENODEV ;
2009-04-27 21:41:25 +04:00
}
2016-05-06 20:44:27 +03:00
/* Reserve the DCT Device */
2016-11-18 01:57:36 +03:00
pvt - > F2 = pci_get_related_function ( pvt - > F3 - > vendor , pci_id2 , pvt - > F3 ) ;
2016-05-06 20:44:27 +03:00
if ( ! pvt - > F2 ) {
2010-10-01 22:11:07 +04:00
pci_dev_put ( pvt - > F1 ) ;
pvt - > F1 = NULL ;
2009-04-27 21:41:25 +04:00
2020-12-15 20:01:31 +03:00
edac_dbg ( 1 , " F2 not found: device 0x%x \n " , pci_id2 ) ;
2016-12-01 13:35:07 +03:00
return - ENODEV ;
2009-04-27 21:41:25 +04:00
}
2016-11-18 01:57:36 +03:00
2020-11-22 17:57:21 +03:00
if ( ! pci_ctl_dev )
pci_ctl_dev = & pvt - > F2 - > dev ;
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " F1: %s \n " , pci_name ( pvt - > F1 ) ) ;
edac_dbg ( 1 , " F2: %s \n " , pci_name ( pvt - > F2 ) ) ;
edac_dbg ( 1 , " F3: %s \n " , pci_name ( pvt - > F3 ) ) ;
2009-04-27 21:41:25 +04:00
return 0 ;
}
2010-10-15 21:25:38 +04:00
static void free_mc_sibling_devs ( struct amd64_pvt * pvt )
2009-04-27 21:41:25 +04:00
{
2016-11-18 01:57:36 +03:00
if ( pvt - > umc ) {
pci_dev_put ( pvt - > F0 ) ;
pci_dev_put ( pvt - > F6 ) ;
} else {
pci_dev_put ( pvt - > F1 ) ;
pci_dev_put ( pvt - > F2 ) ;
}
2009-04-27 21:41:25 +04:00
}
2016-11-18 01:57:37 +03:00
static void determine_ecc_sym_sz ( struct amd64_pvt * pvt )
{
pvt - > ecc_sym_sz = 4 ;
if ( pvt - > umc ) {
u8 i ;
2019-02-28 18:36:10 +03:00
for_each_umc ( i ) {
2016-11-18 01:57:37 +03:00
/* Check enabled channels only: */
2019-02-28 18:36:11 +03:00
if ( pvt - > umc [ i ] . sdp_ctrl & UMC_SDP_INIT ) {
if ( pvt - > umc [ i ] . ecc_ctrl & BIT ( 9 ) ) {
pvt - > ecc_sym_sz = 16 ;
return ;
} else if ( pvt - > umc [ i ] . ecc_ctrl & BIT ( 7 ) ) {
pvt - > ecc_sym_sz = 8 ;
return ;
}
2016-11-18 01:57:37 +03:00
}
}
2019-02-28 18:36:11 +03:00
} else if ( pvt - > fam > = 0x10 ) {
2016-11-18 01:57:37 +03:00
u32 tmp ;
amd64_read_pci_cfg ( pvt - > F3 , EXT_NB_MCA_CFG , & tmp ) ;
/* F16h has only DCT0, so no need to read dbam1. */
if ( pvt - > fam ! = 0x16 )
amd64_read_dct_pci_cfg ( pvt , 1 , DBAM0 , & pvt - > dbam1 ) ;
/* F10h, revD and later can do x8 ECC too. */
if ( ( pvt - > fam > 0x10 | | pvt - > model > 7 ) & & tmp & BIT ( 25 ) )
pvt - > ecc_sym_sz = 8 ;
}
}
/*
* Retrieve the hardware registers of the memory controller .
*/
static void __read_mc_regs_df ( struct amd64_pvt * pvt )
{
u8 nid = pvt - > mc_node_id ;
struct amd64_umc * umc ;
u32 i , umc_base ;
/* Read registers from each UMC */
2019-02-28 18:36:10 +03:00
for_each_umc ( i ) {
2016-11-18 01:57:37 +03:00
umc_base = get_umc_base ( i ) ;
umc = & pvt - > umc [ i ] ;
2016-11-28 17:50:21 +03:00
amd_smn_read ( nid , umc_base + UMCCH_DIMM_CFG , & umc - > dimm_cfg ) ;
amd_smn_read ( nid , umc_base + UMCCH_UMC_CFG , & umc - > umc_cfg ) ;
2016-11-18 01:57:37 +03:00
amd_smn_read ( nid , umc_base + UMCCH_SDP_CTRL , & umc - > sdp_ctrl ) ;
amd_smn_read ( nid , umc_base + UMCCH_ECC_CTRL , & umc - > ecc_ctrl ) ;
2016-11-28 17:50:21 +03:00
amd_smn_read ( nid , umc_base + UMCCH_UMC_CAP_HI , & umc - > umc_cap_hi ) ;
2016-11-18 01:57:37 +03:00
}
}
2009-04-27 21:41:25 +04:00
/*
* Retrieve the hardware registers of the memory controller ( this includes the
* ' Address Map ' and ' Misc ' device regs )
*/
2010-10-15 21:25:38 +04:00
static void read_mc_regs ( struct amd64_pvt * pvt )
2009-04-27 21:41:25 +04:00
{
2016-11-18 01:57:37 +03:00
unsigned int range ;
2009-04-27 21:41:25 +04:00
u64 msr_val ;
/*
* Retrieve TOP_MEM and TOP_MEM2 ; no masking off of reserved bits since
2016-11-18 01:57:37 +03:00
* those are Read - As - Zero .
2009-04-27 21:41:25 +04:00
*/
2009-10-12 17:27:45 +04:00
rdmsrl ( MSR_K8_TOP_MEM1 , pvt - > top_mem ) ;
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " TOP_MEM: 0x%016llx \n " , pvt - > top_mem ) ;
2009-04-27 21:41:25 +04:00
2016-11-18 01:57:37 +03:00
/* Check first whether TOP_MEM2 is enabled: */
2009-04-27 21:41:25 +04:00
rdmsrl ( MSR_K8_SYSCFG , msr_val ) ;
2016-11-18 01:57:37 +03:00
if ( msr_val & BIT ( 21 ) ) {
2009-10-12 17:27:45 +04:00
rdmsrl ( MSR_K8_TOP_MEM2 , pvt - > top_mem2 ) ;
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " TOP_MEM2: 0x%016llx \n " , pvt - > top_mem2 ) ;
2016-11-18 01:57:37 +03:00
} else {
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " TOP_MEM2 disabled \n " ) ;
2016-11-18 01:57:37 +03:00
}
if ( pvt - > umc ) {
__read_mc_regs_df ( pvt ) ;
amd64_read_pci_cfg ( pvt - > F0 , DF_DHAR , & pvt - > dhar ) ;
goto skip ;
}
2009-04-27 21:41:25 +04:00
2011-01-07 18:26:49 +03:00
amd64_read_pci_cfg ( pvt - > F3 , NBCAP , & pvt - > nbcap ) ;
2009-04-27 21:41:25 +04:00
2011-01-17 19:52:57 +03:00
read_dram_ctl_register ( pvt ) ;
2009-04-27 21:41:25 +04:00
2010-10-21 20:52:53 +04:00
for ( range = 0 ; range < DRAM_RANGES ; range + + ) {
u8 rw ;
2009-04-27 21:41:25 +04:00
2010-10-21 20:52:53 +04:00
/* read settings for this DRAM range */
read_dram_base_limit_regs ( pvt , range ) ;
rw = dram_rw ( pvt , range ) ;
if ( ! rw )
continue ;
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " DRAM range[%d], base: 0x%016llx; limit: 0x%016llx \n " ,
range ,
get_dram_base ( pvt , range ) ,
get_dram_limit ( pvt , range ) ) ;
2010-10-21 20:52:53 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " IntlvEn=%s; Range access: %s%s IntlvSel=%d DstNode=%d \n " ,
dram_intlv_en ( pvt , range ) ? " Enabled " : " Disabled " ,
( rw & 0x1 ) ? " R " : " - " ,
( rw & 0x2 ) ? " W " : " - " ,
dram_intlv_sel ( pvt , range ) ,
dram_dst_node ( pvt , range ) ) ;
2009-04-27 21:41:25 +04:00
}
2010-11-11 19:29:13 +03:00
amd64_read_pci_cfg ( pvt - > F1 , DHAR , & pvt - > dhar ) ;
2014-09-15 20:37:38 +04:00
amd64_read_dct_pci_cfg ( pvt , 0 , DBAM0 , & pvt - > dbam0 ) ;
2009-04-27 21:41:25 +04:00
2010-10-01 22:11:07 +04:00
amd64_read_pci_cfg ( pvt - > F3 , F10_ONLINE_SPARE , & pvt - > online_spare ) ;
2009-04-27 21:41:25 +04:00
2014-09-15 20:37:38 +04:00
amd64_read_dct_pci_cfg ( pvt , 0 , DCLR0 , & pvt - > dclr0 ) ;
amd64_read_dct_pci_cfg ( pvt , 0 , DCHR0 , & pvt - > dchr0 ) ;
2009-04-27 21:41:25 +04:00
2010-12-22 21:31:45 +03:00
if ( ! dct_ganging_enabled ( pvt ) ) {
2014-09-15 20:37:38 +04:00
amd64_read_dct_pci_cfg ( pvt , 1 , DCLR0 , & pvt - > dclr1 ) ;
amd64_read_dct_pci_cfg ( pvt , 1 , DCHR0 , & pvt - > dchr1 ) ;
2009-04-27 21:41:25 +04:00
}
2010-03-09 14:46:00 +03:00
2016-11-18 01:57:37 +03:00
skip :
read_dct_base_mask ( pvt ) ;
2014-10-30 14:16:09 +03:00
determine_memory_type ( pvt ) ;
edac_dbg ( 1 , " DIMM type: %s \n " , edac_mem_types [ pvt - > dram_type ] ) ;
2011-01-19 22:35:12 +03:00
2016-11-18 01:57:37 +03:00
determine_ecc_sym_sz ( pvt ) ;
2009-04-27 21:41:25 +04:00
}
/*
* NOTE : CPU Revision Dependent code
*
* Input :
2010-11-29 21:49:02 +03:00
* @ csrow_nr ChipSelect Row Number ( 0. . NUM_CHIPSELECTS - 1 )
2009-04-27 21:41:25 +04:00
* k8 private pointer to - - >
* DRAM Bank Address mapping register
* node_id
* DCL register where dual_channel_active is
*
* The DBAM register consists of 4 sets of 4 bits each definitions :
*
* Bits : CSROWs
* 0 - 3 CSROWs 0 and 1
* 4 - 7 CSROWs 2 and 3
* 8 - 11 CSROWs 4 and 5
* 12 - 15 CSROWs 6 and 7
*
* Values range from : 0 to 15
* The meaning of the values depends on CPU revision and dual - channel state ,
* see relevant BKDG more info .
*
* The memory controller provides for total of only 8 CSROWs in its current
* architecture . Each " pair " of CSROWs normally represents just one DIMM in
* single channel or two ( 2 ) DIMMs in dual channel mode .
*
* The following code logic collapses the various tables for CSROW based on CPU
* revision .
*
* Returns :
* The number of PAGE_SIZE pages on the specified CSROW number it
* encompasses
*
*/
2017-04-27 20:11:54 +03:00
static u32 get_csrow_nr_pages ( struct amd64_pvt * pvt , u8 dct , int csrow_nr_orig )
2009-04-27 21:41:25 +04:00
{
2012-02-23 05:20:38 +04:00
u32 dbam = dct ? pvt - > dbam1 : pvt - > dbam0 ;
2017-04-27 20:11:54 +03:00
int csrow_nr = csrow_nr_orig ;
u32 cs_mode , nr_pages ;
2009-04-27 21:41:25 +04:00
2019-08-22 02:59:59 +03:00
if ( ! pvt - > umc ) {
2017-04-27 20:11:54 +03:00
csrow_nr > > = 1 ;
2019-08-22 02:59:59 +03:00
cs_mode = DBAM_DIMM ( csrow_nr , dbam ) ;
} else {
cs_mode = f17_get_cs_mode ( csrow_nr > > 1 , dct , pvt ) ;
}
2009-04-27 21:41:25 +04:00
2017-04-27 20:11:54 +03:00
nr_pages = pvt - > ops - > dbam_to_cs ( pvt , dct , cs_mode , csrow_nr ) ;
nr_pages < < = 20 - PAGE_SHIFT ;
2009-04-27 21:41:25 +04:00
2012-09-12 21:00:38 +04:00
edac_dbg ( 0 , " csrow: %d, channel: %d, DBAM idx: %d \n " ,
2017-04-27 20:11:54 +03:00
csrow_nr_orig , dct , cs_mode ) ;
2012-09-12 21:00:38 +04:00
edac_dbg ( 0 , " nr_pages/channel: %u \n " , nr_pages ) ;
2009-04-27 21:41:25 +04:00
return nr_pages ;
}
2019-08-22 02:59:57 +03:00
static int init_csrows_df ( struct mem_ctl_info * mci )
{
struct amd64_pvt * pvt = mci - > pvt_info ;
enum edac_type edac_mode = EDAC_NONE ;
enum dev_type dev_type = DEV_UNKNOWN ;
struct dimm_info * dimm ;
int empty = 1 ;
u8 umc , cs ;
if ( mci - > edac_ctl_cap & EDAC_FLAG_S16ECD16ED ) {
edac_mode = EDAC_S16ECD16ED ;
dev_type = DEV_X16 ;
} else if ( mci - > edac_ctl_cap & EDAC_FLAG_S8ECD8ED ) {
edac_mode = EDAC_S8ECD8ED ;
dev_type = DEV_X8 ;
} else if ( mci - > edac_ctl_cap & EDAC_FLAG_S4ECD4ED ) {
edac_mode = EDAC_S4ECD4ED ;
dev_type = DEV_X4 ;
} else if ( mci - > edac_ctl_cap & EDAC_FLAG_SECDED ) {
edac_mode = EDAC_SECDED ;
}
for_each_umc ( umc ) {
for_each_chip_select ( cs , umc , pvt ) {
if ( ! csrow_enabled ( cs , umc , pvt ) )
continue ;
empty = 0 ;
dimm = mci - > csrows [ cs ] - > channels [ umc ] - > dimm ;
edac_dbg ( 1 , " MC node: %d, csrow: %d \n " ,
pvt - > mc_node_id , cs ) ;
dimm - > nr_pages = get_csrow_nr_pages ( pvt , umc , cs ) ;
dimm - > mtype = pvt - > dram_type ;
dimm - > edac_mode = edac_mode ;
dimm - > dtype = dev_type ;
2019-10-22 23:35:14 +03:00
dimm - > grain = 64 ;
2019-08-22 02:59:57 +03:00
}
}
return empty ;
}
2009-04-27 21:41:25 +04:00
/*
* Initialize the array of csrow attribute instances , based on the values
* from pci config hardware registers .
*/
2010-10-15 21:25:38 +04:00
static int init_csrows ( struct mem_ctl_info * mci )
2009-04-27 21:41:25 +04:00
{
2012-09-12 21:00:38 +04:00
struct amd64_pvt * pvt = mci - > pvt_info ;
2016-11-29 17:51:56 +03:00
enum edac_type edac_mode = EDAC_NONE ;
2009-04-27 21:41:25 +04:00
struct csrow_info * csrow ;
2012-04-24 22:05:43 +04:00
struct dimm_info * dimm ;
2012-09-12 21:00:38 +04:00
int i , j , empty = 1 ;
2012-01-28 16:09:38 +04:00
int nr_pages = 0 ;
2012-09-12 21:00:38 +04:00
u32 val ;
2009-04-27 21:41:25 +04:00
2019-08-22 02:59:57 +03:00
if ( pvt - > umc )
return init_csrows_df ( mci ) ;
2009-04-27 21:41:25 +04:00
2019-08-22 02:59:57 +03:00
amd64_read_pci_cfg ( pvt - > F3 , NBCFG , & val ) ;
2009-04-27 21:41:25 +04:00
2019-08-22 02:59:57 +03:00
pvt - > nbcfg = val ;
edac_dbg ( 0 , " node %d, NBCFG=0x%08x[ChipKillEccCap: %d|DramEccEn: %d] \n " ,
pvt - > mc_node_id , val ,
! ! ( val & NBCFG_CHIPKILL ) , ! ! ( val & NBCFG_ECC_ENABLE ) ) ;
2009-04-27 21:41:25 +04:00
2012-09-12 21:00:38 +04:00
/*
* We iterate over DCT0 here but we look at DCT1 in parallel , if needed .
*/
2010-11-29 21:49:02 +03:00
for_each_chip_select ( i , 0 , pvt ) {
2012-09-12 21:00:38 +04:00
bool row_dct0 = ! ! csrow_enabled ( i , 0 , pvt ) ;
bool row_dct1 = false ;
2009-04-27 21:41:25 +04:00
2013-08-10 15:54:48 +04:00
if ( pvt - > fam ! = 0xf )
2012-09-12 21:00:38 +04:00
row_dct1 = ! ! csrow_enabled ( i , 1 , pvt ) ;
if ( ! row_dct0 & & ! row_dct1 )
2009-04-27 21:41:25 +04:00
continue ;
2012-09-12 21:00:38 +04:00
csrow = mci - > csrows [ i ] ;
2009-04-27 21:41:25 +04:00
empty = 0 ;
2012-09-12 21:00:38 +04:00
edac_dbg ( 1 , " MC node: %d, csrow: %d \n " ,
pvt - > mc_node_id , i ) ;
2013-03-11 16:07:46 +04:00
if ( row_dct0 ) {
2013-12-15 20:54:27 +04:00
nr_pages = get_csrow_nr_pages ( pvt , 0 , i ) ;
2013-03-11 16:07:46 +04:00
csrow - > channels [ 0 ] - > dimm - > nr_pages = nr_pages ;
}
2010-11-29 21:49:02 +03:00
2012-09-12 21:00:38 +04:00
/* K8 has only one DCT */
2013-08-10 15:54:48 +04:00
if ( pvt - > fam ! = 0xf & & row_dct1 ) {
2013-12-15 20:54:27 +04:00
int row_dct1_pages = get_csrow_nr_pages ( pvt , 1 , i ) ;
2013-03-11 16:07:46 +04:00
csrow - > channels [ 1 ] - > dimm - > nr_pages = row_dct1_pages ;
nr_pages + = row_dct1_pages ;
}
2009-04-27 21:41:25 +04:00
2012-09-12 21:00:38 +04:00
edac_dbg ( 1 , " Total csrow%d pages: %u \n " , i , nr_pages ) ;
2009-04-27 21:41:25 +04:00
2016-11-29 17:51:56 +03:00
/* Determine DIMM ECC mode: */
2019-08-22 02:59:57 +03:00
if ( pvt - > nbcfg & NBCFG_ECC_ENABLE ) {
2016-11-29 17:51:56 +03:00
edac_mode = ( pvt - > nbcfg & NBCFG_CHIPKILL )
? EDAC_S4ECD4ED
: EDAC_SECDED ;
}
2012-01-28 01:38:08 +04:00
for ( j = 0 ; j < pvt - > channel_count ; j + + ) {
2012-04-24 22:05:43 +04:00
dimm = csrow - > channels [ j ] - > dimm ;
2014-10-30 14:16:09 +03:00
dimm - > mtype = pvt - > dram_type ;
2012-04-24 22:05:43 +04:00
dimm - > edac_mode = edac_mode ;
2019-10-22 23:35:14 +03:00
dimm - > grain = 64 ;
2012-01-28 01:38:08 +04:00
}
2009-04-27 21:41:25 +04:00
}
return empty ;
}
2009-05-06 19:55:27 +04:00
2009-11-03 17:29:26 +03:00
/* get all cores on this DCT */
2012-11-27 10:32:10 +04:00
static void get_cpus_on_this_dct_cpumask ( struct cpumask * mask , u16 nid )
2009-11-03 17:29:26 +03:00
{
int cpu ;
for_each_online_cpu ( cpu )
2020-11-10 00:06:57 +03:00
if ( topology_die_id ( cpu ) = = nid )
2009-11-03 17:29:26 +03:00
cpumask_set_cpu ( cpu , mask ) ;
}
/* check MCG_CTL on all the cpus on this node */
2013-12-15 20:54:27 +04:00
static bool nb_mce_bank_enabled_on_node ( u16 nid )
2009-11-03 17:29:26 +03:00
{
cpumask_var_t mask ;
2009-12-11 20:14:40 +03:00
int cpu , nbe ;
2009-11-03 17:29:26 +03:00
bool ret = false ;
if ( ! zalloc_cpumask_var ( & mask , GFP_KERNEL ) ) {
2010-10-07 20:29:15 +04:00
amd64_warn ( " %s: Error allocating mask \n " , __func__ ) ;
2009-11-03 17:29:26 +03:00
return false ;
}
get_cpus_on_this_dct_cpumask ( mask , nid ) ;
rdmsr_on_cpus ( mask , MSR_IA32_MCG_CTL , msrs ) ;
for_each_cpu ( cpu , mask ) {
2009-12-11 20:14:40 +03:00
struct msr * reg = per_cpu_ptr ( msrs , cpu ) ;
2011-01-07 18:26:49 +03:00
nbe = reg - > l & MSR_MCGCTL_NBE ;
2009-11-03 17:29:26 +03:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " core: %u, MCG_CTL: 0x%llx, NB MSR is %s \n " ,
cpu , reg - > q ,
( nbe ? " enabled " : " disabled " ) ) ;
2009-11-03 17:29:26 +03:00
if ( ! nbe )
goto out ;
}
ret = true ;
out :
free_cpumask_var ( mask ) ;
return ret ;
}
2012-11-30 12:44:20 +04:00
static int toggle_ecc_err_reporting ( struct ecc_settings * s , u16 nid , bool on )
2009-11-03 17:29:26 +03:00
{
cpumask_var_t cmask ;
2009-12-11 20:14:40 +03:00
int cpu ;
2009-11-03 17:29:26 +03:00
if ( ! zalloc_cpumask_var ( & cmask , GFP_KERNEL ) ) {
2010-10-07 20:29:15 +04:00
amd64_warn ( " %s: error allocating mask \n " , __func__ ) ;
2016-12-04 09:07:18 +03:00
return - ENOMEM ;
2009-11-03 17:29:26 +03:00
}
2010-10-14 18:01:30 +04:00
get_cpus_on_this_dct_cpumask ( cmask , nid ) ;
2009-11-03 17:29:26 +03:00
rdmsr_on_cpus ( cmask , MSR_IA32_MCG_CTL , msrs ) ;
for_each_cpu ( cpu , cmask ) {
2009-12-11 20:14:40 +03:00
struct msr * reg = per_cpu_ptr ( msrs , cpu ) ;
2009-11-03 17:29:26 +03:00
if ( on ) {
2011-01-07 18:26:49 +03:00
if ( reg - > l & MSR_MCGCTL_NBE )
2010-10-14 18:01:30 +04:00
s - > flags . nb_mce_enable = 1 ;
2009-11-03 17:29:26 +03:00
2011-01-07 18:26:49 +03:00
reg - > l | = MSR_MCGCTL_NBE ;
2009-11-03 17:29:26 +03:00
} else {
/*
2010-02-24 16:49:47 +03:00
* Turn off NB MCE reporting only when it was off before
2009-11-03 17:29:26 +03:00
*/
2010-10-14 18:01:30 +04:00
if ( ! s - > flags . nb_mce_enable )
2011-01-07 18:26:49 +03:00
reg - > l & = ~ MSR_MCGCTL_NBE ;
2009-11-03 17:29:26 +03:00
}
}
wrmsr_on_cpus ( cmask , MSR_IA32_MCG_CTL , msrs ) ;
free_cpumask_var ( cmask ) ;
return 0 ;
}
2012-11-30 12:44:20 +04:00
static bool enable_ecc_error_reporting ( struct ecc_settings * s , u16 nid ,
2010-10-15 19:44:04 +04:00
struct pci_dev * F3 )
2009-04-27 21:46:08 +04:00
{
2010-10-15 19:44:04 +04:00
bool ret = true ;
2010-12-22 21:48:20 +03:00
u32 value , mask = 0x3 ; /* UECC/CECC enable */
2009-04-27 21:46:08 +04:00
2010-10-15 19:44:04 +04:00
if ( toggle_ecc_err_reporting ( s , nid , ON ) ) {
amd64_warn ( " Error enabling ECC reporting over MCGCTL! \n " ) ;
return false ;
}
2010-12-22 21:48:20 +03:00
amd64_read_pci_cfg ( F3 , NBCTL , & value ) ;
2009-04-27 21:46:08 +04:00
2010-10-14 18:01:30 +04:00
s - > old_nbctl = value & mask ;
s - > nbctl_valid = true ;
2009-04-27 21:46:08 +04:00
value | = mask ;
2010-12-22 21:48:20 +03:00
amd64_write_pci_cfg ( F3 , NBCTL , value ) ;
2009-04-27 21:46:08 +04:00
2010-12-23 16:07:18 +03:00
amd64_read_pci_cfg ( F3 , NBCFG , & value ) ;
2009-04-27 21:46:08 +04:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " 1: node %d, NBCFG=0x%08x[DramEccEn: %d] \n " ,
nid , value , ! ! ( value & NBCFG_ECC_ENABLE ) ) ;
2009-04-27 21:46:08 +04:00
2010-12-23 16:07:18 +03:00
if ( ! ( value & NBCFG_ECC_ENABLE ) ) {
2010-10-07 20:29:15 +04:00
amd64_warn ( " DRAM ECC disabled on this node, enabling... \n " ) ;
2009-04-27 21:46:08 +04:00
2010-10-14 18:01:30 +04:00
s - > flags . nb_ecc_prev = 0 ;
2010-02-24 16:49:47 +03:00
2009-04-27 21:46:08 +04:00
/* Attempt to turn on DRAM ECC Enable */
2010-12-23 16:07:18 +03:00
value | = NBCFG_ECC_ENABLE ;
amd64_write_pci_cfg ( F3 , NBCFG , value ) ;
2009-04-27 21:46:08 +04:00
2010-12-23 16:07:18 +03:00
amd64_read_pci_cfg ( F3 , NBCFG , & value ) ;
2009-04-27 21:46:08 +04:00
2010-12-23 16:07:18 +03:00
if ( ! ( value & NBCFG_ECC_ENABLE ) ) {
2010-10-07 20:29:15 +04:00
amd64_warn ( " Hardware rejected DRAM ECC enable, "
" check memory DIMM configuration. \n " ) ;
2010-10-15 19:44:04 +04:00
ret = false ;
2009-04-27 21:46:08 +04:00
} else {
2010-10-07 20:29:15 +04:00
amd64_info ( " Hardware accepted DRAM ECC Enable \n " ) ;
2009-04-27 21:46:08 +04:00
}
2010-02-24 16:49:47 +03:00
} else {
2010-10-14 18:01:30 +04:00
s - > flags . nb_ecc_prev = 1 ;
2009-04-27 21:46:08 +04:00
}
2010-02-24 16:49:47 +03:00
2012-04-30 00:08:39 +04:00
edac_dbg ( 0 , " 2: node %d, NBCFG=0x%08x[DramEccEn: %d] \n " ,
nid , value , ! ! ( value & NBCFG_ECC_ENABLE ) ) ;
2009-04-27 21:46:08 +04:00
2010-10-15 19:44:04 +04:00
return ret ;
2009-04-27 21:46:08 +04:00
}
2012-11-30 12:44:20 +04:00
static void restore_ecc_error_reporting ( struct ecc_settings * s , u16 nid ,
2010-10-15 21:25:38 +04:00
struct pci_dev * F3 )
2009-04-27 21:46:08 +04:00
{
2010-12-22 21:48:20 +03:00
u32 value , mask = 0x3 ; /* UECC/CECC enable */
2010-10-14 18:01:30 +04:00
if ( ! s - > nbctl_valid )
2009-04-27 21:46:08 +04:00
return ;
2010-12-22 21:48:20 +03:00
amd64_read_pci_cfg ( F3 , NBCTL , & value ) ;
2009-04-27 21:46:08 +04:00
value & = ~ mask ;
2010-10-14 18:01:30 +04:00
value | = s - > old_nbctl ;
2009-04-27 21:46:08 +04:00
2010-12-22 21:48:20 +03:00
amd64_write_pci_cfg ( F3 , NBCTL , value ) ;
2009-04-27 21:46:08 +04:00
2010-10-14 18:01:30 +04:00
/* restore previous BIOS DRAM ECC "off" setting we force-enabled */
if ( ! s - > flags . nb_ecc_prev ) {
2010-12-23 16:07:18 +03:00
amd64_read_pci_cfg ( F3 , NBCFG , & value ) ;
value & = ~ NBCFG_ECC_ENABLE ;
amd64_write_pci_cfg ( F3 , NBCFG , value ) ;
2010-02-24 16:49:47 +03:00
}
/* restore the NB Enable MCGCTL bit */
2010-10-15 19:44:04 +04:00
if ( toggle_ecc_err_reporting ( s , nid , OFF ) )
2010-10-07 20:29:15 +04:00
amd64_warn ( " Error restoring NB MCGCTL settings! \n " ) ;
2009-04-27 21:46:08 +04:00
}
2019-10-22 23:35:12 +03:00
static bool ecc_enabled ( struct amd64_pvt * pvt )
2009-04-27 21:46:08 +04:00
{
2019-10-22 23:35:12 +03:00
u16 nid = pvt - > mc_node_id ;
2009-09-16 15:05:46 +04:00
bool nb_mce_en = false ;
2016-11-18 01:57:34 +03:00
u8 ecc_en = 0 , i ;
u32 value ;
2009-04-27 21:46:08 +04:00
2016-11-18 01:57:34 +03:00
if ( boot_cpu_data . x86 > = 0x17 ) {
u8 umc_en_mask = 0 , ecc_en_mask = 0 ;
2019-10-22 23:35:12 +03:00
struct amd64_umc * umc ;
2009-04-27 21:46:08 +04:00
2019-02-28 18:36:10 +03:00
for_each_umc ( i ) {
2019-10-22 23:35:12 +03:00
umc = & pvt - > umc [ i ] ;
2016-11-18 01:57:34 +03:00
/* Only check enabled UMCs. */
2019-10-22 23:35:12 +03:00
if ( ! ( umc - > sdp_ctrl & UMC_SDP_INIT ) )
2016-11-18 01:57:34 +03:00
continue ;
umc_en_mask | = BIT ( i ) ;
2019-10-22 23:35:12 +03:00
if ( umc - > umc_cap_hi & UMC_ECC_ENABLED )
2016-11-18 01:57:34 +03:00
ecc_en_mask | = BIT ( i ) ;
}
/* Check whether at least one UMC is enabled: */
if ( umc_en_mask )
ecc_en = umc_en_mask = = ecc_en_mask ;
2017-01-27 20:24:19 +03:00
else
edac_dbg ( 0 , " Node %d: No enabled UMCs. \n " , nid ) ;
2016-11-18 01:57:34 +03:00
/* Assume UMC MCA banks are enabled. */
nb_mce_en = true ;
} else {
2019-10-22 23:35:12 +03:00
amd64_read_pci_cfg ( pvt - > F3 , NBCFG , & value ) ;
2009-04-27 21:46:08 +04:00
2016-11-18 01:57:34 +03:00
ecc_en = ! ! ( value & NBCFG_ECC_ENABLE ) ;
nb_mce_en = nb_mce_bank_enabled_on_node ( nid ) ;
if ( ! nb_mce_en )
2017-01-27 20:24:19 +03:00
edac_dbg ( 0 , " NB MCE bank disabled, set MSR 0x%08x[4] on node %d to enable. \n " ,
2016-11-18 01:57:34 +03:00
MSR_IA32_MCG_CTL , nid ) ;
}
2021-01-13 22:13:30 +03:00
edac_dbg ( 3 , " Node %d: DRAM ECC %s. \n " , nid , ( ecc_en ? " enabled " : " disabled " ) ) ;
2009-04-27 21:46:08 +04:00
2019-11-09 12:00:54 +03:00
if ( ! ecc_en | | ! nb_mce_en )
2010-10-15 19:44:04 +04:00
return false ;
2019-11-09 12:00:54 +03:00
else
return true ;
2009-04-27 21:46:08 +04:00
}
2016-11-29 17:51:56 +03:00
static inline void
f17h_determine_edac_ctl_cap ( struct mem_ctl_info * mci , struct amd64_pvt * pvt )
{
2019-08-22 02:59:56 +03:00
u8 i , ecc_en = 1 , cpk_en = 1 , dev_x4 = 1 , dev_x16 = 1 ;
2016-11-29 17:51:56 +03:00
2019-02-28 18:36:10 +03:00
for_each_umc ( i ) {
2016-11-29 17:51:56 +03:00
if ( pvt - > umc [ i ] . sdp_ctrl & UMC_SDP_INIT ) {
ecc_en & = ! ! ( pvt - > umc [ i ] . umc_cap_hi & UMC_ECC_ENABLED ) ;
cpk_en & = ! ! ( pvt - > umc [ i ] . umc_cap_hi & UMC_ECC_CHIPKILL_CAP ) ;
2019-08-22 02:59:56 +03:00
dev_x4 & = ! ! ( pvt - > umc [ i ] . dimm_cfg & BIT ( 6 ) ) ;
dev_x16 & = ! ! ( pvt - > umc [ i ] . dimm_cfg & BIT ( 7 ) ) ;
2016-11-29 17:51:56 +03:00
}
}
/* Set chipkill only if ECC is enabled: */
if ( ecc_en ) {
mci - > edac_ctl_cap | = EDAC_FLAG_SECDED ;
2019-08-22 02:59:56 +03:00
if ( ! cpk_en )
return ;
if ( dev_x4 )
2016-11-29 17:51:56 +03:00
mci - > edac_ctl_cap | = EDAC_FLAG_S4ECD4ED ;
2019-08-22 02:59:56 +03:00
else if ( dev_x16 )
mci - > edac_ctl_cap | = EDAC_FLAG_S16ECD16ED ;
else
mci - > edac_ctl_cap | = EDAC_FLAG_S8ECD8ED ;
2016-11-29 17:51:56 +03:00
}
}
2019-10-22 23:35:09 +03:00
static void setup_mci_misc_attrs ( struct mem_ctl_info * mci )
2009-04-27 22:01:01 +04:00
{
struct amd64_pvt * pvt = mci - > pvt_info ;
mci - > mtype_cap = MEM_FLAG_DDR2 | MEM_FLAG_RDDR2 ;
mci - > edac_ctl_cap = EDAC_FLAG_NONE ;
2016-11-29 17:51:56 +03:00
if ( pvt - > umc ) {
f17h_determine_edac_ctl_cap ( mci , pvt ) ;
} else {
if ( pvt - > nbcap & NBCAP_SECDED )
mci - > edac_ctl_cap | = EDAC_FLAG_SECDED ;
2009-04-27 22:01:01 +04:00
2016-11-29 17:51:56 +03:00
if ( pvt - > nbcap & NBCAP_CHIPKILL )
mci - > edac_ctl_cap | = EDAC_FLAG_S4ECD4ED ;
}
2009-04-27 22:01:01 +04:00
2013-12-15 20:54:27 +04:00
mci - > edac_cap = determine_edac_cap ( pvt ) ;
2009-04-27 22:01:01 +04:00
mci - > mod_name = EDAC_MOD_STR ;
2019-10-22 23:35:09 +03:00
mci - > ctl_name = fam_type - > ctl_name ;
2016-11-18 01:57:30 +03:00
mci - > dev_name = pci_name ( pvt - > F3 ) ;
2009-04-27 22:01:01 +04:00
mci - > ctl_page_to_phys = NULL ;
/* memory scrubber interface */
2013-12-15 20:54:27 +04:00
mci - > set_sdram_scrub_rate = set_scrub_rate ;
mci - > get_sdram_scrub_rate = get_scrub_rate ;
2009-04-27 22:01:01 +04:00
}
2010-10-01 21:20:05 +04:00
/*
* returns a pointer to the family descriptor on success , NULL otherwise .
*/
2013-12-15 20:54:27 +04:00
static struct amd64_family_type * per_family_init ( struct amd64_pvt * pvt )
2010-10-01 20:38:19 +04:00
{
2013-08-09 20:54:49 +04:00
pvt - > ext_model = boot_cpu_data . x86_model > > 4 ;
2018-01-01 04:52:10 +03:00
pvt - > stepping = boot_cpu_data . x86_stepping ;
2013-08-09 20:54:49 +04:00
pvt - > model = boot_cpu_data . x86_model ;
pvt - > fam = boot_cpu_data . x86 ;
switch ( pvt - > fam ) {
2010-10-01 20:38:19 +04:00
case 0xf :
2013-12-15 20:54:27 +04:00
fam_type = & family_types [ K8_CPUS ] ;
pvt - > ops = & family_types [ K8_CPUS ] . ops ;
2010-10-01 20:38:19 +04:00
break ;
2011-01-19 20:15:10 +03:00
2010-10-01 20:38:19 +04:00
case 0x10 :
2013-12-15 20:54:27 +04:00
fam_type = & family_types [ F10_CPUS ] ;
pvt - > ops = & family_types [ F10_CPUS ] . ops ;
2011-01-19 20:15:10 +03:00
break ;
case 0x15 :
2013-08-09 20:54:49 +04:00
if ( pvt - > model = = 0x30 ) {
2013-12-15 20:54:27 +04:00
fam_type = & family_types [ F15_M30H_CPUS ] ;
pvt - > ops = & family_types [ F15_M30H_CPUS ] . ops ;
2013-08-09 20:54:49 +04:00
break ;
2014-10-30 14:16:09 +03:00
} else if ( pvt - > model = = 0x60 ) {
fam_type = & family_types [ F15_M60H_CPUS ] ;
pvt - > ops = & family_types [ F15_M60H_CPUS ] . ops ;
break ;
2020-12-12 17:20:28 +03:00
/* Richland is only client */
} else if ( pvt - > model = = 0x13 ) {
return NULL ;
} else {
fam_type = & family_types [ F15_CPUS ] ;
pvt - > ops = & family_types [ F15_CPUS ] . ops ;
2013-08-09 20:54:49 +04:00
}
2010-10-01 20:38:19 +04:00
break ;
2013-04-17 23:57:13 +04:00
case 0x16 :
2014-02-20 20:28:46 +04:00
if ( pvt - > model = = 0x30 ) {
fam_type = & family_types [ F16_M30H_CPUS ] ;
pvt - > ops = & family_types [ F16_M30H_CPUS ] . ops ;
break ;
}
2013-12-15 20:54:27 +04:00
fam_type = & family_types [ F16_CPUS ] ;
pvt - > ops = & family_types [ F16_CPUS ] . ops ;
2013-04-17 23:57:13 +04:00
break ;
2016-11-18 01:57:35 +03:00
case 0x17 :
2018-08-16 22:28:40 +03:00
if ( pvt - > model > = 0x10 & & pvt - > model < = 0x2f ) {
fam_type = & family_types [ F17_M10H_CPUS ] ;
pvt - > ops = & family_types [ F17_M10H_CPUS ] . ops ;
break ;
2019-02-28 18:36:09 +03:00
} else if ( pvt - > model > = 0x30 & & pvt - > model < = 0x3f ) {
fam_type = & family_types [ F17_M30H_CPUS ] ;
pvt - > ops = & family_types [ F17_M30H_CPUS ] . ops ;
break ;
2020-05-10 23:48:42 +03:00
} else if ( pvt - > model > = 0x60 & & pvt - > model < = 0x6f ) {
fam_type = & family_types [ F17_M60H_CPUS ] ;
pvt - > ops = & family_types [ F17_M60H_CPUS ] . ops ;
break ;
2019-09-07 02:21:38 +03:00
} else if ( pvt - > model > = 0x70 & & pvt - > model < = 0x7f ) {
fam_type = & family_types [ F17_M70H_CPUS ] ;
pvt - > ops = & family_types [ F17_M70H_CPUS ] . ops ;
break ;
2018-08-16 22:28:40 +03:00
}
2020-08-24 01:36:59 +03:00
fallthrough ;
2018-09-27 17:31:28 +03:00
case 0x18 :
2016-11-18 01:57:35 +03:00
fam_type = & family_types [ F17_CPUS ] ;
pvt - > ops = & family_types [ F17_CPUS ] . ops ;
2018-09-27 17:31:28 +03:00
if ( pvt - > fam = = 0x18 )
family_types [ F17_CPUS ] . ctl_name = " F18h " ;
2016-11-18 01:57:35 +03:00
break ;
2020-01-10 04:56:50 +03:00
case 0x19 :
2020-10-09 20:18:03 +03:00
if ( pvt - > model > = 0x20 & & pvt - > model < = 0x2f ) {
fam_type = & family_types [ F17_M70H_CPUS ] ;
pvt - > ops = & family_types [ F17_M70H_CPUS ] . ops ;
fam_type - > ctl_name = " F19h_M20h " ;
break ;
}
2020-01-10 04:56:50 +03:00
fam_type = & family_types [ F19_CPUS ] ;
pvt - > ops = & family_types [ F19_CPUS ] . ops ;
family_types [ F19_CPUS ] . ctl_name = " F19h " ;
break ;
2010-10-01 20:38:19 +04:00
default :
2010-10-07 20:29:15 +04:00
amd64_err ( " Unsupported family! \n " ) ;
2010-10-01 21:20:05 +04:00
return NULL ;
2010-10-01 20:38:19 +04:00
}
2010-10-01 21:20:05 +04:00
return fam_type ;
2010-10-01 20:38:19 +04:00
}
2015-02-04 13:48:53 +03:00
static const struct attribute_group * amd64_edac_attr_groups [ ] = {
# ifdef CONFIG_EDAC_DEBUG
2020-12-14 22:47:11 +03:00
& dbg_group ,
2020-12-15 11:18:44 +03:00
& inj_group ,
2015-02-04 13:48:53 +03:00
# endif
NULL
} ;
2019-10-22 23:35:10 +03:00
static int hw_info_get ( struct amd64_pvt * pvt )
2009-04-27 22:01:01 +04:00
{
2016-11-18 01:57:36 +03:00
u16 pci_id1 , pci_id2 ;
2020-04-29 18:48:47 +03:00
int ret ;
2010-10-01 20:38:19 +04:00
2016-11-18 01:57:36 +03:00
if ( pvt - > fam > = 0x17 ) {
2019-10-22 23:35:11 +03:00
pvt - > umc = kcalloc ( fam_type - > max_mcs , sizeof ( struct amd64_umc ) , GFP_KERNEL ) ;
2019-10-22 23:35:10 +03:00
if ( ! pvt - > umc )
return - ENOMEM ;
2016-11-18 01:57:36 +03:00
pci_id1 = fam_type - > f0_id ;
pci_id2 = fam_type - > f6_id ;
} else {
pci_id1 = fam_type - > f1_id ;
pci_id2 = fam_type - > f2_id ;
}
2019-10-22 23:35:10 +03:00
ret = reserve_mc_sibling_devs ( pvt , pci_id1 , pci_id2 ) ;
if ( ret )
return ret ;
2009-04-27 22:01:01 +04:00
2010-10-15 21:25:38 +04:00
read_mc_regs ( pvt ) ;
2009-04-27 22:01:01 +04:00
2019-10-22 23:35:10 +03:00
return 0 ;
}
static void hw_info_put ( struct amd64_pvt * pvt )
{
if ( pvt - > F0 | | pvt - > F1 )
free_mc_sibling_devs ( pvt ) ;
kfree ( pvt - > umc ) ;
}
static int init_one_instance ( struct amd64_pvt * pvt )
{
struct mem_ctl_info * mci = NULL ;
struct edac_mc_layer layers [ 2 ] ;
int ret = - EINVAL ;
2009-04-27 22:01:01 +04:00
/*
* We need to determine how many memory channels there are . Then use
* that information for calculating the size of the dynamic instance
2010-10-15 21:25:38 +04:00
* tables in the ' mci ' structure .
2009-04-27 22:01:01 +04:00
*/
pvt - > channel_count = pvt - > ops - > early_channel_count ( pvt ) ;
if ( pvt - > channel_count < 0 )
2019-10-22 23:35:10 +03:00
return ret ;
2009-04-27 22:01:01 +04:00
ret = - ENOMEM ;
2012-04-16 22:03:50 +04:00
layers [ 0 ] . type = EDAC_MC_LAYER_CHIP_SELECT ;
layers [ 0 ] . size = pvt - > csels [ 0 ] . b_cnt ;
layers [ 0 ] . is_virt_csrow = true ;
layers [ 1 ] . type = EDAC_MC_LAYER_CHANNEL ;
2013-07-23 22:01:23 +04:00
/*
* Always allocate two channels since we can have setups with DIMMs on
* only one channel . Also , this simplifies handling later for the price
* of a couple of KBs tops .
*/
2019-10-22 23:35:11 +03:00
layers [ 1 ] . size = fam_type - > max_mcs ;
2012-04-16 22:03:50 +04:00
layers [ 1 ] . is_virt_csrow = false ;
2013-07-23 22:01:23 +04:00
2019-10-22 23:35:10 +03:00
mci = edac_mc_alloc ( pvt - > mc_node_id , ARRAY_SIZE ( layers ) , layers , 0 ) ;
2009-04-27 22:01:01 +04:00
if ( ! mci )
2019-10-22 23:35:10 +03:00
return ret ;
2009-04-27 22:01:01 +04:00
mci - > pvt_info = pvt ;
2016-05-06 20:44:27 +03:00
mci - > pdev = & pvt - > F3 - > dev ;
2009-04-27 22:01:01 +04:00
2019-10-22 23:35:09 +03:00
setup_mci_misc_attrs ( mci ) ;
2010-10-15 21:25:38 +04:00
if ( init_csrows ( mci ) )
2009-04-27 22:01:01 +04:00
mci - > edac_cap = EDAC_FLAG_NONE ;
ret = - ENODEV ;
2015-02-04 13:48:53 +03:00
if ( edac_mc_add_mc_with_groups ( mci , amd64_edac_attr_groups ) ) {
2012-04-30 00:08:39 +04:00
edac_dbg ( 1 , " failed edac_mc_add_mc() \n " ) ;
2019-10-22 23:35:10 +03:00
edac_mc_free ( mci ) ;
return ret ;
2009-04-27 22:01:01 +04:00
}
return 0 ;
}
2019-11-06 04:25:01 +03:00
static bool instance_has_memory ( struct amd64_pvt * pvt )
{
bool cs_enabled = false ;
int cs = 0 , dct = 0 ;
for ( dct = 0 ; dct < fam_type - > max_mcs ; dct + + ) {
for_each_chip_select ( cs , dct , pvt )
cs_enabled | = csrow_enabled ( cs , dct , pvt ) ;
}
return cs_enabled ;
}
2016-05-06 20:44:27 +03:00
static int probe_one_instance ( unsigned int nid )
2009-04-27 22:01:01 +04:00
{
2010-10-15 19:44:04 +04:00
struct pci_dev * F3 = node_to_amd_nb ( nid ) - > misc ;
2019-10-22 23:35:10 +03:00
struct amd64_pvt * pvt = NULL ;
2010-10-14 18:01:30 +04:00
struct ecc_settings * s ;
2016-05-06 20:44:27 +03:00
int ret ;
2009-04-27 22:01:01 +04:00
2010-10-14 18:01:30 +04:00
ret = - ENOMEM ;
s = kzalloc ( sizeof ( struct ecc_settings ) , GFP_KERNEL ) ;
if ( ! s )
2010-10-15 19:44:04 +04:00
goto err_out ;
2010-10-14 18:01:30 +04:00
ecc_stngs [ nid ] = s ;
2019-10-22 23:35:10 +03:00
pvt = kzalloc ( sizeof ( struct amd64_pvt ) , GFP_KERNEL ) ;
if ( ! pvt )
goto err_settings ;
pvt - > mc_node_id = nid ;
pvt - > F3 = F3 ;
2020-12-12 17:20:28 +03:00
ret = - ENODEV ;
2019-10-22 23:35:10 +03:00
fam_type = per_family_init ( pvt ) ;
if ( ! fam_type )
goto err_enable ;
ret = hw_info_get ( pvt ) ;
if ( ret < 0 )
goto err_enable ;
2019-11-06 04:25:01 +03:00
ret = 0 ;
if ( ! instance_has_memory ( pvt ) ) {
amd64_info ( " Node %d: No DIMMs detected. \n " , nid ) ;
goto err_enable ;
}
2019-10-22 23:35:12 +03:00
if ( ! ecc_enabled ( pvt ) ) {
2019-11-06 04:25:01 +03:00
ret = - ENODEV ;
2010-10-15 19:44:04 +04:00
if ( ! ecc_enable_override )
goto err_enable ;
2016-11-23 00:40:16 +03:00
if ( boot_cpu_data . x86 > = 0x17 ) {
amd64_warn ( " Forcing ECC on is not recommended on newer systems. Please enable ECC in BIOS. " ) ;
goto err_enable ;
} else
amd64_warn ( " Forcing ECC on! \n " ) ;
2010-10-15 19:44:04 +04:00
if ( ! enable_ecc_error_reporting ( s , nid , F3 ) )
goto err_enable ;
}
2019-10-22 23:35:10 +03:00
ret = init_one_instance ( pvt ) ;
2010-10-15 21:25:38 +04:00
if ( ret < 0 ) {
2010-10-14 18:01:30 +04:00
amd64_err ( " Error probing instance: %d \n " , nid ) ;
2016-11-23 00:40:16 +03:00
if ( boot_cpu_data . x86 < 0x17 )
restore_ecc_error_reporting ( s , nid , F3 ) ;
2017-01-25 01:32:24 +03:00
goto err_enable ;
2010-10-15 21:25:38 +04:00
}
2009-04-27 22:01:01 +04:00
2021-01-13 22:13:30 +03:00
amd64_info ( " %s %sdetected (node %d). \n " , fam_type - > ctl_name ,
( pvt - > fam = = 0xf ?
( pvt - > ext_model > = K8_REV_F ? " revF or later "
: " revE or earlier " )
: " " ) , pvt - > mc_node_id ) ;
2019-11-06 04:25:01 +03:00
dump_misc_regs ( pvt ) ;
2009-04-27 22:01:01 +04:00
return ret ;
2010-10-15 19:44:04 +04:00
err_enable :
2019-10-22 23:35:10 +03:00
hw_info_put ( pvt ) ;
kfree ( pvt ) ;
err_settings :
2010-10-15 19:44:04 +04:00
kfree ( s ) ;
ecc_stngs [ nid ] = NULL ;
err_out :
return ret ;
2009-04-27 22:01:01 +04:00
}
2016-05-06 20:44:27 +03:00
static void remove_one_instance ( unsigned int nid )
2009-04-27 22:01:01 +04:00
{
2010-10-15 21:25:38 +04:00
struct pci_dev * F3 = node_to_amd_nb ( nid ) - > misc ;
struct ecc_settings * s = ecc_stngs [ nid ] ;
2016-05-06 20:44:27 +03:00
struct mem_ctl_info * mci ;
struct amd64_pvt * pvt ;
2009-04-27 22:01:01 +04:00
/* Remove from EDAC CORE tracking list */
2016-05-06 20:44:27 +03:00
mci = edac_mc_del_mc ( & F3 - > dev ) ;
2009-04-27 22:01:01 +04:00
if ( ! mci )
return ;
pvt = mci - > pvt_info ;
2010-10-15 21:25:38 +04:00
restore_ecc_error_reporting ( s , nid , F3 ) ;
2009-04-27 22:01:01 +04:00
2010-10-15 21:25:38 +04:00
kfree ( ecc_stngs [ nid ] ) ;
ecc_stngs [ nid ] = NULL ;
2010-10-14 18:01:30 +04:00
2009-04-27 22:01:01 +04:00
/* Free the EDAC CORE resources */
2009-12-21 17:15:59 +03:00
mci - > pvt_info = NULL ;
2019-10-22 23:35:10 +03:00
hw_info_put ( pvt ) ;
2009-12-21 17:15:59 +03:00
kfree ( pvt ) ;
2009-04-27 22:01:01 +04:00
edac_mc_free ( mci ) ;
}
2010-10-15 21:25:38 +04:00
static void setup_pci_device ( void )
2009-04-27 22:01:01 +04:00
{
2013-12-15 20:54:27 +04:00
if ( pci_ctl )
2009-04-27 22:01:01 +04:00
return ;
2020-11-22 17:57:21 +03:00
pci_ctl = edac_pci_create_generic_ctl ( pci_ctl_dev , EDAC_MOD_STR ) ;
2013-12-15 20:54:27 +04:00
if ( ! pci_ctl ) {
pr_warn ( " %s(): Unable to create PCI control \n " , __func__ ) ;
pr_warn ( " %s(): PCI error report via EDAC not set \n " , __func__ ) ;
2009-04-27 22:01:01 +04:00
}
}
2016-09-16 03:07:17 +03:00
static const struct x86_cpu_id amd64_cpuids [ ] = {
2020-03-20 16:13:55 +03:00
X86_MATCH_VENDOR_FAM ( AMD , 0x0F , NULL ) ,
X86_MATCH_VENDOR_FAM ( AMD , 0x10 , NULL ) ,
X86_MATCH_VENDOR_FAM ( AMD , 0x15 , NULL ) ,
X86_MATCH_VENDOR_FAM ( AMD , 0x16 , NULL ) ,
X86_MATCH_VENDOR_FAM ( AMD , 0x17 , NULL ) ,
X86_MATCH_VENDOR_FAM ( HYGON , 0x18 , NULL ) ,
X86_MATCH_VENDOR_FAM ( AMD , 0x19 , NULL ) ,
2016-09-16 03:07:17 +03:00
{ }
} ;
MODULE_DEVICE_TABLE ( x86cpu , amd64_cpuids ) ;
2009-04-27 22:01:01 +04:00
static int __init amd64_edac_init ( void )
{
2017-08-24 01:54:47 +03:00
const char * owner ;
2010-10-15 21:25:38 +04:00
int err = - ENODEV ;
2016-05-06 20:44:27 +03:00
int i ;
2009-04-27 22:01:01 +04:00
2017-08-24 01:54:47 +03:00
owner = edac_get_owner ( ) ;
if ( owner & & strncmp ( owner , EDAC_MOD_STR , sizeof ( EDAC_MOD_STR ) ) )
return - EBUSY ;
2017-01-27 20:24:23 +03:00
if ( ! x86_match_cpu ( amd64_cpuids ) )
return - ENODEV ;
2010-10-29 19:14:31 +04:00
if ( amd_cache_northbridges ( ) < 0 )
2017-01-27 20:24:23 +03:00
return - ENODEV ;
2009-04-27 22:01:01 +04:00
2016-06-16 02:13:18 +03:00
opstate_init ( ) ;
2010-10-13 18:11:59 +04:00
err = - ENOMEM ;
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-13 00:03:40 +03:00
ecc_stngs = kcalloc ( amd_nb_num ( ) , sizeof ( ecc_stngs [ 0 ] ) , GFP_KERNEL ) ;
2015-02-17 12:58:34 +03:00
if ( ! ecc_stngs )
2011-03-29 20:10:53 +04:00
goto err_free ;
2010-10-13 18:11:59 +04:00
2009-12-11 20:14:40 +03:00
msrs = msrs_alloc ( ) ;
2009-12-21 20:13:01 +03:00
if ( ! msrs )
2010-10-15 21:25:38 +04:00
goto err_free ;
2009-12-11 20:14:40 +03:00
2017-01-13 18:52:19 +03:00
for ( i = 0 ; i < amd_nb_num ( ) ; i + + ) {
err = probe_one_instance ( i ) ;
if ( err ) {
2016-05-06 20:44:27 +03:00
/* unwind properly */
while ( - - i > = 0 )
remove_one_instance ( i ) ;
2009-04-27 22:01:01 +04:00
2016-05-06 20:44:27 +03:00
goto err_pci ;
}
2017-01-13 18:52:19 +03:00
}
2009-04-27 22:01:01 +04:00
2017-01-27 20:24:22 +03:00
if ( ! edac_has_mcs ( ) ) {
err = - ENODEV ;
goto err_pci ;
}
2017-01-25 01:32:25 +03:00
/* register stuff with EDAC MCE */
if ( boot_cpu_data . x86 > = 0x17 )
amd_register_ecc_decoder ( decode_umc_error ) ;
else
amd_register_ecc_decoder ( decode_bus_error ) ;
2010-10-15 21:25:38 +04:00
setup_pci_device ( ) ;
2014-11-02 13:22:12 +03:00
# ifdef CONFIG_X86_32
amd64_err ( " %s on 32-bit is unsupported. USE AT YOUR OWN RISK! \n " , EDAC_MOD_STR ) ;
# endif
2016-04-27 13:21:21 +03:00
printk ( KERN_INFO " AMD64 EDAC driver v%s \n " , EDAC_AMD64_VERSION ) ;
2010-10-15 21:25:38 +04:00
return 0 ;
2009-04-27 22:01:01 +04:00
2009-12-21 20:13:01 +03:00
err_pci :
2020-11-22 17:57:21 +03:00
pci_ctl_dev = NULL ;
2009-12-21 20:13:01 +03:00
msrs_free ( msrs ) ;
msrs = NULL ;
2010-10-13 18:11:59 +04:00
2010-10-15 21:25:38 +04:00
err_free :
kfree ( ecc_stngs ) ;
ecc_stngs = NULL ;
2009-04-27 22:01:01 +04:00
return err ;
}
static void __exit amd64_edac_exit ( void )
{
2016-05-06 20:44:27 +03:00
int i ;
2013-12-15 20:54:27 +04:00
if ( pci_ctl )
edac_pci_release_generic_ctl ( pci_ctl ) ;
2009-04-27 22:01:01 +04:00
2017-01-25 01:32:25 +03:00
/* unregister from EDAC MCE */
if ( boot_cpu_data . x86 > = 0x17 )
amd_unregister_ecc_decoder ( decode_umc_error ) ;
else
amd_unregister_ecc_decoder ( decode_bus_error ) ;
2016-05-06 20:44:27 +03:00
for ( i = 0 ; i < amd_nb_num ( ) ; i + + )
remove_one_instance ( i ) ;
2009-12-11 20:14:40 +03:00
2010-10-14 18:01:30 +04:00
kfree ( ecc_stngs ) ;
ecc_stngs = NULL ;
2020-11-22 17:57:21 +03:00
pci_ctl_dev = NULL ;
2009-12-11 20:14:40 +03:00
msrs_free ( msrs ) ;
msrs = NULL ;
2009-04-27 22:01:01 +04:00
}
module_init ( amd64_edac_init ) ;
module_exit ( amd64_edac_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " SoftwareBitMaker: Doug Thompson, "
" Dave Peterson, Thayne Harbaugh " ) ;
MODULE_DESCRIPTION ( " MC support for AMD64 memory controllers - "
EDAC_AMD64_VERSION ) ;
module_param ( edac_op_state , int , 0444 ) ;
MODULE_PARM_DESC ( edac_op_state , " EDAC Error Reporting state: 0=Poll,1=NMI " ) ;