提交 4140c542 编写于 作者: B Borislav Petkov 提交者: Mauro Carvalho Chehab

i7core_edac: Drop the edac_mce facility

Remove edac_mce pieces and use the normal MCE decoder notifier chain by
retaining the same functionality with considerably less code.
Signed-off-by: NBorislav Petkov <borislav.petkov@amd.com>
Signed-off-by: NMauro Carvalho Chehab <mchehab@redhat.com>
上级 f0cb5452
...@@ -36,7 +36,6 @@ ...@@ -36,7 +36,6 @@
#include <linux/fs.h> #include <linux/fs.h>
#include <linux/mm.h> #include <linux/mm.h>
#include <linux/debugfs.h> #include <linux/debugfs.h>
#include <linux/edac_mce.h>
#include <linux/irq_work.h> #include <linux/irq_work.h>
#include <asm/processor.h> #include <asm/processor.h>
......
...@@ -71,9 +71,6 @@ config EDAC_MM_EDAC ...@@ -71,9 +71,6 @@ config EDAC_MM_EDAC
occurred so that a particular failing memory module can be occurred so that a particular failing memory module can be
replaced. If unsure, select 'Y'. replaced. If unsure, select 'Y'.
config EDAC_MCE
bool
config EDAC_AMD64 config EDAC_AMD64
tristate "AMD64 (Opteron, Athlon64) K8, F10h" tristate "AMD64 (Opteron, Athlon64) K8, F10h"
depends on EDAC_MM_EDAC && AMD_NB && X86_64 && EDAC_DECODE_MCE depends on EDAC_MM_EDAC && AMD_NB && X86_64 && EDAC_DECODE_MCE
......
...@@ -8,7 +8,6 @@ ...@@ -8,7 +8,6 @@
obj-$(CONFIG_EDAC) := edac_stub.o obj-$(CONFIG_EDAC) := edac_stub.o
obj-$(CONFIG_EDAC_MM_EDAC) += edac_core.o obj-$(CONFIG_EDAC_MM_EDAC) += edac_core.o
obj-$(CONFIG_EDAC_MCE) += edac_mce.o
edac_core-y := edac_mc.o edac_device.o edac_mc_sysfs.o edac_pci_sysfs.o edac_core-y := edac_mc.o edac_device.o edac_mc_sysfs.o edac_pci_sysfs.o
edac_core-y += edac_module.o edac_device_sysfs.o edac_core-y += edac_module.o edac_device_sysfs.o
......
/* Provides edac interface to mcelog events
*
* This file may be distributed under the terms of the
* GNU General Public License version 2.
*
* Copyright (c) 2009 by:
* Mauro Carvalho Chehab <mchehab@redhat.com>
*
* Red Hat Inc. http://www.redhat.com
*/
#include <linux/module.h>
#include <linux/edac_mce.h>
#include <asm/mce.h>
int edac_mce_enabled;
EXPORT_SYMBOL_GPL(edac_mce_enabled);
/*
* Extension interface
*/
static LIST_HEAD(edac_mce_list);
static DEFINE_MUTEX(edac_mce_lock);
int edac_mce_register(struct edac_mce *edac_mce)
{
mutex_lock(&edac_mce_lock);
list_add_tail(&edac_mce->list, &edac_mce_list);
mutex_unlock(&edac_mce_lock);
return 0;
}
EXPORT_SYMBOL(edac_mce_register);
void edac_mce_unregister(struct edac_mce *edac_mce)
{
mutex_lock(&edac_mce_lock);
list_del(&edac_mce->list);
mutex_unlock(&edac_mce_lock);
}
EXPORT_SYMBOL(edac_mce_unregister);
int edac_mce_parse(struct mce *mce)
{
struct edac_mce *edac_mce;
list_for_each_entry(edac_mce, &edac_mce_list, list) {
if (edac_mce->check_error(edac_mce->priv, mce))
return 1;
}
/* Nobody queued the error */
return 0;
}
EXPORT_SYMBOL_GPL(edac_mce_parse);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Mauro Carvalho Chehab <mchehab@redhat.com>");
MODULE_AUTHOR("Red Hat Inc. (http://www.redhat.com)");
MODULE_DESCRIPTION("EDAC Driver for mcelog captured errors");
...@@ -33,8 +33,8 @@ ...@@ -33,8 +33,8 @@
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/edac.h> #include <linux/edac.h>
#include <linux/mmzone.h> #include <linux/mmzone.h>
#include <linux/edac_mce.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <asm/mce.h>
#include <asm/processor.h> #include <asm/processor.h>
#include "edac_core.h" #include "edac_core.h"
...@@ -265,9 +265,6 @@ struct i7core_pvt { ...@@ -265,9 +265,6 @@ struct i7core_pvt {
bool is_registered, enable_scrub; bool is_registered, enable_scrub;
/* mcelog glue */
struct edac_mce edac_mce;
/* Fifo double buffers */ /* Fifo double buffers */
struct mce mce_entry[MCE_LOG_LEN]; struct mce mce_entry[MCE_LOG_LEN];
struct mce mce_outentry[MCE_LOG_LEN]; struct mce mce_outentry[MCE_LOG_LEN];
...@@ -1899,33 +1896,43 @@ static void i7core_check_error(struct mem_ctl_info *mci) ...@@ -1899,33 +1896,43 @@ static void i7core_check_error(struct mem_ctl_info *mci)
* WARNING: As this routine should be called at NMI time, extra care should * WARNING: As this routine should be called at NMI time, extra care should
* be taken to avoid deadlocks, and to be as fast as possible. * be taken to avoid deadlocks, and to be as fast as possible.
*/ */
static int i7core_mce_check_error(void *priv, struct mce *mce) static int i7core_mce_check_error(struct notifier_block *nb, unsigned long val,
void *data)
{ {
struct mem_ctl_info *mci = priv; struct mce *mce = (struct mce *)data;
struct i7core_pvt *pvt = mci->pvt_info; struct i7core_dev *i7_dev;
struct mem_ctl_info *mci;
struct i7core_pvt *pvt;
i7_dev = get_i7core_dev(mce->socketid);
if (!i7_dev)
return NOTIFY_BAD;
mci = i7_dev->mci;
pvt = mci->pvt_info;
/* /*
* Just let mcelog handle it if the error is * Just let mcelog handle it if the error is
* outside the memory controller * outside the memory controller
*/ */
if (((mce->status & 0xffff) >> 7) != 1) if (((mce->status & 0xffff) >> 7) != 1)
return 0; return NOTIFY_DONE;
/* Bank 8 registers are the only ones that we know how to handle */ /* Bank 8 registers are the only ones that we know how to handle */
if (mce->bank != 8) if (mce->bank != 8)
return 0; return NOTIFY_DONE;
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
/* Only handle if it is the right mc controller */ /* Only handle if it is the right mc controller */
if (mce->socketid != pvt->i7core_dev->socket) if (mce->socketid != pvt->i7core_dev->socket)
return 0; return NOTIFY_DONE;
#endif #endif
smp_rmb(); smp_rmb();
if ((pvt->mce_out + 1) % MCE_LOG_LEN == pvt->mce_in) { if ((pvt->mce_out + 1) % MCE_LOG_LEN == pvt->mce_in) {
smp_wmb(); smp_wmb();
pvt->mce_overrun++; pvt->mce_overrun++;
return 0; return NOTIFY_DONE;
} }
/* Copy memory error at the ringbuffer */ /* Copy memory error at the ringbuffer */
...@@ -1938,9 +1945,13 @@ static int i7core_mce_check_error(void *priv, struct mce *mce) ...@@ -1938,9 +1945,13 @@ static int i7core_mce_check_error(void *priv, struct mce *mce)
i7core_check_error(mci); i7core_check_error(mci);
/* Advise mcelog that the errors were handled */ /* Advise mcelog that the errors were handled */
return 1; return NOTIFY_STOP;
} }
static struct notifier_block i7_mce_dec = {
.notifier_call = i7core_mce_check_error,
};
/* /*
* set_sdram_scrub_rate This routine sets byte/sec bandwidth scrub rate * set_sdram_scrub_rate This routine sets byte/sec bandwidth scrub rate
* to hardware according to SCRUBINTERVAL formula * to hardware according to SCRUBINTERVAL formula
...@@ -2093,8 +2104,7 @@ static void i7core_unregister_mci(struct i7core_dev *i7core_dev) ...@@ -2093,8 +2104,7 @@ static void i7core_unregister_mci(struct i7core_dev *i7core_dev)
if (pvt->enable_scrub) if (pvt->enable_scrub)
disable_sdram_scrub_setting(mci); disable_sdram_scrub_setting(mci);
/* Disable MCE NMI handler */ atomic_notifier_chain_unregister(&x86_mce_decoder_chain, &i7_mce_dec);
edac_mce_unregister(&pvt->edac_mce);
/* Disable EDAC polling */ /* Disable EDAC polling */
i7core_pci_ctl_release(pvt); i7core_pci_ctl_release(pvt);
...@@ -2193,21 +2203,10 @@ static int i7core_register_mci(struct i7core_dev *i7core_dev) ...@@ -2193,21 +2203,10 @@ static int i7core_register_mci(struct i7core_dev *i7core_dev)
/* allocating generic PCI control info */ /* allocating generic PCI control info */
i7core_pci_ctl_create(pvt); i7core_pci_ctl_create(pvt);
/* Registers on edac_mce in order to receive memory errors */ atomic_notifier_chain_register(&x86_mce_decoder_chain, &i7_mce_dec);
pvt->edac_mce.priv = mci;
pvt->edac_mce.check_error = i7core_mce_check_error;
rc = edac_mce_register(&pvt->edac_mce);
if (unlikely(rc < 0)) {
debugf0("MC: " __FILE__
": %s(): failed edac_mce_register()\n", __func__);
goto fail1;
}
return 0; return 0;
fail1:
i7core_pci_ctl_release(pvt);
edac_mc_del_mc(mci->dev);
fail0: fail0:
kfree(mci->ctl_name); kfree(mci->ctl_name);
edac_mc_free(mci); edac_mc_free(mci);
......
/* Provides edac interface to mcelog events
*
* This file may be distributed under the terms of the
* GNU General Public License version 2.
*
* Copyright (c) 2009 by:
* Mauro Carvalho Chehab <mchehab@redhat.com>
*
* Red Hat Inc. http://www.redhat.com
*/
#if defined(CONFIG_EDAC_MCE) || \
(defined(CONFIG_EDAC_MCE_MODULE) && defined(MODULE))
#include <asm/mce.h>
#include <linux/list.h>
struct edac_mce {
struct list_head list;
void *priv;
int (*check_error)(void *priv, struct mce *mce);
};
int edac_mce_register(struct edac_mce *edac_mce);
void edac_mce_unregister(struct edac_mce *edac_mce);
int edac_mce_parse(struct mce *mce);
#else
#define edac_mce_parse(mce) (0)
#endif
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册