提交 8da8533d 编写于 作者: L Linus Torvalds

Merge git://git.kernel.org/pub/scm/linux/kernel/git/mchehab/linux-edac

Pull EDAC patches from Mauro Carvalho Chehab:

 - the second part of the EDAC rework:
    - Add the sysfs nodes that exports the real memory layout, instead
      of the fake one (needed to properly represent Intel memory
      controllers since 2002)
    - convert EDAC MC to use "struct device" instead of creating the
      sysfs nodes via the kobj API
    - adds a tracepoint to represent memory errors

 - some cleanup patches

 - some fixes at i5000, i5400 and EDAC core

 - a new EDAC driver for Caldera.

* git://git.kernel.org/pub/scm/linux/kernel/git/mchehab/linux-edac: (33 commits)
  edac i5000, i5400: fix pointer math in i5000_get_mc_regs()
  edac: allow specifying the error count with fake_inject
  edac: add support for Calxeda highbank L2 cache ecc
  edac: add support for Calxeda highbank memory controller
  edac: create top-level debugfs directory
  sb_edac: properly handle error count
  i7core_edac: properly handle error count
  edac: edac_mc_handle_error(): add an error_count parameter
  edac: remove arch-specific parameter for the error handler
  amd64_edac: Don't pass driver name as an error parameter
  edac_mc: check for allocation failure in edac_mc_alloc()
  edac: Increase version to 3.0.0
  edac_mc: Cleanup per-dimm_info debug messages
  edac: Convert debugfX to edac_dbg(X,
  edac: Use more normal debugging macro style
  edac: Don't add __func__ or __FILE__ for debugf[0-9] msgs
  Edac: Add ABI Documentation for the new device nodes
  edac: move documentation ABI to ABI/testing/sysfs-devices-edac
  i7core_edac: change the mem allocation scheme to make Documentation/kobject.txt happy
  edac: change the mem allocation scheme to make Documentation/kobject.txt happy
  ...
What: /sys/devices/system/edac/mc/mc*/reset_counters
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This write-only control file will zero all the statistical
counters for UE and CE errors on the given memory controller.
Zeroing the counters will also reset the timer indicating how
long since the last counter were reset. This is useful for
computing errors/time. Since the counters are always reset
at driver initialization time, no module/kernel parameter
is available.
What: /sys/devices/system/edac/mc/mc*/seconds_since_reset
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays how many seconds have elapsed
since the last counter reset. This can be used with the error
counters to measure error rates.
What: /sys/devices/system/edac/mc/mc*/mc_name
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays the type of memory controller
that is being utilized.
What: /sys/devices/system/edac/mc/mc*/size_mb
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays, in count of megabytes, of memory
that this memory controller manages.
What: /sys/devices/system/edac/mc/mc*/ue_count
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays the total count of uncorrectable
errors that have occurred on this memory controller. If
panic_on_ue is set, this counter will not have a chance to
increment, since EDAC will panic the system
What: /sys/devices/system/edac/mc/mc*/ue_noinfo_count
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays the number of UEs that have
occurred on this memory controller with no information as to
which DIMM slot is having errors.
What: /sys/devices/system/edac/mc/mc*/ce_count
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays the total count of correctable
errors that have occurred on this memory controller. This
count is very important to examine. CEs provide early
indications that a DIMM is beginning to fail. This count
field should be monitored for non-zero values and report
such information to the system administrator.
What: /sys/devices/system/edac/mc/mc*/ce_noinfo_count
Date: January 2006
Contact: linux-edac@vger.kernel.org
Description: This attribute file displays the number of CEs that
have occurred on this memory controller wherewith no
information as to which DIMM slot is having errors. Memory is
handicapped, but operational, yet no information is available
to indicate which slot the failing memory is in. This count
field should be also be monitored for non-zero values.
What: /sys/devices/system/edac/mc/mc*/sdram_scrub_rate
Date: February 2007
Contact: linux-edac@vger.kernel.org
Description: Read/Write attribute file that controls memory scrubbing.
The scrubbing rate used by the memory controller is set by
writing a minimum bandwidth in bytes/sec to the attribute file.
The rate will be translated to an internal value that gives at
least the specified rate.
Reading the file will return the actual scrubbing rate employed.
If configuration fails or memory scrubbing is not implemented,
the value of the attribute file will be -1.
What: /sys/devices/system/edac/mc/mc*/max_location
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This attribute file displays the information about the last
available memory slot in this memory controller. It is used by
userspace tools in order to display the memory filling layout.
What: /sys/devices/system/edac/mc/mc*/(dimm|rank)*/size
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This attribute file will display the size of dimm or rank.
For dimm*/size, this is the size, in MB of the DIMM memory
stick. For rank*/size, this is the size, in MB for one rank
of the DIMM memory stick. On single rank memories (1R), this
is also the total size of the dimm. On dual rank (2R) memories,
this is half the size of the total DIMM memories.
What: /sys/devices/system/edac/mc/mc*/(dimm|rank)*/dimm_dev_type
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This attribute file will display what type of DRAM device is
being utilized on this DIMM (x1, x2, x4, x8, ...).
What: /sys/devices/system/edac/mc/mc*/(dimm|rank)*/dimm_edac_mode
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This attribute file will display what type of Error detection
and correction is being utilized. For example: S4ECD4ED would
mean a Chipkill with x4 DRAM.
What: /sys/devices/system/edac/mc/mc*/(dimm|rank)*/dimm_label
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This control file allows this DIMM to have a label assigned
to it. With this label in the module, when errors occur
the output can provide the DIMM label in the system log.
This becomes vital for panic events to isolate the
cause of the UE event.
DIMM Labels must be assigned after booting, with information
that correctly identifies the physical slot with its
silk screen label. This information is currently very
motherboard specific and determination of this information
must occur in userland at this time.
What: /sys/devices/system/edac/mc/mc*/(dimm|rank)*/dimm_location
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This attribute file will display the location (csrow/channel,
branch/channel/slot or channel/slot) of the dimm or rank.
What: /sys/devices/system/edac/mc/mc*/(dimm|rank)*/dimm_mem_type
Date: April 2012
Contact: Mauro Carvalho Chehab <mchehab@redhat.com>
linux-edac@vger.kernel.org
Description: This attribute file will display what type of memory is
currently on this csrow. Normally, either buffered or
unbuffered memory (for example, Unbuffered-DDR3).
Calxeda Highbank L2 cache ECC
Properties:
- compatible : Should be "calxeda,hb-sregs-l2-ecc"
- reg : Address and size for ECC error interrupt clear registers.
- interrupts : Should be single bit error interrupt, then double bit error
interrupt.
Example:
sregs@fff3c200 {
compatible = "calxeda,hb-sregs-l2-ecc";
reg = <0xfff3c200 0x100>;
interrupts = <0 71 4 0 72 4>;
};
Calxeda DDR memory controller
Properties:
- compatible : Should be "calxeda,hb-ddr-ctrl"
- reg : Address and size for DDR controller registers.
- interrupts : Interrupt for DDR controller.
Example:
memory-controller@fff00000 {
compatible = "calxeda,hb-ddr-ctrl";
reg = <0xfff00000 0x1000>;
interrupts = <0 91 4>;
};
...@@ -232,116 +232,20 @@ EDAC control and attribute files. ...@@ -232,116 +232,20 @@ EDAC control and attribute files.
In 'mcX' directories are EDAC control and attribute files for In 'mcX' directories are EDAC control and attribute files for
this 'X' instance of the memory controllers: this 'X' instance of the memory controllers.
Counter reset control file:
'reset_counters'
This write-only control file will zero all the statistical counters
for UE and CE errors. Zeroing the counters will also reset the timer
indicating how long since the last counter zero. This is useful
for computing errors/time. Since the counters are always reset at
driver initialization time, no module/kernel parameter is available.
RUN TIME: echo "anything" >/sys/devices/system/edac/mc/mc0/counter_reset
This resets the counters on memory controller 0
Seconds since last counter reset control file:
'seconds_since_reset'
This attribute file displays how many seconds have elapsed since the
last counter reset. This can be used with the error counters to
measure error rates.
Memory Controller name attribute file:
'mc_name'
This attribute file displays the type of memory controller
that is being utilized.
Total memory managed by this memory controller attribute file:
'size_mb'
This attribute file displays, in count of megabytes, of memory
that this instance of memory controller manages.
Total Uncorrectable Errors count attribute file:
'ue_count'
This attribute file displays the total count of uncorrectable
errors that have occurred on this memory controller. If panic_on_ue
is set this counter will not have a chance to increment,
since EDAC will panic the system.
Total UE count that had no information attribute fileY:
'ue_noinfo_count'
This attribute file displays the number of UEs that have occurred
with no information as to which DIMM slot is having errors.
Total Correctable Errors count attribute file:
'ce_count'
This attribute file displays the total count of correctable
errors that have occurred on this memory controller. This
count is very important to examine. CEs provide early
indications that a DIMM is beginning to fail. This count
field should be monitored for non-zero values and report
such information to the system administrator.
Total Correctable Errors count attribute file:
'ce_noinfo_count'
This attribute file displays the number of CEs that
have occurred wherewith no information as to which DIMM slot
is having errors. Memory is handicapped, but operational,
yet no information is available to indicate which slot
the failing memory is in. This count field should be also
be monitored for non-zero values.
Device Symlink:
'device'
Symlink to the memory controller device.
Sdram memory scrubbing rate:
'sdram_scrub_rate'
Read/Write attribute file that controls memory scrubbing. The scrubbing
rate is set by writing a minimum bandwidth in bytes/sec to the attribute
file. The rate will be translated to an internal value that gives at
least the specified rate.
Reading the file will return the actual scrubbing rate employed.
If configuration fails or memory scrubbing is not implemented, accessing
that attribute will fail.
For a description of the sysfs API, please see:
Documentation/ABI/testing/sysfs/devices-edac
============================================================================ ============================================================================
'csrowX' DIRECTORIES 'csrowX' DIRECTORIES
When CONFIG_EDAC_LEGACY_SYSFS is enabled, the sysfs will contain the
csrowX directories. As this API doesn't work properly for Rambus, FB-DIMMs
and modern Intel Memory Controllers, this is being deprecated in favor
of dimmX directories.
In the 'csrowX' directories are EDAC control and attribute files for In the 'csrowX' directories are EDAC control and attribute files for
this 'X' instance of csrow: this 'X' instance of csrow:
......
...@@ -130,6 +130,12 @@ ...@@ -130,6 +130,12 @@
clocks = <&eclk>; clocks = <&eclk>;
}; };
memory-controller@fff00000 {
compatible = "calxeda,hb-ddr-ctrl";
reg = <0xfff00000 0x1000>;
interrupts = <0 91 4>;
};
ipc@fff20000 { ipc@fff20000 {
compatible = "arm,pl320", "arm,primecell"; compatible = "arm,pl320", "arm,primecell";
reg = <0xfff20000 0x1000>; reg = <0xfff20000 0x1000>;
...@@ -275,6 +281,12 @@ ...@@ -275,6 +281,12 @@
}; };
}; };
sregs@fff3c200 {
compatible = "calxeda,hb-sregs-l2-ecc";
reg = <0xfff3c200 0x100>;
interrupts = <0 71 4 0 72 4>;
};
dma@fff3d000 { dma@fff3d000 {
compatible = "arm,pl330", "arm,primecell"; compatible = "arm,pl330", "arm,primecell";
reg = <0xfff3d000 0x1000>; reg = <0xfff3d000 0x1000>;
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
menuconfig EDAC menuconfig EDAC
bool "EDAC (Error Detection And Correction) reporting" bool "EDAC (Error Detection And Correction) reporting"
depends on HAS_IOMEM depends on HAS_IOMEM
depends on X86 || PPC || TILE depends on X86 || PPC || TILE || ARM
help help
EDAC is designed to report errors in the core system. EDAC is designed to report errors in the core system.
These are low-level errors that are reported in the CPU or These are low-level errors that are reported in the CPU or
...@@ -31,6 +31,14 @@ if EDAC ...@@ -31,6 +31,14 @@ if EDAC
comment "Reporting subsystems" comment "Reporting subsystems"
config EDAC_LEGACY_SYSFS
bool "EDAC legacy sysfs"
default y
help
Enable the compatibility sysfs nodes.
Use 'Y' if your edac utilities aren't ported to work with the newer
structures.
config EDAC_DEBUG config EDAC_DEBUG
bool "Debugging" bool "Debugging"
help help
...@@ -294,4 +302,18 @@ config EDAC_TILE ...@@ -294,4 +302,18 @@ config EDAC_TILE
Support for error detection and correction on the Support for error detection and correction on the
Tilera memory controller. Tilera memory controller.
config EDAC_HIGHBANK_MC
tristate "Highbank Memory Controller"
depends on EDAC_MM_EDAC && ARCH_HIGHBANK
help
Support for error detection and correction on the
Calxeda Highbank memory controller.
config EDAC_HIGHBANK_L2
tristate "Highbank L2 Cache"
depends on EDAC_MM_EDAC && ARCH_HIGHBANK
help
Support for error detection and correction on the
Calxeda Highbank memory controller.
endif # EDAC endif # EDAC
...@@ -55,3 +55,6 @@ obj-$(CONFIG_EDAC_AMD8111) += amd8111_edac.o ...@@ -55,3 +55,6 @@ obj-$(CONFIG_EDAC_AMD8111) += amd8111_edac.o
obj-$(CONFIG_EDAC_AMD8131) += amd8131_edac.o obj-$(CONFIG_EDAC_AMD8131) += amd8131_edac.o
obj-$(CONFIG_EDAC_TILE) += tile_edac.o obj-$(CONFIG_EDAC_TILE) += tile_edac.o
obj-$(CONFIG_EDAC_HIGHBANK_MC) += highbank_mc_edac.o
obj-$(CONFIG_EDAC_HIGHBANK_L2) += highbank_l2_edac.o
此差异已折叠。
...@@ -413,20 +413,33 @@ struct ecc_settings { ...@@ -413,20 +413,33 @@ struct ecc_settings {
}; };
#ifdef CONFIG_EDAC_DEBUG #ifdef CONFIG_EDAC_DEBUG
#define NUM_DBG_ATTRS 5 int amd64_create_sysfs_dbg_files(struct mem_ctl_info *mci);
void amd64_remove_sysfs_dbg_files(struct mem_ctl_info *mci);
#else #else
#define NUM_DBG_ATTRS 0 static inline int amd64_create_sysfs_dbg_files(struct mem_ctl_info *mci)
{
return 0;
}
static void inline amd64_remove_sysfs_dbg_files(struct mem_ctl_info *mci)
{
}
#endif #endif
#ifdef CONFIG_EDAC_AMD64_ERROR_INJECTION #ifdef CONFIG_EDAC_AMD64_ERROR_INJECTION
#define NUM_INJ_ATTRS 5 int amd64_create_sysfs_inject_files(struct mem_ctl_info *mci);
void amd64_remove_sysfs_inject_files(struct mem_ctl_info *mci);
#else #else
#define NUM_INJ_ATTRS 0 static inline int amd64_create_sysfs_inject_files(struct mem_ctl_info *mci)
{
return 0;
}
static inline void amd64_remove_sysfs_inject_files(struct mem_ctl_info *mci)
{
}
#endif #endif
extern struct mcidev_sysfs_attribute amd64_dbg_attrs[NUM_DBG_ATTRS],
amd64_inj_attrs[NUM_INJ_ATTRS];
/* /*
* Each of the PCI Device IDs types have their own set of hardware accessor * Each of the PCI Device IDs types have their own set of hardware accessor
* functions and per device encoding/decoding logic. * functions and per device encoding/decoding logic.
...@@ -460,3 +473,5 @@ int __amd64_write_pci_cfg_dword(struct pci_dev *pdev, int offset, ...@@ -460,3 +473,5 @@ int __amd64_write_pci_cfg_dword(struct pci_dev *pdev, int offset,
int amd64_get_dram_hole_info(struct mem_ctl_info *mci, u64 *hole_base, int amd64_get_dram_hole_info(struct mem_ctl_info *mci, u64 *hole_base,
u64 *hole_offset, u64 *hole_size); u64 *hole_offset, u64 *hole_size);
#define to_mci(k) container_of(k, struct mem_ctl_info, dev)
#include "amd64_edac.h" #include "amd64_edac.h"
#define EDAC_DCT_ATTR_SHOW(reg) \ #define EDAC_DCT_ATTR_SHOW(reg) \
static ssize_t amd64_##reg##_show(struct mem_ctl_info *mci, char *data) \ static ssize_t amd64_##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; \ struct amd64_pvt *pvt = mci->pvt_info; \
return sprintf(data, "0x%016llx\n", (u64)pvt->reg); \ return sprintf(data, "0x%016llx\n", (u64)pvt->reg); \
} }
...@@ -12,8 +15,12 @@ EDAC_DCT_ATTR_SHOW(dbam0); ...@@ -12,8 +15,12 @@ EDAC_DCT_ATTR_SHOW(dbam0);
EDAC_DCT_ATTR_SHOW(top_mem); EDAC_DCT_ATTR_SHOW(top_mem);
EDAC_DCT_ATTR_SHOW(top_mem2); EDAC_DCT_ATTR_SHOW(top_mem2);
static ssize_t amd64_hole_show(struct mem_ctl_info *mci, char *data) static ssize_t amd64_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_base = 0;
u64 hole_offset = 0; u64 hole_offset = 0;
u64 hole_size = 0; u64 hole_size = 0;
...@@ -27,46 +34,40 @@ static ssize_t amd64_hole_show(struct mem_ctl_info *mci, char *data) ...@@ -27,46 +34,40 @@ static ssize_t amd64_hole_show(struct mem_ctl_info *mci, char *data)
/* /*
* update NUM_DBG_ATTRS in case you add new members * update NUM_DBG_ATTRS in case you add new members
*/ */
struct mcidev_sysfs_attribute amd64_dbg_attrs[] = { static DEVICE_ATTR(dhar, S_IRUGO, amd64_dhar_show, NULL);
static DEVICE_ATTR(dbam, S_IRUGO, amd64_dbam0_show, NULL);
static DEVICE_ATTR(topmem, S_IRUGO, amd64_top_mem_show, NULL);
static DEVICE_ATTR(topmem2, S_IRUGO, amd64_top_mem2_show, NULL);
static DEVICE_ATTR(dram_hole, S_IRUGO, amd64_hole_show, NULL);
int amd64_create_sysfs_dbg_files(struct mem_ctl_info *mci)
{
int rc;
rc = device_create_file(&mci->dev, &dev_attr_dhar);
if (rc < 0)
return rc;
rc = device_create_file(&mci->dev, &dev_attr_dbam);
if (rc < 0)
return rc;
rc = device_create_file(&mci->dev, &dev_attr_topmem);
if (rc < 0)
return rc;
rc = device_create_file(&mci->dev, &dev_attr_topmem2);
if (rc < 0)
return rc;
rc = device_create_file(&mci->dev, &dev_attr_dram_hole);
if (rc < 0)
return rc;
{ return 0;
.attr = { }
.name = "dhar",
.mode = (S_IRUGO) void amd64_remove_sysfs_dbg_files(struct mem_ctl_info *mci)
}, {
.show = amd64_dhar_show, device_remove_file(&mci->dev, &dev_attr_dhar);
.store = NULL, device_remove_file(&mci->dev, &dev_attr_dbam);
}, device_remove_file(&mci->dev, &dev_attr_topmem);
{ device_remove_file(&mci->dev, &dev_attr_topmem2);
.attr = { device_remove_file(&mci->dev, &dev_attr_dram_hole);
.name = "dbam", }
.mode = (S_IRUGO)
},
.show = amd64_dbam0_show,
.store = NULL,
},
{
.attr = {
.name = "topmem",
.mode = (S_IRUGO)
},
.show = amd64_top_mem_show,
.store = NULL,
},
{
.attr = {
.name = "topmem2",
.mode = (S_IRUGO)
},
.show = amd64_top_mem2_show,
.store = NULL,
},
{
.attr = {
.name = "dram_hole",
.mode = (S_IRUGO)
},
.show = amd64_hole_show,
.store = NULL,
},
};
#include "amd64_edac.h" #include "amd64_edac.h"
static ssize_t amd64_inject_section_show(struct mem_ctl_info *mci, char *buf) static ssize_t amd64_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; struct amd64_pvt *pvt = mci->pvt_info;
return sprintf(buf, "0x%x\n", pvt->injection.section); return sprintf(buf, "0x%x\n", pvt->injection.section);
} }
...@@ -12,9 +15,11 @@ static ssize_t amd64_inject_section_show(struct mem_ctl_info *mci, char *buf) ...@@ -12,9 +15,11 @@ static ssize_t amd64_inject_section_show(struct mem_ctl_info *mci, char *buf)
* *
* range: 0..3 * range: 0..3
*/ */
static ssize_t amd64_inject_section_store(struct mem_ctl_info *mci, static ssize_t amd64_inject_section_store(struct device *dev,
struct device_attribute *mattr,
const char *data, size_t count) const char *data, size_t count)
{ {
struct mem_ctl_info *mci = to_mci(dev);
struct amd64_pvt *pvt = mci->pvt_info; struct amd64_pvt *pvt = mci->pvt_info;
unsigned long value; unsigned long value;
int ret = 0; int ret = 0;
...@@ -33,8 +38,11 @@ static ssize_t amd64_inject_section_store(struct mem_ctl_info *mci, ...@@ -33,8 +38,11 @@ static ssize_t amd64_inject_section_store(struct mem_ctl_info *mci,
return ret; return ret;
} }
static ssize_t amd64_inject_word_show(struct mem_ctl_info *mci, char *buf) static ssize_t amd64_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; struct amd64_pvt *pvt = mci->pvt_info;
return sprintf(buf, "0x%x\n", pvt->injection.word); return sprintf(buf, "0x%x\n", pvt->injection.word);
} }
...@@ -45,9 +53,11 @@ static ssize_t amd64_inject_word_show(struct mem_ctl_info *mci, char *buf) ...@@ -45,9 +53,11 @@ static ssize_t amd64_inject_word_show(struct mem_ctl_info *mci, char *buf)
* *
* range: 0..8 * range: 0..8
*/ */
static ssize_t amd64_inject_word_store(struct mem_ctl_info *mci, static ssize_t amd64_inject_word_store(struct device *dev,
struct device_attribute *mattr,
const char *data, size_t count) const char *data, size_t count)
{ {
struct mem_ctl_info *mci = to_mci(dev);
struct amd64_pvt *pvt = mci->pvt_info; struct amd64_pvt *pvt = mci->pvt_info;
unsigned long value; unsigned long value;
int ret = 0; int ret = 0;
...@@ -66,8 +76,11 @@ static ssize_t amd64_inject_word_store(struct mem_ctl_info *mci, ...@@ -66,8 +76,11 @@ static ssize_t amd64_inject_word_store(struct mem_ctl_info *mci,
return ret; return ret;
} }
static ssize_t amd64_inject_ecc_vector_show(struct mem_ctl_info *mci, char *buf) static ssize_t amd64_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; struct amd64_pvt *pvt = mci->pvt_info;
return sprintf(buf, "0x%x\n", pvt->injection.bit_map); return sprintf(buf, "0x%x\n", pvt->injection.bit_map);
} }
...@@ -77,9 +90,11 @@ static ssize_t amd64_inject_ecc_vector_show(struct mem_ctl_info *mci, char *buf) ...@@ -77,9 +90,11 @@ static ssize_t amd64_inject_ecc_vector_show(struct mem_ctl_info *mci, char *buf)
* corresponding bit within the error injection word above. When used during a * 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. * DRAM ECC read, it holds the contents of the of the DRAM ECC bits.
*/ */
static ssize_t amd64_inject_ecc_vector_store(struct mem_ctl_info *mci, static ssize_t amd64_inject_ecc_vector_store(struct device *dev,
struct device_attribute *mattr,
const char *data, size_t count) const char *data, size_t count)
{ {
struct mem_ctl_info *mci = to_mci(dev);
struct amd64_pvt *pvt = mci->pvt_info; struct amd64_pvt *pvt = mci->pvt_info;
unsigned long value; unsigned long value;
int ret = 0; int ret = 0;
...@@ -103,9 +118,11 @@ static ssize_t amd64_inject_ecc_vector_store(struct mem_ctl_info *mci, ...@@ -103,9 +118,11 @@ static ssize_t amd64_inject_ecc_vector_store(struct mem_ctl_info *mci,
* Do a DRAM ECC read. Assemble staged values in the pvt area, format into * 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. * fields needed by the injection registers and read the NB Array Data Port.
*/ */
static ssize_t amd64_inject_read_store(struct mem_ctl_info *mci, static ssize_t amd64_inject_read_store(struct device *dev,
struct device_attribute *mattr,
const char *data, size_t count) const char *data, size_t count)
{ {
struct mem_ctl_info *mci = to_mci(dev);
struct amd64_pvt *pvt = mci->pvt_info; struct amd64_pvt *pvt = mci->pvt_info;
unsigned long value; unsigned long value;
u32 section, word_bits; u32 section, word_bits;
...@@ -125,7 +142,8 @@ static ssize_t amd64_inject_read_store(struct mem_ctl_info *mci, ...@@ -125,7 +142,8 @@ static ssize_t amd64_inject_read_store(struct mem_ctl_info *mci,
/* Issue 'word' and 'bit' along with the READ request */ /* Issue 'word' and 'bit' along with the READ request */
amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, word_bits); amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, word_bits);
debugf0("section=0x%x word_bits=0x%x\n", section, word_bits); edac_dbg(0, "section=0x%x word_bits=0x%x\n",
section, word_bits);
return count; return count;
} }
...@@ -136,9 +154,11 @@ static ssize_t amd64_inject_read_store(struct mem_ctl_info *mci, ...@@ -136,9 +154,11 @@ static ssize_t amd64_inject_read_store(struct mem_ctl_info *mci,
* Do a DRAM ECC write. Assemble staged values in the pvt area and format into * Do a DRAM ECC write. Assemble staged values in the pvt area and format into
* fields needed by the injection registers. * fields needed by the injection registers.
*/ */
static ssize_t amd64_inject_write_store(struct mem_ctl_info *mci, static ssize_t amd64_inject_write_store(struct device *dev,
struct device_attribute *mattr,
const char *data, size_t count) const char *data, size_t count)
{ {
struct mem_ctl_info *mci = to_mci(dev);
struct amd64_pvt *pvt = mci->pvt_info; struct amd64_pvt *pvt = mci->pvt_info;
unsigned long value; unsigned long value;
u32 section, word_bits; u32 section, word_bits;
...@@ -158,7 +178,8 @@ static ssize_t amd64_inject_write_store(struct mem_ctl_info *mci, ...@@ -158,7 +178,8 @@ static ssize_t amd64_inject_write_store(struct mem_ctl_info *mci,
/* Issue 'word' and 'bit' along with the READ request */ /* Issue 'word' and 'bit' along with the READ request */
amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, word_bits); amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, word_bits);
debugf0("section=0x%x word_bits=0x%x\n", section, word_bits); edac_dbg(0, "section=0x%x word_bits=0x%x\n",
section, word_bits);
return count; return count;
} }
...@@ -168,46 +189,47 @@ static ssize_t amd64_inject_write_store(struct mem_ctl_info *mci, ...@@ -168,46 +189,47 @@ static ssize_t amd64_inject_write_store(struct mem_ctl_info *mci,
/* /*
* update NUM_INJ_ATTRS in case you add new members * update NUM_INJ_ATTRS in case you add new members
*/ */
struct mcidev_sysfs_attribute amd64_inj_attrs[] = {
static DEVICE_ATTR(inject_section, S_IRUGO | S_IWUSR,
{ amd64_inject_section_show, amd64_inject_section_store);
.attr = { static DEVICE_ATTR(inject_word, S_IRUGO | S_IWUSR,
.name = "inject_section", amd64_inject_word_show, amd64_inject_word_store);
.mode = (S_IRUGO | S_IWUSR) static DEVICE_ATTR(inject_ecc_vector, S_IRUGO | S_IWUSR,
}, amd64_inject_ecc_vector_show, amd64_inject_ecc_vector_store);
.show = amd64_inject_section_show, static DEVICE_ATTR(inject_write, S_IRUGO | S_IWUSR,
.store = amd64_inject_section_store, NULL, amd64_inject_write_store);
}, static DEVICE_ATTR(inject_read, S_IRUGO | S_IWUSR,
{ NULL, amd64_inject_read_store);
.attr = {
.name = "inject_word",
.mode = (S_IRUGO | S_IWUSR) int amd64_create_sysfs_inject_files(struct mem_ctl_info *mci)
}, {
.show = amd64_inject_word_show, int rc;
.store = amd64_inject_word_store,
}, rc = device_create_file(&mci->dev, &dev_attr_inject_section);
{ if (rc < 0)
.attr = { return rc;
.name = "inject_ecc_vector", rc = device_create_file(&mci->dev, &dev_attr_inject_word);
.mode = (S_IRUGO | S_IWUSR) if (rc < 0)
}, return rc;
.show = amd64_inject_ecc_vector_show, rc = device_create_file(&mci->dev, &dev_attr_inject_ecc_vector);
.store = amd64_inject_ecc_vector_store, if (rc < 0)
}, return rc;
{ rc = device_create_file(&mci->dev, &dev_attr_inject_write);
.attr = { if (rc < 0)
.name = "inject_write", return rc;
.mode = (S_IRUGO | S_IWUSR) rc = device_create_file(&mci->dev, &dev_attr_inject_read);
}, if (rc < 0)
.show = NULL, return rc;
.store = amd64_inject_write_store,
}, return 0;
{ }
.attr = {
.name = "inject_read", void amd64_remove_sysfs_inject_files(struct mem_ctl_info *mci)
.mode = (S_IRUGO | S_IWUSR) {
}, device_remove_file(&mci->dev, &dev_attr_inject_section);
.show = NULL, device_remove_file(&mci->dev, &dev_attr_inject_word);
.store = amd64_inject_read_store, device_remove_file(&mci->dev, &dev_attr_inject_ecc_vector);
}, device_remove_file(&mci->dev, &dev_attr_inject_write);
}; device_remove_file(&mci->dev, &dev_attr_inject_read);
}
...@@ -105,7 +105,7 @@ static void amd76x_get_error_info(struct mem_ctl_info *mci, ...@@ -105,7 +105,7 @@ static void amd76x_get_error_info(struct mem_ctl_info *mci,
{ {
struct pci_dev *pdev; struct pci_dev *pdev;
pdev = to_pci_dev(mci->dev); pdev = to_pci_dev(mci->pdev);
pci_read_config_dword(pdev, AMD76X_ECC_MODE_STATUS, pci_read_config_dword(pdev, AMD76X_ECC_MODE_STATUS,
&info->ecc_mode_status); &info->ecc_mode_status);
...@@ -145,10 +145,10 @@ static int amd76x_process_error_info(struct mem_ctl_info *mci, ...@@ -145,10 +145,10 @@ static int amd76x_process_error_info(struct mem_ctl_info *mci,
if (handle_errors) { if (handle_errors) {
row = (info->ecc_mode_status >> 4) & 0xf; row = (info->ecc_mode_status >> 4) & 0xf;
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1,
mci->csrows[row].first_page, 0, 0, mci->csrows[row]->first_page, 0, 0,
row, 0, -1, row, 0, -1,
mci->ctl_name, "", NULL); mci->ctl_name, "");
} }
} }
...@@ -160,10 +160,10 @@ static int amd76x_process_error_info(struct mem_ctl_info *mci, ...@@ -160,10 +160,10 @@ static int amd76x_process_error_info(struct mem_ctl_info *mci,
if (handle_errors) { if (handle_errors) {
row = info->ecc_mode_status & 0xf; row = info->ecc_mode_status & 0xf;
edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, 1,
mci->csrows[row].first_page, 0, 0, mci->csrows[row]->first_page, 0, 0,
row, 0, -1, row, 0, -1,
mci->ctl_name, "", NULL); mci->ctl_name, "");
} }
} }
...@@ -180,7 +180,7 @@ static int amd76x_process_error_info(struct mem_ctl_info *mci, ...@@ -180,7 +180,7 @@ static int amd76x_process_error_info(struct mem_ctl_info *mci,
static void amd76x_check(struct mem_ctl_info *mci) static void amd76x_check(struct mem_ctl_info *mci)
{ {
struct amd76x_error_info info; struct amd76x_error_info info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
amd76x_get_error_info(mci, &info); amd76x_get_error_info(mci, &info);
amd76x_process_error_info(mci, &info, 1); amd76x_process_error_info(mci, &info, 1);
} }
...@@ -194,8 +194,8 @@ static void amd76x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -194,8 +194,8 @@ static void amd76x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
int index; int index;
for (index = 0; index < mci->nr_csrows; index++) { for (index = 0; index < mci->nr_csrows; index++) {
csrow = &mci->csrows[index]; csrow = mci->csrows[index];
dimm = csrow->channels[0].dimm; dimm = csrow->channels[0]->dimm;
/* find the DRAM Chip Select Base address and mask */ /* find the DRAM Chip Select Base address and mask */
pci_read_config_dword(pdev, pci_read_config_dword(pdev,
...@@ -241,7 +241,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -241,7 +241,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx)
u32 ems_mode; u32 ems_mode;
struct amd76x_error_info discard; struct amd76x_error_info discard;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
pci_read_config_dword(pdev, AMD76X_ECC_MODE_STATUS, &ems); pci_read_config_dword(pdev, AMD76X_ECC_MODE_STATUS, &ems);
ems_mode = (ems >> 10) & 0x3; ems_mode = (ems >> 10) & 0x3;
...@@ -256,8 +256,8 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -256,8 +256,8 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx)
if (mci == NULL) if (mci == NULL)
return -ENOMEM; return -ENOMEM;
debugf0("%s(): mci = %p\n", __func__, mci); edac_dbg(0, "mci = %p\n", mci);
mci->dev = &pdev->dev; mci->pdev = &pdev->dev;
mci->mtype_cap = MEM_FLAG_RDDR; mci->mtype_cap = MEM_FLAG_RDDR;
mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_EC | EDAC_FLAG_SECDED; mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_EC | EDAC_FLAG_SECDED;
mci->edac_cap = ems_mode ? mci->edac_cap = ems_mode ?
...@@ -276,7 +276,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -276,7 +276,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx)
* type of memory controller. The ID is therefore hardcoded to 0. * type of memory controller. The ID is therefore hardcoded to 0.
*/ */
if (edac_mc_add_mc(mci)) { if (edac_mc_add_mc(mci)) {
debugf3("%s(): failed edac_mc_add_mc()\n", __func__); edac_dbg(3, "failed edac_mc_add_mc()\n");
goto fail; goto fail;
} }
...@@ -292,7 +292,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -292,7 +292,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx)
} }
/* get this far and it's successful */ /* get this far and it's successful */
debugf3("%s(): success\n", __func__); edac_dbg(3, "success\n");
return 0; return 0;
fail: fail:
...@@ -304,7 +304,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -304,7 +304,7 @@ static int amd76x_probe1(struct pci_dev *pdev, int dev_idx)
static int __devinit amd76x_init_one(struct pci_dev *pdev, static int __devinit amd76x_init_one(struct pci_dev *pdev,
const struct pci_device_id *ent) const struct pci_device_id *ent)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
/* don't need to call pci_enable_device() */ /* don't need to call pci_enable_device() */
return amd76x_probe1(pdev, ent->driver_data); return amd76x_probe1(pdev, ent->driver_data);
...@@ -322,7 +322,7 @@ static void __devexit amd76x_remove_one(struct pci_dev *pdev) ...@@ -322,7 +322,7 @@ static void __devexit amd76x_remove_one(struct pci_dev *pdev)
{ {
struct mem_ctl_info *mci; struct mem_ctl_info *mci;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
if (amd76x_pci) if (amd76x_pci)
edac_pci_release_generic_ctl(amd76x_pci); edac_pci_release_generic_ctl(amd76x_pci);
......
...@@ -33,10 +33,10 @@ struct cell_edac_priv ...@@ -33,10 +33,10 @@ struct cell_edac_priv
static void cell_edac_count_ce(struct mem_ctl_info *mci, int chan, u64 ar) static void cell_edac_count_ce(struct mem_ctl_info *mci, int chan, u64 ar)
{ {
struct cell_edac_priv *priv = mci->pvt_info; struct cell_edac_priv *priv = mci->pvt_info;
struct csrow_info *csrow = &mci->csrows[0]; struct csrow_info *csrow = mci->csrows[0];
unsigned long address, pfn, offset, syndrome; unsigned long address, pfn, offset, syndrome;
dev_dbg(mci->dev, "ECC CE err on node %d, channel %d, ar = 0x%016llx\n", dev_dbg(mci->pdev, "ECC CE err on node %d, channel %d, ar = 0x%016llx\n",
priv->node, chan, ar); priv->node, chan, ar);
/* Address decoding is likely a bit bogus, to dbl check */ /* Address decoding is likely a bit bogus, to dbl check */
...@@ -48,18 +48,18 @@ static void cell_edac_count_ce(struct mem_ctl_info *mci, int chan, u64 ar) ...@@ -48,18 +48,18 @@ static void cell_edac_count_ce(struct mem_ctl_info *mci, int chan, u64 ar)
syndrome = (ar & 0x000000001fe00000ul) >> 21; syndrome = (ar & 0x000000001fe00000ul) >> 21;
/* TODO: Decoding of the error address */ /* TODO: Decoding of the error address */
edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, 1,
csrow->first_page + pfn, offset, syndrome, csrow->first_page + pfn, offset, syndrome,
0, chan, -1, "", "", NULL); 0, chan, -1, "", "");
} }
static void cell_edac_count_ue(struct mem_ctl_info *mci, int chan, u64 ar) static void cell_edac_count_ue(struct mem_ctl_info *mci, int chan, u64 ar)
{ {
struct cell_edac_priv *priv = mci->pvt_info; struct cell_edac_priv *priv = mci->pvt_info;
struct csrow_info *csrow = &mci->csrows[0]; struct csrow_info *csrow = mci->csrows[0];
unsigned long address, pfn, offset; unsigned long address, pfn, offset;
dev_dbg(mci->dev, "ECC UE err on node %d, channel %d, ar = 0x%016llx\n", dev_dbg(mci->pdev, "ECC UE err on node %d, channel %d, ar = 0x%016llx\n",
priv->node, chan, ar); priv->node, chan, ar);
/* Address decoding is likely a bit bogus, to dbl check */ /* Address decoding is likely a bit bogus, to dbl check */
...@@ -70,9 +70,9 @@ static void cell_edac_count_ue(struct mem_ctl_info *mci, int chan, u64 ar) ...@@ -70,9 +70,9 @@ static void cell_edac_count_ue(struct mem_ctl_info *mci, int chan, u64 ar)
offset = address & ~PAGE_MASK; offset = address & ~PAGE_MASK;
/* TODO: Decoding of the error address */ /* TODO: Decoding of the error address */
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1,
csrow->first_page + pfn, offset, 0, csrow->first_page + pfn, offset, 0,
0, chan, -1, "", "", NULL); 0, chan, -1, "", "");
} }
static void cell_edac_check(struct mem_ctl_info *mci) static void cell_edac_check(struct mem_ctl_info *mci)
...@@ -83,7 +83,7 @@ static void cell_edac_check(struct mem_ctl_info *mci) ...@@ -83,7 +83,7 @@ static void cell_edac_check(struct mem_ctl_info *mci)
fir = in_be64(&priv->regs->mic_fir); fir = in_be64(&priv->regs->mic_fir);
#ifdef DEBUG #ifdef DEBUG
if (fir != priv->prev_fir) { if (fir != priv->prev_fir) {
dev_dbg(mci->dev, "fir change : 0x%016lx\n", fir); dev_dbg(mci->pdev, "fir change : 0x%016lx\n", fir);
priv->prev_fir = fir; priv->prev_fir = fir;
} }
#endif #endif
...@@ -119,14 +119,14 @@ static void cell_edac_check(struct mem_ctl_info *mci) ...@@ -119,14 +119,14 @@ static void cell_edac_check(struct mem_ctl_info *mci)
mb(); /* sync up */ mb(); /* sync up */
#ifdef DEBUG #ifdef DEBUG
fir = in_be64(&priv->regs->mic_fir); fir = in_be64(&priv->regs->mic_fir);
dev_dbg(mci->dev, "fir clear : 0x%016lx\n", fir); dev_dbg(mci->pdev, "fir clear : 0x%016lx\n", fir);
#endif #endif
} }
} }
static void __devinit cell_edac_init_csrows(struct mem_ctl_info *mci) static void __devinit cell_edac_init_csrows(struct mem_ctl_info *mci)
{ {
struct csrow_info *csrow = &mci->csrows[0]; struct csrow_info *csrow = mci->csrows[0];
struct dimm_info *dimm; struct dimm_info *dimm;
struct cell_edac_priv *priv = mci->pvt_info; struct cell_edac_priv *priv = mci->pvt_info;
struct device_node *np; struct device_node *np;
...@@ -150,12 +150,12 @@ static void __devinit cell_edac_init_csrows(struct mem_ctl_info *mci) ...@@ -150,12 +150,12 @@ static void __devinit cell_edac_init_csrows(struct mem_ctl_info *mci)
csrow->last_page = csrow->first_page + nr_pages - 1; csrow->last_page = csrow->first_page + nr_pages - 1;
for (j = 0; j < csrow->nr_channels; j++) { for (j = 0; j < csrow->nr_channels; j++) {
dimm = csrow->channels[j].dimm; dimm = csrow->channels[j]->dimm;
dimm->mtype = MEM_XDR; dimm->mtype = MEM_XDR;
dimm->edac_mode = EDAC_SECDED; dimm->edac_mode = EDAC_SECDED;
dimm->nr_pages = nr_pages / csrow->nr_channels; dimm->nr_pages = nr_pages / csrow->nr_channels;
} }
dev_dbg(mci->dev, dev_dbg(mci->pdev,
"Initialized on node %d, chanmask=0x%x," "Initialized on node %d, chanmask=0x%x,"
" first_page=0x%lx, nr_pages=0x%x\n", " first_page=0x%lx, nr_pages=0x%x\n",
priv->node, priv->chanmask, priv->node, priv->chanmask,
...@@ -212,7 +212,7 @@ static int __devinit cell_edac_probe(struct platform_device *pdev) ...@@ -212,7 +212,7 @@ static int __devinit cell_edac_probe(struct platform_device *pdev)
priv->regs = regs; priv->regs = regs;
priv->node = pdev->id; priv->node = pdev->id;
priv->chanmask = chanmask; priv->chanmask = chanmask;
mci->dev = &pdev->dev; mci->pdev = &pdev->dev;
mci->mtype_cap = MEM_FLAG_XDR; mci->mtype_cap = MEM_FLAG_XDR;
mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_EC | EDAC_FLAG_SECDED; mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_EC | EDAC_FLAG_SECDED;
mci->edac_cap = EDAC_FLAG_EC | EDAC_FLAG_SECDED; mci->edac_cap = EDAC_FLAG_EC | EDAC_FLAG_SECDED;
......
...@@ -316,13 +316,12 @@ static void get_total_mem(struct cpc925_mc_pdata *pdata) ...@@ -316,13 +316,12 @@ static void get_total_mem(struct cpc925_mc_pdata *pdata)
reg += aw; reg += aw;
size = of_read_number(reg, sw); size = of_read_number(reg, sw);
reg += sw; reg += sw;
debugf1("%s: start 0x%lx, size 0x%lx\n", __func__, edac_dbg(1, "start 0x%lx, size 0x%lx\n", start, size);
start, size);
pdata->total_mem += size; pdata->total_mem += size;
} while (reg < reg_end); } while (reg < reg_end);
of_node_put(np); of_node_put(np);
debugf0("%s: total_mem 0x%lx\n", __func__, pdata->total_mem); edac_dbg(0, "total_mem 0x%lx\n", pdata->total_mem);
} }
static void cpc925_init_csrows(struct mem_ctl_info *mci) static void cpc925_init_csrows(struct mem_ctl_info *mci)
...@@ -330,8 +329,9 @@ static void cpc925_init_csrows(struct mem_ctl_info *mci) ...@@ -330,8 +329,9 @@ static void cpc925_init_csrows(struct mem_ctl_info *mci)
struct cpc925_mc_pdata *pdata = mci->pvt_info; struct cpc925_mc_pdata *pdata = mci->pvt_info;
struct csrow_info *csrow; struct csrow_info *csrow;
struct dimm_info *dimm; struct dimm_info *dimm;
enum dev_type dtype;
int index, j; int index, j;
u32 mbmr, mbbar, bba; u32 mbmr, mbbar, bba, grain;
unsigned long row_size, nr_pages, last_nr_pages = 0; unsigned long row_size, nr_pages, last_nr_pages = 0;
get_total_mem(pdata); get_total_mem(pdata);
...@@ -347,7 +347,7 @@ static void cpc925_init_csrows(struct mem_ctl_info *mci) ...@@ -347,7 +347,7 @@ static void cpc925_init_csrows(struct mem_ctl_info *mci)
if (bba == 0) if (bba == 0)
continue; /* not populated */ continue; /* not populated */
csrow = &mci->csrows[index]; csrow = mci->csrows[index];
row_size = bba * (1UL << 28); /* 256M */ row_size = bba * (1UL << 28); /* 256M */
csrow->first_page = last_nr_pages; csrow->first_page = last_nr_pages;
...@@ -355,37 +355,36 @@ static void cpc925_init_csrows(struct mem_ctl_info *mci) ...@@ -355,37 +355,36 @@ static void cpc925_init_csrows(struct mem_ctl_info *mci)
csrow->last_page = csrow->first_page + nr_pages - 1; csrow->last_page = csrow->first_page + nr_pages - 1;
last_nr_pages = csrow->last_page + 1; last_nr_pages = csrow->last_page + 1;
for (j = 0; j < csrow->nr_channels; j++) {
dimm = csrow->channels[j].dimm;
dimm->nr_pages = nr_pages / csrow->nr_channels;
dimm->mtype = MEM_RDDR;
dimm->edac_mode = EDAC_SECDED;
switch (csrow->nr_channels) { switch (csrow->nr_channels) {
case 1: /* Single channel */ case 1: /* Single channel */
dimm->grain = 32; /* four-beat burst of 32 bytes */ grain = 32; /* four-beat burst of 32 bytes */
break; break;
case 2: /* Dual channel */ case 2: /* Dual channel */
default: default:
dimm->grain = 64; /* four-beat burst of 64 bytes */ grain = 64; /* four-beat burst of 64 bytes */
break; break;
} }
switch ((mbmr & MBMR_MODE_MASK) >> MBMR_MODE_SHIFT) { switch ((mbmr & MBMR_MODE_MASK) >> MBMR_MODE_SHIFT) {
case 6: /* 0110, no way to differentiate X8 VS X16 */ case 6: /* 0110, no way to differentiate X8 VS X16 */
case 5: /* 0101 */ case 5: /* 0101 */
case 8: /* 1000 */ case 8: /* 1000 */
dimm->dtype = DEV_X16; dtype = DEV_X16;
break; break;
case 7: /* 0111 */ case 7: /* 0111 */
case 9: /* 1001 */ case 9: /* 1001 */
dimm->dtype = DEV_X8; dtype = DEV_X8;
break; break;
default: default:
dimm->dtype = DEV_UNKNOWN; dtype = DEV_UNKNOWN;
break; break;
} }
for (j = 0; j < csrow->nr_channels; j++) {
dimm = csrow->channels[j]->dimm;
dimm->nr_pages = nr_pages / csrow->nr_channels;
dimm->mtype = MEM_RDDR;
dimm->edac_mode = EDAC_SECDED;
dimm->grain = grain;
dimm->dtype = dtype;
} }
} }
} }
...@@ -463,7 +462,7 @@ static void cpc925_mc_get_pfn(struct mem_ctl_info *mci, u32 mear, ...@@ -463,7 +462,7 @@ static void cpc925_mc_get_pfn(struct mem_ctl_info *mci, u32 mear,
*csrow = rank; *csrow = rank;
#ifdef CONFIG_EDAC_DEBUG #ifdef CONFIG_EDAC_DEBUG
if (mci->csrows[rank].first_page == 0) { if (mci->csrows[rank]->first_page == 0) {
cpc925_mc_printk(mci, KERN_ERR, "ECC occurs in a " cpc925_mc_printk(mci, KERN_ERR, "ECC occurs in a "
"non-populated csrow, broken hardware?\n"); "non-populated csrow, broken hardware?\n");
return; return;
...@@ -471,7 +470,7 @@ static void cpc925_mc_get_pfn(struct mem_ctl_info *mci, u32 mear, ...@@ -471,7 +470,7 @@ static void cpc925_mc_get_pfn(struct mem_ctl_info *mci, u32 mear,
#endif #endif
/* Revert csrow number */ /* Revert csrow number */
pa = mci->csrows[rank].first_page << PAGE_SHIFT; pa = mci->csrows[rank]->first_page << PAGE_SHIFT;
/* Revert column address */ /* Revert column address */
col += bcnt; col += bcnt;
...@@ -512,7 +511,7 @@ static void cpc925_mc_get_pfn(struct mem_ctl_info *mci, u32 mear, ...@@ -512,7 +511,7 @@ static void cpc925_mc_get_pfn(struct mem_ctl_info *mci, u32 mear,
*offset = pa & (PAGE_SIZE - 1); *offset = pa & (PAGE_SIZE - 1);
*pfn = pa >> PAGE_SHIFT; *pfn = pa >> PAGE_SHIFT;
debugf0("%s: ECC physical address 0x%lx\n", __func__, pa); edac_dbg(0, "ECC physical address 0x%lx\n", pa);
} }
static int cpc925_mc_find_channel(struct mem_ctl_info *mci, u16 syndrome) static int cpc925_mc_find_channel(struct mem_ctl_info *mci, u16 syndrome)
...@@ -555,18 +554,18 @@ static void cpc925_mc_check(struct mem_ctl_info *mci) ...@@ -555,18 +554,18 @@ static void cpc925_mc_check(struct mem_ctl_info *mci)
if (apiexcp & CECC_EXCP_DETECTED) { if (apiexcp & CECC_EXCP_DETECTED) {
cpc925_mc_printk(mci, KERN_INFO, "DRAM CECC Fault\n"); cpc925_mc_printk(mci, KERN_INFO, "DRAM CECC Fault\n");
channel = cpc925_mc_find_channel(mci, syndrome); channel = cpc925_mc_find_channel(mci, syndrome);
edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, 1,
pfn, offset, syndrome, pfn, offset, syndrome,
csrow, channel, -1, csrow, channel, -1,
mci->ctl_name, "", NULL); mci->ctl_name, "");
} }
if (apiexcp & UECC_EXCP_DETECTED) { if (apiexcp & UECC_EXCP_DETECTED) {
cpc925_mc_printk(mci, KERN_INFO, "DRAM UECC Fault\n"); cpc925_mc_printk(mci, KERN_INFO, "DRAM UECC Fault\n");
edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, 1,
pfn, offset, 0, pfn, offset, 0,
csrow, -1, -1, csrow, -1, -1,
mci->ctl_name, "", NULL); mci->ctl_name, "");
} }
cpc925_mc_printk(mci, KERN_INFO, "Dump registers:\n"); cpc925_mc_printk(mci, KERN_INFO, "Dump registers:\n");
...@@ -852,8 +851,8 @@ static void cpc925_add_edac_devices(void __iomem *vbase) ...@@ -852,8 +851,8 @@ static void cpc925_add_edac_devices(void __iomem *vbase)
goto err2; goto err2;
} }
debugf0("%s: Successfully added edac device for %s\n", edac_dbg(0, "Successfully added edac device for %s\n",
__func__, dev_info->ctl_name); dev_info->ctl_name);
continue; continue;
...@@ -884,8 +883,8 @@ static void cpc925_del_edac_devices(void) ...@@ -884,8 +883,8 @@ static void cpc925_del_edac_devices(void)
if (dev_info->exit) if (dev_info->exit)
dev_info->exit(dev_info); dev_info->exit(dev_info);
debugf0("%s: Successfully deleted edac device for %s\n", edac_dbg(0, "Successfully deleted edac device for %s\n",
__func__, dev_info->ctl_name); dev_info->ctl_name);
} }
} }
...@@ -900,7 +899,7 @@ static int cpc925_get_sdram_scrub_rate(struct mem_ctl_info *mci) ...@@ -900,7 +899,7 @@ static int cpc925_get_sdram_scrub_rate(struct mem_ctl_info *mci)
mscr = __raw_readl(pdata->vbase + REG_MSCR_OFFSET); mscr = __raw_readl(pdata->vbase + REG_MSCR_OFFSET);
si = (mscr & MSCR_SI_MASK) >> MSCR_SI_SHIFT; si = (mscr & MSCR_SI_MASK) >> MSCR_SI_SHIFT;
debugf0("%s, Mem Scrub Ctrl Register 0x%x\n", __func__, mscr); edac_dbg(0, "Mem Scrub Ctrl Register 0x%x\n", mscr);
if (((mscr & MSCR_SCRUB_MOD_MASK) != MSCR_BACKGR_SCRUB) || if (((mscr & MSCR_SCRUB_MOD_MASK) != MSCR_BACKGR_SCRUB) ||
(si == 0)) { (si == 0)) {
...@@ -928,8 +927,7 @@ static int cpc925_mc_get_channels(void __iomem *vbase) ...@@ -928,8 +927,7 @@ static int cpc925_mc_get_channels(void __iomem *vbase)
((mbcr & MBCR_64BITBUS_MASK) == 0)) ((mbcr & MBCR_64BITBUS_MASK) == 0))
dual = 1; dual = 1;
debugf0("%s: %s channel\n", __func__, edac_dbg(0, "%s channel\n", (dual > 0) ? "Dual" : "Single");
(dual > 0) ? "Dual" : "Single");
return dual; return dual;
} }
...@@ -944,7 +942,7 @@ static int __devinit cpc925_probe(struct platform_device *pdev) ...@@ -944,7 +942,7 @@ static int __devinit cpc925_probe(struct platform_device *pdev)
struct resource *r; struct resource *r;
int res = 0, nr_channels; int res = 0, nr_channels;
debugf0("%s: %s platform device found!\n", __func__, pdev->name); edac_dbg(0, "%s platform device found!\n", pdev->name);
if (!devres_open_group(&pdev->dev, cpc925_probe, GFP_KERNEL)) { if (!devres_open_group(&pdev->dev, cpc925_probe, GFP_KERNEL)) {
res = -ENOMEM; res = -ENOMEM;
...@@ -995,7 +993,7 @@ static int __devinit cpc925_probe(struct platform_device *pdev) ...@@ -995,7 +993,7 @@ static int __devinit cpc925_probe(struct platform_device *pdev)
pdata->edac_idx = edac_mc_idx++; pdata->edac_idx = edac_mc_idx++;
pdata->name = pdev->name; pdata->name = pdev->name;
mci->dev = &pdev->dev; mci->pdev = &pdev->dev;
platform_set_drvdata(pdev, mci); platform_set_drvdata(pdev, mci);
mci->dev_name = dev_name(&pdev->dev); mci->dev_name = dev_name(&pdev->dev);
mci->mtype_cap = MEM_FLAG_RDDR | MEM_FLAG_DDR; mci->mtype_cap = MEM_FLAG_RDDR | MEM_FLAG_DDR;
...@@ -1026,7 +1024,7 @@ static int __devinit cpc925_probe(struct platform_device *pdev) ...@@ -1026,7 +1024,7 @@ static int __devinit cpc925_probe(struct platform_device *pdev)
cpc925_add_edac_devices(vbase); cpc925_add_edac_devices(vbase);
/* get this far and it's successful */ /* get this far and it's successful */
debugf0("%s: success\n", __func__); edac_dbg(0, "success\n");
res = 0; res = 0;
goto out; goto out;
......
...@@ -309,7 +309,7 @@ static unsigned long ctl_page_to_phys(struct mem_ctl_info *mci, ...@@ -309,7 +309,7 @@ static unsigned long ctl_page_to_phys(struct mem_ctl_info *mci,
u32 remap; u32 remap;
struct e752x_pvt *pvt = (struct e752x_pvt *)mci->pvt_info; struct e752x_pvt *pvt = (struct e752x_pvt *)mci->pvt_info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
if (page < pvt->tolm) if (page < pvt->tolm)
return page; return page;
...@@ -335,7 +335,7 @@ static void do_process_ce(struct mem_ctl_info *mci, u16 error_one, ...@@ -335,7 +335,7 @@ static void do_process_ce(struct mem_ctl_info *mci, u16 error_one,
int i; int i;
struct e752x_pvt *pvt = (struct e752x_pvt *)mci->pvt_info; struct e752x_pvt *pvt = (struct e752x_pvt *)mci->pvt_info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
/* convert the addr to 4k page */ /* convert the addr to 4k page */
page = sec1_add >> (PAGE_SHIFT - 4); page = sec1_add >> (PAGE_SHIFT - 4);
...@@ -371,10 +371,10 @@ static void do_process_ce(struct mem_ctl_info *mci, u16 error_one, ...@@ -371,10 +371,10 @@ static void do_process_ce(struct mem_ctl_info *mci, u16 error_one,
channel = !(error_one & 1); channel = !(error_one & 1);
/* e752x mc reads 34:6 of the DRAM linear address */ /* e752x mc reads 34:6 of the DRAM linear address */
edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, 1,
page, offset_in_page(sec1_add << 4), sec1_syndrome, page, offset_in_page(sec1_add << 4), sec1_syndrome,
row, channel, -1, row, channel, -1,
"e752x CE", "", NULL); "e752x CE", "");
} }
static inline void process_ce(struct mem_ctl_info *mci, u16 error_one, static inline void process_ce(struct mem_ctl_info *mci, u16 error_one,
...@@ -394,7 +394,7 @@ static void do_process_ue(struct mem_ctl_info *mci, u16 error_one, ...@@ -394,7 +394,7 @@ static void do_process_ue(struct mem_ctl_info *mci, u16 error_one,
int row; int row;
struct e752x_pvt *pvt = (struct e752x_pvt *)mci->pvt_info; struct e752x_pvt *pvt = (struct e752x_pvt *)mci->pvt_info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
if (error_one & 0x0202) { if (error_one & 0x0202) {
error_2b = ded_add; error_2b = ded_add;
...@@ -408,11 +408,11 @@ static void do_process_ue(struct mem_ctl_info *mci, u16 error_one, ...@@ -408,11 +408,11 @@ static void do_process_ue(struct mem_ctl_info *mci, u16 error_one,
edac_mc_find_csrow_by_page(mci, block_page); edac_mc_find_csrow_by_page(mci, block_page);
/* e752x mc reads 34:6 of the DRAM linear address */ /* e752x mc reads 34:6 of the DRAM linear address */
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1,
block_page, block_page,
offset_in_page(error_2b << 4), 0, offset_in_page(error_2b << 4), 0,
row, -1, -1, row, -1, -1,
"e752x UE from Read", "", NULL); "e752x UE from Read", "");
} }
if (error_one & 0x0404) { if (error_one & 0x0404) {
...@@ -427,11 +427,11 @@ static void do_process_ue(struct mem_ctl_info *mci, u16 error_one, ...@@ -427,11 +427,11 @@ static void do_process_ue(struct mem_ctl_info *mci, u16 error_one,
edac_mc_find_csrow_by_page(mci, block_page); edac_mc_find_csrow_by_page(mci, block_page);
/* e752x mc reads 34:6 of the DRAM linear address */ /* e752x mc reads 34:6 of the DRAM linear address */
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1,
block_page, block_page,
offset_in_page(error_2b << 4), 0, offset_in_page(error_2b << 4), 0,
row, -1, -1, row, -1, -1,
"e752x UE from Scruber", "", NULL); "e752x UE from Scruber", "");
} }
} }
...@@ -453,10 +453,10 @@ static inline void process_ue_no_info_wr(struct mem_ctl_info *mci, ...@@ -453,10 +453,10 @@ static inline void process_ue_no_info_wr(struct mem_ctl_info *mci,
if (!handle_error) if (!handle_error)
return; return;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 0, 0, 0, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1, 0, 0, 0,
-1, -1, -1, -1, -1, -1,
"e752x UE log memory write", "", NULL); "e752x UE log memory write", "");
} }
static void do_process_ded_retry(struct mem_ctl_info *mci, u16 error, static void do_process_ded_retry(struct mem_ctl_info *mci, u16 error,
...@@ -982,7 +982,7 @@ static void e752x_check(struct mem_ctl_info *mci) ...@@ -982,7 +982,7 @@ static void e752x_check(struct mem_ctl_info *mci)
{ {
struct e752x_error_info info; struct e752x_error_info info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
e752x_get_error_info(mci, &info); e752x_get_error_info(mci, &info);
e752x_process_error_info(mci, &info, 1); e752x_process_error_info(mci, &info, 1);
} }
...@@ -1069,6 +1069,7 @@ static void e752x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -1069,6 +1069,7 @@ static void e752x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
u16 ddrcsr) u16 ddrcsr)
{ {
struct csrow_info *csrow; struct csrow_info *csrow;
enum edac_type edac_mode;
unsigned long last_cumul_size; unsigned long last_cumul_size;
int index, mem_dev, drc_chan; int index, mem_dev, drc_chan;
int drc_drbg; /* DRB granularity 0=64mb, 1=128mb */ int drc_drbg; /* DRB granularity 0=64mb, 1=128mb */
...@@ -1095,14 +1096,13 @@ static void e752x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -1095,14 +1096,13 @@ static void e752x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
for (last_cumul_size = index = 0; index < mci->nr_csrows; index++) { for (last_cumul_size = index = 0; index < mci->nr_csrows; index++) {
/* mem_dev 0=x8, 1=x4 */ /* mem_dev 0=x8, 1=x4 */
mem_dev = (dra >> (index * 4 + 2)) & 0x3; mem_dev = (dra >> (index * 4 + 2)) & 0x3;
csrow = &mci->csrows[remap_csrow_index(mci, index)]; csrow = mci->csrows[remap_csrow_index(mci, index)];
mem_dev = (mem_dev == 2); mem_dev = (mem_dev == 2);
pci_read_config_byte(pdev, E752X_DRB + index, &value); pci_read_config_byte(pdev, E752X_DRB + index, &value);
/* convert a 128 or 64 MiB DRB to a page size. */ /* convert a 128 or 64 MiB DRB to a page size. */
cumul_size = value << (25 + drc_drbg - PAGE_SHIFT); cumul_size = value << (25 + drc_drbg - PAGE_SHIFT);
debugf3("%s(): (%d) cumul_size 0x%x\n", __func__, index, edac_dbg(3, "(%d) cumul_size 0x%x\n", index, cumul_size);
cumul_size);
if (cumul_size == last_cumul_size) if (cumul_size == last_cumul_size)
continue; /* not populated */ continue; /* not populated */
...@@ -1111,29 +1111,29 @@ static void e752x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -1111,29 +1111,29 @@ static void e752x_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
nr_pages = cumul_size - last_cumul_size; nr_pages = cumul_size - last_cumul_size;
last_cumul_size = cumul_size; last_cumul_size = cumul_size;
for (i = 0; i < csrow->nr_channels; i++) {
struct dimm_info *dimm = csrow->channels[i].dimm;
debugf3("Initializing rank at (%i,%i)\n", index, i);
dimm->nr_pages = nr_pages / csrow->nr_channels;
dimm->grain = 1 << 12; /* 4KiB - resolution of CELOG */
dimm->mtype = MEM_RDDR; /* only one type supported */
dimm->dtype = mem_dev ? DEV_X4 : DEV_X8;
/* /*
* if single channel or x8 devices then SECDED * if single channel or x8 devices then SECDED
* if dual channel and x4 then S4ECD4ED * if dual channel and x4 then S4ECD4ED
*/ */
if (drc_ddim) { if (drc_ddim) {
if (drc_chan && mem_dev) { if (drc_chan && mem_dev) {
dimm->edac_mode = EDAC_S4ECD4ED; edac_mode = EDAC_S4ECD4ED;
mci->edac_cap |= EDAC_FLAG_S4ECD4ED; mci->edac_cap |= EDAC_FLAG_S4ECD4ED;
} else { } else {
dimm->edac_mode = EDAC_SECDED; edac_mode = EDAC_SECDED;
mci->edac_cap |= EDAC_FLAG_SECDED; mci->edac_cap |= EDAC_FLAG_SECDED;
} }
} else } else
dimm->edac_mode = EDAC_NONE; edac_mode = EDAC_NONE;
for (i = 0; i < csrow->nr_channels; i++) {
struct dimm_info *dimm = csrow->channels[i]->dimm;
edac_dbg(3, "Initializing rank at (%i,%i)\n", index, i);
dimm->nr_pages = nr_pages / csrow->nr_channels;
dimm->grain = 1 << 12; /* 4KiB - resolution of CELOG */
dimm->mtype = MEM_RDDR; /* only one type supported */
dimm->dtype = mem_dev ? DEV_X4 : DEV_X8;
dimm->edac_mode = edac_mode;
} }
} }
} }
...@@ -1269,8 +1269,8 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1269,8 +1269,8 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
int drc_chan; /* Number of channels 0=1chan,1=2chan */ int drc_chan; /* Number of channels 0=1chan,1=2chan */
struct e752x_error_info discard; struct e752x_error_info discard;
debugf0("%s(): mci\n", __func__); edac_dbg(0, "mci\n");
debugf0("Starting Probe1\n"); edac_dbg(0, "Starting Probe1\n");
/* check to see if device 0 function 1 is enabled; if it isn't, we /* check to see if device 0 function 1 is enabled; if it isn't, we
* assume the BIOS has reserved it for a reason and is expecting * assume the BIOS has reserved it for a reason and is expecting
...@@ -1300,7 +1300,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1300,7 +1300,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
if (mci == NULL) if (mci == NULL)
return -ENOMEM; return -ENOMEM;
debugf3("%s(): init mci\n", __func__); edac_dbg(3, "init mci\n");
mci->mtype_cap = MEM_FLAG_RDDR; mci->mtype_cap = MEM_FLAG_RDDR;
/* 3100 IMCH supports SECDEC only */ /* 3100 IMCH supports SECDEC only */
mci->edac_ctl_cap = (dev_idx == I3100) ? EDAC_FLAG_SECDED : mci->edac_ctl_cap = (dev_idx == I3100) ? EDAC_FLAG_SECDED :
...@@ -1308,9 +1308,9 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1308,9 +1308,9 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
/* FIXME - what if different memory types are in different csrows? */ /* FIXME - what if different memory types are in different csrows? */
mci->mod_name = EDAC_MOD_STR; mci->mod_name = EDAC_MOD_STR;
mci->mod_ver = E752X_REVISION; mci->mod_ver = E752X_REVISION;
mci->dev = &pdev->dev; mci->pdev = &pdev->dev;
debugf3("%s(): init pvt\n", __func__); edac_dbg(3, "init pvt\n");
pvt = (struct e752x_pvt *)mci->pvt_info; pvt = (struct e752x_pvt *)mci->pvt_info;
pvt->dev_info = &e752x_devs[dev_idx]; pvt->dev_info = &e752x_devs[dev_idx];
pvt->mc_symmetric = ((ddrcsr & 0x10) != 0); pvt->mc_symmetric = ((ddrcsr & 0x10) != 0);
...@@ -1320,7 +1320,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1320,7 +1320,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
return -ENODEV; return -ENODEV;
} }
debugf3("%s(): more mci init\n", __func__); edac_dbg(3, "more mci init\n");
mci->ctl_name = pvt->dev_info->ctl_name; mci->ctl_name = pvt->dev_info->ctl_name;
mci->dev_name = pci_name(pdev); mci->dev_name = pci_name(pdev);
mci->edac_check = e752x_check; mci->edac_check = e752x_check;
...@@ -1342,7 +1342,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1342,7 +1342,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
mci->edac_cap = EDAC_FLAG_SECDED; /* the only mode supported */ mci->edac_cap = EDAC_FLAG_SECDED; /* the only mode supported */
else else
mci->edac_cap |= EDAC_FLAG_NONE; mci->edac_cap |= EDAC_FLAG_NONE;
debugf3("%s(): tolm, remapbase, remaplimit\n", __func__); edac_dbg(3, "tolm, remapbase, remaplimit\n");
/* load the top of low memory, remap base, and remap limit vars */ /* load the top of low memory, remap base, and remap limit vars */
pci_read_config_word(pdev, E752X_TOLM, &pci_data); pci_read_config_word(pdev, E752X_TOLM, &pci_data);
...@@ -1359,7 +1359,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1359,7 +1359,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
* type of memory controller. The ID is therefore hardcoded to 0. * type of memory controller. The ID is therefore hardcoded to 0.
*/ */
if (edac_mc_add_mc(mci)) { if (edac_mc_add_mc(mci)) {
debugf3("%s(): failed edac_mc_add_mc()\n", __func__); edac_dbg(3, "failed edac_mc_add_mc()\n");
goto fail; goto fail;
} }
...@@ -1377,7 +1377,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1377,7 +1377,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
} }
/* get this far and it's successful */ /* get this far and it's successful */
debugf3("%s(): success\n", __func__); edac_dbg(3, "success\n");
return 0; return 0;
fail: fail:
...@@ -1393,7 +1393,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -1393,7 +1393,7 @@ static int e752x_probe1(struct pci_dev *pdev, int dev_idx)
static int __devinit e752x_init_one(struct pci_dev *pdev, static int __devinit e752x_init_one(struct pci_dev *pdev,
const struct pci_device_id *ent) const struct pci_device_id *ent)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
/* wake up and enable device */ /* wake up and enable device */
if (pci_enable_device(pdev) < 0) if (pci_enable_device(pdev) < 0)
...@@ -1407,7 +1407,7 @@ static void __devexit e752x_remove_one(struct pci_dev *pdev) ...@@ -1407,7 +1407,7 @@ static void __devexit e752x_remove_one(struct pci_dev *pdev)
struct mem_ctl_info *mci; struct mem_ctl_info *mci;
struct e752x_pvt *pvt; struct e752x_pvt *pvt;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
if (e752x_pci) if (e752x_pci)
edac_pci_release_generic_ctl(e752x_pci); edac_pci_release_generic_ctl(e752x_pci);
...@@ -1453,7 +1453,7 @@ static int __init e752x_init(void) ...@@ -1453,7 +1453,7 @@ static int __init e752x_init(void)
{ {
int pci_rc; int pci_rc;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
/* Ensure that the OPSTATE is set correctly for POLL or NMI */ /* Ensure that the OPSTATE is set correctly for POLL or NMI */
opstate_init(); opstate_init();
...@@ -1464,7 +1464,7 @@ static int __init e752x_init(void) ...@@ -1464,7 +1464,7 @@ static int __init e752x_init(void)
static void __exit e752x_exit(void) static void __exit e752x_exit(void)
{ {
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
pci_unregister_driver(&e752x_driver); pci_unregister_driver(&e752x_driver);
} }
......
...@@ -166,7 +166,7 @@ static const struct e7xxx_dev_info e7xxx_devs[] = { ...@@ -166,7 +166,7 @@ static const struct e7xxx_dev_info e7xxx_devs[] = {
/* FIXME - is this valid for both SECDED and S4ECD4ED? */ /* FIXME - is this valid for both SECDED and S4ECD4ED? */
static inline int e7xxx_find_channel(u16 syndrome) static inline int e7xxx_find_channel(u16 syndrome)
{ {
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
if ((syndrome & 0xff00) == 0) if ((syndrome & 0xff00) == 0)
return 0; return 0;
...@@ -186,7 +186,7 @@ static unsigned long ctl_page_to_phys(struct mem_ctl_info *mci, ...@@ -186,7 +186,7 @@ static unsigned long ctl_page_to_phys(struct mem_ctl_info *mci,
u32 remap; u32 remap;
struct e7xxx_pvt *pvt = (struct e7xxx_pvt *)mci->pvt_info; struct e7xxx_pvt *pvt = (struct e7xxx_pvt *)mci->pvt_info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
if ((page < pvt->tolm) || if ((page < pvt->tolm) ||
((page >= 0x100000) && (page < pvt->remapbase))) ((page >= 0x100000) && (page < pvt->remapbase)))
...@@ -208,7 +208,7 @@ static void process_ce(struct mem_ctl_info *mci, struct e7xxx_error_info *info) ...@@ -208,7 +208,7 @@ static void process_ce(struct mem_ctl_info *mci, struct e7xxx_error_info *info)
int row; int row;
int channel; int channel;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
/* read the error address */ /* read the error address */
error_1b = info->dram_celog_add; error_1b = info->dram_celog_add;
/* FIXME - should use PAGE_SHIFT */ /* FIXME - should use PAGE_SHIFT */
...@@ -219,15 +219,15 @@ static void process_ce(struct mem_ctl_info *mci, struct e7xxx_error_info *info) ...@@ -219,15 +219,15 @@ static void process_ce(struct mem_ctl_info *mci, struct e7xxx_error_info *info)
row = edac_mc_find_csrow_by_page(mci, page); row = edac_mc_find_csrow_by_page(mci, page);
/* convert syndrome to channel */ /* convert syndrome to channel */
channel = e7xxx_find_channel(syndrome); channel = e7xxx_find_channel(syndrome);
edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, page, 0, syndrome, edac_mc_handle_error(HW_EVENT_ERR_CORRECTED, mci, 1, page, 0, syndrome,
row, channel, -1, "e7xxx CE", "", NULL); row, channel, -1, "e7xxx CE", "");
} }
static void process_ce_no_info(struct mem_ctl_info *mci) static void process_ce_no_info(struct mem_ctl_info *mci)
{ {
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 0, 0, 0, -1, -1, -1, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1, 0, 0, 0, -1, -1, -1,
"e7xxx CE log register overflow", "", NULL); "e7xxx CE log register overflow", "");
} }
static void process_ue(struct mem_ctl_info *mci, struct e7xxx_error_info *info) static void process_ue(struct mem_ctl_info *mci, struct e7xxx_error_info *info)
...@@ -235,23 +235,23 @@ static void process_ue(struct mem_ctl_info *mci, struct e7xxx_error_info *info) ...@@ -235,23 +235,23 @@ static void process_ue(struct mem_ctl_info *mci, struct e7xxx_error_info *info)
u32 error_2b, block_page; u32 error_2b, block_page;
int row; int row;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
/* read the error address */ /* read the error address */
error_2b = info->dram_uelog_add; error_2b = info->dram_uelog_add;
/* FIXME - should use PAGE_SHIFT */ /* FIXME - should use PAGE_SHIFT */
block_page = error_2b >> 6; /* convert to 4k address */ block_page = error_2b >> 6; /* convert to 4k address */
row = edac_mc_find_csrow_by_page(mci, block_page); row = edac_mc_find_csrow_by_page(mci, block_page);
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, block_page, 0, 0, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1, block_page, 0, 0,
row, -1, -1, "e7xxx UE", "", NULL); row, -1, -1, "e7xxx UE", "");
} }
static void process_ue_no_info(struct mem_ctl_info *mci) static void process_ue_no_info(struct mem_ctl_info *mci)
{ {
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 0, 0, 0, -1, -1, -1, edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED, mci, 1, 0, 0, 0, -1, -1, -1,
"e7xxx UE log register overflow", "", NULL); "e7xxx UE log register overflow", "");
} }
static void e7xxx_get_error_info(struct mem_ctl_info *mci, static void e7xxx_get_error_info(struct mem_ctl_info *mci,
...@@ -334,7 +334,7 @@ static void e7xxx_check(struct mem_ctl_info *mci) ...@@ -334,7 +334,7 @@ static void e7xxx_check(struct mem_ctl_info *mci)
{ {
struct e7xxx_error_info info; struct e7xxx_error_info info;
debugf3("%s()\n", __func__); edac_dbg(3, "\n");
e7xxx_get_error_info(mci, &info); e7xxx_get_error_info(mci, &info);
e7xxx_process_error_info(mci, &info, 1); e7xxx_process_error_info(mci, &info, 1);
} }
...@@ -362,6 +362,7 @@ static void e7xxx_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -362,6 +362,7 @@ static void e7xxx_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
int drc_chan, drc_drbg, drc_ddim, mem_dev; int drc_chan, drc_drbg, drc_ddim, mem_dev;
struct csrow_info *csrow; struct csrow_info *csrow;
struct dimm_info *dimm; struct dimm_info *dimm;
enum edac_type edac_mode;
pci_read_config_dword(pdev, E7XXX_DRA, &dra); pci_read_config_dword(pdev, E7XXX_DRA, &dra);
drc_chan = dual_channel_active(drc, dev_idx); drc_chan = dual_channel_active(drc, dev_idx);
...@@ -377,13 +378,12 @@ static void e7xxx_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -377,13 +378,12 @@ static void e7xxx_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
for (index = 0; index < mci->nr_csrows; index++) { for (index = 0; index < mci->nr_csrows; index++) {
/* mem_dev 0=x8, 1=x4 */ /* mem_dev 0=x8, 1=x4 */
mem_dev = (dra >> (index * 4 + 3)) & 0x1; mem_dev = (dra >> (index * 4 + 3)) & 0x1;
csrow = &mci->csrows[index]; csrow = mci->csrows[index];
pci_read_config_byte(pdev, E7XXX_DRB + index, &value); pci_read_config_byte(pdev, E7XXX_DRB + index, &value);
/* convert a 64 or 32 MiB DRB to a page size. */ /* convert a 64 or 32 MiB DRB to a page size. */
cumul_size = value << (25 + drc_drbg - PAGE_SHIFT); cumul_size = value << (25 + drc_drbg - PAGE_SHIFT);
debugf3("%s(): (%d) cumul_size 0x%x\n", __func__, index, edac_dbg(3, "(%d) cumul_size 0x%x\n", index, cumul_size);
cumul_size);
if (cumul_size == last_cumul_size) if (cumul_size == last_cumul_size)
continue; /* not populated */ continue; /* not populated */
...@@ -392,28 +392,29 @@ static void e7xxx_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev, ...@@ -392,28 +392,29 @@ static void e7xxx_init_csrows(struct mem_ctl_info *mci, struct pci_dev *pdev,
nr_pages = cumul_size - last_cumul_size; nr_pages = cumul_size - last_cumul_size;
last_cumul_size = cumul_size; last_cumul_size = cumul_size;
for (j = 0; j < drc_chan + 1; j++) {
dimm = csrow->channels[j].dimm;
dimm->nr_pages = nr_pages / (drc_chan + 1);
dimm->grain = 1 << 12; /* 4KiB - resolution of CELOG */
dimm->mtype = MEM_RDDR; /* only one type supported */
dimm->dtype = mem_dev ? DEV_X4 : DEV_X8;
/* /*
* if single channel or x8 devices then SECDED * if single channel or x8 devices then SECDED
* if dual channel and x4 then S4ECD4ED * if dual channel and x4 then S4ECD4ED
*/ */
if (drc_ddim) { if (drc_ddim) {
if (drc_chan && mem_dev) { if (drc_chan && mem_dev) {
dimm->edac_mode = EDAC_S4ECD4ED; edac_mode = EDAC_S4ECD4ED;
mci->edac_cap |= EDAC_FLAG_S4ECD4ED; mci->edac_cap |= EDAC_FLAG_S4ECD4ED;
} else { } else {
dimm->edac_mode = EDAC_SECDED; edac_mode = EDAC_SECDED;
mci->edac_cap |= EDAC_FLAG_SECDED; mci->edac_cap |= EDAC_FLAG_SECDED;
} }
} else } else
dimm->edac_mode = EDAC_NONE; edac_mode = EDAC_NONE;
for (j = 0; j < drc_chan + 1; j++) {
dimm = csrow->channels[j]->dimm;
dimm->nr_pages = nr_pages / (drc_chan + 1);
dimm->grain = 1 << 12; /* 4KiB - resolution of CELOG */
dimm->mtype = MEM_RDDR; /* only one type supported */
dimm->dtype = mem_dev ? DEV_X4 : DEV_X8;
dimm->edac_mode = edac_mode;
} }
} }
} }
...@@ -428,7 +429,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -428,7 +429,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx)
int drc_chan; int drc_chan;
struct e7xxx_error_info discard; struct e7xxx_error_info discard;
debugf0("%s(): mci\n", __func__); edac_dbg(0, "mci\n");
pci_read_config_dword(pdev, E7XXX_DRC, &drc); pci_read_config_dword(pdev, E7XXX_DRC, &drc);
...@@ -451,15 +452,15 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -451,15 +452,15 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx)
if (mci == NULL) if (mci == NULL)
return -ENOMEM; return -ENOMEM;
debugf3("%s(): init mci\n", __func__); edac_dbg(3, "init mci\n");
mci->mtype_cap = MEM_FLAG_RDDR; mci->mtype_cap = MEM_FLAG_RDDR;
mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_SECDED | mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_SECDED |
EDAC_FLAG_S4ECD4ED; EDAC_FLAG_S4ECD4ED;
/* FIXME - what if different memory types are in different csrows? */ /* FIXME - what if different memory types are in different csrows? */
mci->mod_name = EDAC_MOD_STR; mci->mod_name = EDAC_MOD_STR;
mci->mod_ver = E7XXX_REVISION; mci->mod_ver = E7XXX_REVISION;
mci->dev = &pdev->dev; mci->pdev = &pdev->dev;
debugf3("%s(): init pvt\n", __func__); edac_dbg(3, "init pvt\n");
pvt = (struct e7xxx_pvt *)mci->pvt_info; pvt = (struct e7xxx_pvt *)mci->pvt_info;
pvt->dev_info = &e7xxx_devs[dev_idx]; pvt->dev_info = &e7xxx_devs[dev_idx];
pvt->bridge_ck = pci_get_device(PCI_VENDOR_ID_INTEL, pvt->bridge_ck = pci_get_device(PCI_VENDOR_ID_INTEL,
...@@ -472,14 +473,14 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -472,14 +473,14 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx)
goto fail0; goto fail0;
} }
debugf3("%s(): more mci init\n", __func__); edac_dbg(3, "more mci init\n");
mci->ctl_name = pvt->dev_info->ctl_name; mci->ctl_name = pvt->dev_info->ctl_name;
mci->dev_name = pci_name(pdev); mci->dev_name = pci_name(pdev);
mci->edac_check = e7xxx_check; mci->edac_check = e7xxx_check;
mci->ctl_page_to_phys = ctl_page_to_phys; mci->ctl_page_to_phys = ctl_page_to_phys;
e7xxx_init_csrows(mci, pdev, dev_idx, drc); e7xxx_init_csrows(mci, pdev, dev_idx, drc);
mci->edac_cap |= EDAC_FLAG_NONE; mci->edac_cap |= EDAC_FLAG_NONE;
debugf3("%s(): tolm, remapbase, remaplimit\n", __func__); edac_dbg(3, "tolm, remapbase, remaplimit\n");
/* load the top of low memory, remap base, and remap limit vars */ /* load the top of low memory, remap base, and remap limit vars */
pci_read_config_word(pdev, E7XXX_TOLM, &pci_data); pci_read_config_word(pdev, E7XXX_TOLM, &pci_data);
pvt->tolm = ((u32) pci_data) << 4; pvt->tolm = ((u32) pci_data) << 4;
...@@ -498,7 +499,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -498,7 +499,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx)
* type of memory controller. The ID is therefore hardcoded to 0. * type of memory controller. The ID is therefore hardcoded to 0.
*/ */
if (edac_mc_add_mc(mci)) { if (edac_mc_add_mc(mci)) {
debugf3("%s(): failed edac_mc_add_mc()\n", __func__); edac_dbg(3, "failed edac_mc_add_mc()\n");
goto fail1; goto fail1;
} }
...@@ -514,7 +515,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -514,7 +515,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx)
} }
/* get this far and it's successful */ /* get this far and it's successful */
debugf3("%s(): success\n", __func__); edac_dbg(3, "success\n");
return 0; return 0;
fail1: fail1:
...@@ -530,7 +531,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx) ...@@ -530,7 +531,7 @@ static int e7xxx_probe1(struct pci_dev *pdev, int dev_idx)
static int __devinit e7xxx_init_one(struct pci_dev *pdev, static int __devinit e7xxx_init_one(struct pci_dev *pdev,
const struct pci_device_id *ent) const struct pci_device_id *ent)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
/* wake up and enable device */ /* wake up and enable device */
return pci_enable_device(pdev) ? return pci_enable_device(pdev) ?
...@@ -542,7 +543,7 @@ static void __devexit e7xxx_remove_one(struct pci_dev *pdev) ...@@ -542,7 +543,7 @@ static void __devexit e7xxx_remove_one(struct pci_dev *pdev)
struct mem_ctl_info *mci; struct mem_ctl_info *mci;
struct e7xxx_pvt *pvt; struct e7xxx_pvt *pvt;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
if (e7xxx_pci) if (e7xxx_pci)
edac_pci_release_generic_ctl(e7xxx_pci); edac_pci_release_generic_ctl(e7xxx_pci);
......
...@@ -71,26 +71,21 @@ extern const char *edac_mem_types[]; ...@@ -71,26 +71,21 @@ extern const char *edac_mem_types[];
#ifdef CONFIG_EDAC_DEBUG #ifdef CONFIG_EDAC_DEBUG
extern int edac_debug_level; extern int edac_debug_level;
#define edac_debug_printk(level, fmt, arg...) \ #define edac_dbg(level, fmt, ...) \
do { \ do { \
if (level <= edac_debug_level) \ if (level <= edac_debug_level) \
edac_printk(KERN_DEBUG, EDAC_DEBUG, \ edac_printk(KERN_DEBUG, EDAC_DEBUG, \
"%s: " fmt, __func__, ##arg); \ "%s: " fmt, __func__, ##__VA_ARGS__); \
} while (0) } while (0)
#define debugf0( ... ) edac_debug_printk(0, __VA_ARGS__ )
#define debugf1( ... ) edac_debug_printk(1, __VA_ARGS__ )
#define debugf2( ... ) edac_debug_printk(2, __VA_ARGS__ )
#define debugf3( ... ) edac_debug_printk(3, __VA_ARGS__ )
#define debugf4( ... ) edac_debug_printk(4, __VA_ARGS__ )
#else /* !CONFIG_EDAC_DEBUG */ #else /* !CONFIG_EDAC_DEBUG */
#define debugf0( ... ) #define edac_dbg(level, fmt, ...) \
#define debugf1( ... ) do { \
#define debugf2( ... ) if (0) \
#define debugf3( ... ) edac_printk(KERN_DEBUG, EDAC_DEBUG, \
#define debugf4( ... ) "%s: " fmt, __func__, ##__VA_ARGS__); \
} while (0)
#endif /* !CONFIG_EDAC_DEBUG */ #endif /* !CONFIG_EDAC_DEBUG */
...@@ -460,15 +455,15 @@ extern int edac_mc_find_csrow_by_page(struct mem_ctl_info *mci, ...@@ -460,15 +455,15 @@ extern int edac_mc_find_csrow_by_page(struct mem_ctl_info *mci,
unsigned long page); unsigned long page);
void edac_mc_handle_error(const enum hw_event_mc_err_type type, void edac_mc_handle_error(const enum hw_event_mc_err_type type,
struct mem_ctl_info *mci, struct mem_ctl_info *mci,
const u16 error_count,
const unsigned long page_frame_number, const unsigned long page_frame_number,
const unsigned long offset_in_page, const unsigned long offset_in_page,
const unsigned long syndrome, const unsigned long syndrome,
const int layer0, const int top_layer,
const int layer1, const int mid_layer,
const int layer2, const int low_layer,
const char *msg, const char *msg,
const char *other_detail, const char *other_detail);
const void *mcelog);
/* /*
* edac_device APIs * edac_device APIs
......
...@@ -40,12 +40,13 @@ static LIST_HEAD(edac_device_list); ...@@ -40,12 +40,13 @@ static LIST_HEAD(edac_device_list);
#ifdef CONFIG_EDAC_DEBUG #ifdef CONFIG_EDAC_DEBUG
static void edac_device_dump_device(struct edac_device_ctl_info *edac_dev) static void edac_device_dump_device(struct edac_device_ctl_info *edac_dev)
{ {
debugf3("\tedac_dev = %p dev_idx=%d \n", edac_dev, edac_dev->dev_idx); edac_dbg(3, "\tedac_dev = %p dev_idx=%d\n",
debugf4("\tedac_dev->edac_check = %p\n", edac_dev->edac_check); edac_dev, edac_dev->dev_idx);
debugf3("\tdev = %p\n", edac_dev->dev); edac_dbg(4, "\tedac_dev->edac_check = %p\n", edac_dev->edac_check);
debugf3("\tmod_name:ctl_name = %s:%s\n", edac_dbg(3, "\tdev = %p\n", edac_dev->dev);
edac_dbg(3, "\tmod_name:ctl_name = %s:%s\n",
edac_dev->mod_name, edac_dev->ctl_name); edac_dev->mod_name, edac_dev->ctl_name);
debugf3("\tpvt_info = %p\n\n", edac_dev->pvt_info); edac_dbg(3, "\tpvt_info = %p\n\n", edac_dev->pvt_info);
} }
#endif /* CONFIG_EDAC_DEBUG */ #endif /* CONFIG_EDAC_DEBUG */
...@@ -82,8 +83,7 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info( ...@@ -82,8 +83,7 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info(
void *pvt, *p; void *pvt, *p;
int err; int err;
debugf4("%s() instances=%d blocks=%d\n", edac_dbg(4, "instances=%d blocks=%d\n", nr_instances, nr_blocks);
__func__, nr_instances, nr_blocks);
/* Calculate the size of memory we need to allocate AND /* Calculate the size of memory we need to allocate AND
* determine the offsets of the various item arrays * determine the offsets of the various item arrays
...@@ -156,8 +156,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info( ...@@ -156,8 +156,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info(
/* Name of this edac device */ /* Name of this edac device */
snprintf(dev_ctl->name,sizeof(dev_ctl->name),"%s",edac_device_name); snprintf(dev_ctl->name,sizeof(dev_ctl->name),"%s",edac_device_name);
debugf4("%s() edac_dev=%p next after end=%p\n", edac_dbg(4, "edac_dev=%p next after end=%p\n",
__func__, dev_ctl, pvt + sz_private ); dev_ctl, pvt + sz_private);
/* Initialize every Instance */ /* Initialize every Instance */
for (instance = 0; instance < nr_instances; instance++) { for (instance = 0; instance < nr_instances; instance++) {
...@@ -178,10 +178,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info( ...@@ -178,10 +178,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info(
snprintf(blk->name, sizeof(blk->name), snprintf(blk->name, sizeof(blk->name),
"%s%d", edac_block_name, block+offset_value); "%s%d", edac_block_name, block+offset_value);
debugf4("%s() instance=%d inst_p=%p block=#%d " edac_dbg(4, "instance=%d inst_p=%p block=#%d block_p=%p name='%s'\n",
"block_p=%p name='%s'\n", instance, inst, block, blk, blk->name);
__func__, instance, inst, block,
blk, blk->name);
/* if there are NO attributes OR no attribute pointer /* if there are NO attributes OR no attribute pointer
* then continue on to next block iteration * then continue on to next block iteration
...@@ -194,8 +192,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info( ...@@ -194,8 +192,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info(
attrib_p = &dev_attrib[block*nr_instances*nr_attrib]; attrib_p = &dev_attrib[block*nr_instances*nr_attrib];
blk->block_attributes = attrib_p; blk->block_attributes = attrib_p;
debugf4("%s() THIS BLOCK_ATTRIB=%p\n", edac_dbg(4, "THIS BLOCK_ATTRIB=%p\n",
__func__, blk->block_attributes); blk->block_attributes);
/* Initialize every user specified attribute in this /* Initialize every user specified attribute in this
* block with the data the caller passed in * block with the data the caller passed in
...@@ -214,9 +212,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info( ...@@ -214,9 +212,8 @@ struct edac_device_ctl_info *edac_device_alloc_ctl_info(
attrib->block = blk; /* up link */ attrib->block = blk; /* up link */
debugf4("%s() alloc-attrib=%p attrib_name='%s' " edac_dbg(4, "alloc-attrib=%p attrib_name='%s' attrib-spec=%p spec-name=%s\n",
"attrib-spec=%p spec-name=%s\n", attrib, attrib->attr.name,
__func__, attrib, attrib->attr.name,
&attrib_spec[attr], &attrib_spec[attr],
attrib_spec[attr].attr.name attrib_spec[attr].attr.name
); );
...@@ -273,7 +270,7 @@ static struct edac_device_ctl_info *find_edac_device_by_dev(struct device *dev) ...@@ -273,7 +270,7 @@ static struct edac_device_ctl_info *find_edac_device_by_dev(struct device *dev)
struct edac_device_ctl_info *edac_dev; struct edac_device_ctl_info *edac_dev;
struct list_head *item; struct list_head *item;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
list_for_each(item, &edac_device_list) { list_for_each(item, &edac_device_list) {
edac_dev = list_entry(item, struct edac_device_ctl_info, link); edac_dev = list_entry(item, struct edac_device_ctl_info, link);
...@@ -408,7 +405,7 @@ static void edac_device_workq_function(struct work_struct *work_req) ...@@ -408,7 +405,7 @@ static void edac_device_workq_function(struct work_struct *work_req)
void edac_device_workq_setup(struct edac_device_ctl_info *edac_dev, void edac_device_workq_setup(struct edac_device_ctl_info *edac_dev,
unsigned msec) unsigned msec)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
/* take the arg 'msec' and set it into the control structure /* take the arg 'msec' and set it into the control structure
* to used in the time period calculation * to used in the time period calculation
...@@ -496,7 +493,7 @@ EXPORT_SYMBOL_GPL(edac_device_alloc_index); ...@@ -496,7 +493,7 @@ EXPORT_SYMBOL_GPL(edac_device_alloc_index);
*/ */
int edac_device_add_device(struct edac_device_ctl_info *edac_dev) int edac_device_add_device(struct edac_device_ctl_info *edac_dev)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
#ifdef CONFIG_EDAC_DEBUG #ifdef CONFIG_EDAC_DEBUG
if (edac_debug_level >= 3) if (edac_debug_level >= 3)
...@@ -570,7 +567,7 @@ struct edac_device_ctl_info *edac_device_del_device(struct device *dev) ...@@ -570,7 +567,7 @@ struct edac_device_ctl_info *edac_device_del_device(struct device *dev)
{ {
struct edac_device_ctl_info *edac_dev; struct edac_device_ctl_info *edac_dev;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
mutex_lock(&device_ctls_mutex); mutex_lock(&device_ctls_mutex);
......
...@@ -202,7 +202,7 @@ static void edac_device_ctrl_master_release(struct kobject *kobj) ...@@ -202,7 +202,7 @@ static void edac_device_ctrl_master_release(struct kobject *kobj)
{ {
struct edac_device_ctl_info *edac_dev = to_edacdev(kobj); struct edac_device_ctl_info *edac_dev = to_edacdev(kobj);
debugf4("%s() control index=%d\n", __func__, edac_dev->dev_idx); edac_dbg(4, "control index=%d\n", edac_dev->dev_idx);
/* decrement the EDAC CORE module ref count */ /* decrement the EDAC CORE module ref count */
module_put(edac_dev->owner); module_put(edac_dev->owner);
...@@ -233,12 +233,12 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev) ...@@ -233,12 +233,12 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev)
struct bus_type *edac_subsys; struct bus_type *edac_subsys;
int err; int err;
debugf1("%s()\n", __func__); edac_dbg(1, "\n");
/* get the /sys/devices/system/edac reference */ /* get the /sys/devices/system/edac reference */
edac_subsys = edac_get_sysfs_subsys(); edac_subsys = edac_get_sysfs_subsys();
if (edac_subsys == NULL) { if (edac_subsys == NULL) {
debugf1("%s() no edac_subsys error\n", __func__); edac_dbg(1, "no edac_subsys error\n");
err = -ENODEV; err = -ENODEV;
goto err_out; goto err_out;
} }
...@@ -264,8 +264,8 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev) ...@@ -264,8 +264,8 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev)
&edac_subsys->dev_root->kobj, &edac_subsys->dev_root->kobj,
"%s", edac_dev->name); "%s", edac_dev->name);
if (err) { if (err) {
debugf1("%s()Failed to register '.../edac/%s'\n", edac_dbg(1, "Failed to register '.../edac/%s'\n",
__func__, edac_dev->name); edac_dev->name);
goto err_kobj_reg; goto err_kobj_reg;
} }
kobject_uevent(&edac_dev->kobj, KOBJ_ADD); kobject_uevent(&edac_dev->kobj, KOBJ_ADD);
...@@ -274,8 +274,7 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev) ...@@ -274,8 +274,7 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev)
* edac_device_unregister_sysfs_main_kobj() must be used * edac_device_unregister_sysfs_main_kobj() must be used
*/ */
debugf4("%s() Registered '.../edac/%s' kobject\n", edac_dbg(4, "Registered '.../edac/%s' kobject\n", edac_dev->name);
__func__, edac_dev->name);
return 0; return 0;
...@@ -296,9 +295,8 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev) ...@@ -296,9 +295,8 @@ int edac_device_register_sysfs_main_kobj(struct edac_device_ctl_info *edac_dev)
*/ */
void edac_device_unregister_sysfs_main_kobj(struct edac_device_ctl_info *dev) void edac_device_unregister_sysfs_main_kobj(struct edac_device_ctl_info *dev)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
debugf4("%s() name of kobject is: %s\n", edac_dbg(4, "name of kobject is: %s\n", kobject_name(&dev->kobj));
__func__, kobject_name(&dev->kobj));
/* /*
* Unregister the edac device's kobject and * Unregister the edac device's kobject and
...@@ -336,7 +334,7 @@ static void edac_device_ctrl_instance_release(struct kobject *kobj) ...@@ -336,7 +334,7 @@ static void edac_device_ctrl_instance_release(struct kobject *kobj)
{ {
struct edac_device_instance *instance; struct edac_device_instance *instance;
debugf1("%s()\n", __func__); edac_dbg(1, "\n");
/* map from this kobj to the main control struct /* map from this kobj to the main control struct
* and then dec the main kobj count * and then dec the main kobj count
...@@ -442,7 +440,7 @@ static void edac_device_ctrl_block_release(struct kobject *kobj) ...@@ -442,7 +440,7 @@ static void edac_device_ctrl_block_release(struct kobject *kobj)
{ {
struct edac_device_block *block; struct edac_device_block *block;
debugf1("%s()\n", __func__); edac_dbg(1, "\n");
/* get the container of the kobj */ /* get the container of the kobj */
block = to_block(kobj); block = to_block(kobj);
...@@ -524,10 +522,10 @@ static int edac_device_create_block(struct edac_device_ctl_info *edac_dev, ...@@ -524,10 +522,10 @@ static int edac_device_create_block(struct edac_device_ctl_info *edac_dev,
struct edac_dev_sysfs_block_attribute *sysfs_attrib; struct edac_dev_sysfs_block_attribute *sysfs_attrib;
struct kobject *main_kobj; struct kobject *main_kobj;
debugf4("%s() Instance '%s' inst_p=%p block '%s' block_p=%p\n", edac_dbg(4, "Instance '%s' inst_p=%p block '%s' block_p=%p\n",
__func__, instance->name, instance, block->name, block); instance->name, instance, block->name, block);
debugf4("%s() block kobj=%p block kobj->parent=%p\n", edac_dbg(4, "block kobj=%p block kobj->parent=%p\n",
__func__, &block->kobj, &block->kobj.parent); &block->kobj, &block->kobj.parent);
/* init this block's kobject */ /* init this block's kobject */
memset(&block->kobj, 0, sizeof(struct kobject)); memset(&block->kobj, 0, sizeof(struct kobject));
...@@ -546,8 +544,7 @@ static int edac_device_create_block(struct edac_device_ctl_info *edac_dev, ...@@ -546,8 +544,7 @@ static int edac_device_create_block(struct edac_device_ctl_info *edac_dev,
&instance->kobj, &instance->kobj,
"%s", block->name); "%s", block->name);
if (err) { if (err) {
debugf1("%s() Failed to register instance '%s'\n", edac_dbg(1, "Failed to register instance '%s'\n", block->name);
__func__, block->name);
kobject_put(main_kobj); kobject_put(main_kobj);
err = -ENODEV; err = -ENODEV;
goto err_out; goto err_out;
...@@ -560,9 +557,7 @@ static int edac_device_create_block(struct edac_device_ctl_info *edac_dev, ...@@ -560,9 +557,7 @@ static int edac_device_create_block(struct edac_device_ctl_info *edac_dev,
if (sysfs_attrib && block->nr_attribs) { if (sysfs_attrib && block->nr_attribs) {
for (i = 0; i < block->nr_attribs; i++, sysfs_attrib++) { for (i = 0; i < block->nr_attribs; i++, sysfs_attrib++) {
debugf4("%s() creating block attrib='%s' " edac_dbg(4, "creating block attrib='%s' attrib->%p to kobj=%p\n",
"attrib->%p to kobj=%p\n",
__func__,
sysfs_attrib->attr.name, sysfs_attrib->attr.name,
sysfs_attrib, &block->kobj); sysfs_attrib, &block->kobj);
...@@ -647,14 +642,14 @@ static int edac_device_create_instance(struct edac_device_ctl_info *edac_dev, ...@@ -647,14 +642,14 @@ static int edac_device_create_instance(struct edac_device_ctl_info *edac_dev,
err = kobject_init_and_add(&instance->kobj, &ktype_instance_ctrl, err = kobject_init_and_add(&instance->kobj, &ktype_instance_ctrl,
&edac_dev->kobj, "%s", instance->name); &edac_dev->kobj, "%s", instance->name);
if (err != 0) { if (err != 0) {
debugf2("%s() Failed to register instance '%s'\n", edac_dbg(2, "Failed to register instance '%s'\n",
__func__, instance->name); instance->name);
kobject_put(main_kobj); kobject_put(main_kobj);
goto err_out; goto err_out;
} }
debugf4("%s() now register '%d' blocks for instance %d\n", edac_dbg(4, "now register '%d' blocks for instance %d\n",
__func__, instance->nr_blocks, idx); instance->nr_blocks, idx);
/* register all blocks of this instance */ /* register all blocks of this instance */
for (i = 0; i < instance->nr_blocks; i++) { for (i = 0; i < instance->nr_blocks; i++) {
...@@ -670,8 +665,8 @@ static int edac_device_create_instance(struct edac_device_ctl_info *edac_dev, ...@@ -670,8 +665,8 @@ static int edac_device_create_instance(struct edac_device_ctl_info *edac_dev,
} }
kobject_uevent(&instance->kobj, KOBJ_ADD); kobject_uevent(&instance->kobj, KOBJ_ADD);
debugf4("%s() Registered instance %d '%s' kobject\n", edac_dbg(4, "Registered instance %d '%s' kobject\n",
__func__, idx, instance->name); idx, instance->name);
return 0; return 0;
...@@ -715,7 +710,7 @@ static int edac_device_create_instances(struct edac_device_ctl_info *edac_dev) ...@@ -715,7 +710,7 @@ static int edac_device_create_instances(struct edac_device_ctl_info *edac_dev)
int i, j; int i, j;
int err; int err;
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
/* iterate over creation of the instances */ /* iterate over creation of the instances */
for (i = 0; i < edac_dev->nr_instances; i++) { for (i = 0; i < edac_dev->nr_instances; i++) {
...@@ -817,12 +812,12 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev) ...@@ -817,12 +812,12 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev)
int err; int err;
struct kobject *edac_kobj = &edac_dev->kobj; struct kobject *edac_kobj = &edac_dev->kobj;
debugf0("%s() idx=%d\n", __func__, edac_dev->dev_idx); edac_dbg(0, "idx=%d\n", edac_dev->dev_idx);
/* go create any main attributes callers wants */ /* go create any main attributes callers wants */
err = edac_device_add_main_sysfs_attributes(edac_dev); err = edac_device_add_main_sysfs_attributes(edac_dev);
if (err) { if (err) {
debugf0("%s() failed to add sysfs attribs\n", __func__); edac_dbg(0, "failed to add sysfs attribs\n");
goto err_out; goto err_out;
} }
...@@ -832,8 +827,7 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev) ...@@ -832,8 +827,7 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev)
err = sysfs_create_link(edac_kobj, err = sysfs_create_link(edac_kobj,
&edac_dev->dev->kobj, EDAC_DEVICE_SYMLINK); &edac_dev->dev->kobj, EDAC_DEVICE_SYMLINK);
if (err) { if (err) {
debugf0("%s() sysfs_create_link() returned err= %d\n", edac_dbg(0, "sysfs_create_link() returned err= %d\n", err);
__func__, err);
goto err_remove_main_attribs; goto err_remove_main_attribs;
} }
...@@ -843,14 +837,13 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev) ...@@ -843,14 +837,13 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev)
*/ */
err = edac_device_create_instances(edac_dev); err = edac_device_create_instances(edac_dev);
if (err) { if (err) {
debugf0("%s() edac_device_create_instances() " edac_dbg(0, "edac_device_create_instances() returned err= %d\n",
"returned err= %d\n", __func__, err); err);
goto err_remove_link; goto err_remove_link;
} }
debugf4("%s() create-instances done, idx=%d\n", edac_dbg(4, "create-instances done, idx=%d\n", edac_dev->dev_idx);
__func__, edac_dev->dev_idx);
return 0; return 0;
...@@ -873,7 +866,7 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev) ...@@ -873,7 +866,7 @@ int edac_device_create_sysfs(struct edac_device_ctl_info *edac_dev)
*/ */
void edac_device_remove_sysfs(struct edac_device_ctl_info *edac_dev) void edac_device_remove_sysfs(struct edac_device_ctl_info *edac_dev)
{ {
debugf0("%s()\n", __func__); edac_dbg(0, "\n");
/* remove any main attributes for this device */ /* remove any main attributes for this device */
edac_device_remove_main_sysfs_attributes(edac_dev); edac_device_remove_main_sysfs_attributes(edac_dev);
......
此差异已折叠。
此差异已折叠。
此差异已折叠。
...@@ -19,12 +19,12 @@ ...@@ -19,12 +19,12 @@
* *
* edac_mc objects * edac_mc objects
*/ */
extern int edac_sysfs_setup_mc_kset(void); /* on edac_mc_sysfs.c */
extern void edac_sysfs_teardown_mc_kset(void); int edac_mc_sysfs_init(void);
extern int edac_mc_register_sysfs_main_kobj(struct mem_ctl_info *mci); void edac_mc_sysfs_exit(void);
extern void edac_mc_unregister_sysfs_main_kobj(struct mem_ctl_info *mci);
extern int edac_create_sysfs_mci_device(struct mem_ctl_info *mci); extern int edac_create_sysfs_mci_device(struct mem_ctl_info *mci);
extern void edac_remove_sysfs_mci_device(struct mem_ctl_info *mci); extern void edac_remove_sysfs_mci_device(struct mem_ctl_info *mci);
void edac_unregister_sysfs(struct mem_ctl_info *mci);
extern int edac_get_log_ue(void); extern int edac_get_log_ue(void);
extern int edac_get_log_ce(void); extern int edac_get_log_ce(void);
extern int edac_get_panic_on_ue(void); extern int edac_get_panic_on_ue(void);
...@@ -34,6 +34,10 @@ extern int edac_mc_get_panic_on_ue(void); ...@@ -34,6 +34,10 @@ extern int edac_mc_get_panic_on_ue(void);
extern int edac_get_poll_msec(void); extern int edac_get_poll_msec(void);
extern int edac_mc_get_poll_msec(void); extern int edac_mc_get_poll_msec(void);
unsigned edac_dimm_info_location(struct dimm_info *dimm, char *buf,
unsigned len);
/* on edac_device.c */
extern int edac_device_register_sysfs_main_kobj( extern int edac_device_register_sysfs_main_kobj(
struct edac_device_ctl_info *edac_dev); struct edac_device_ctl_info *edac_dev);
extern void edac_device_unregister_sysfs_main_kobj( extern void edac_device_unregister_sysfs_main_kobj(
...@@ -52,6 +56,20 @@ extern void edac_mc_reset_delay_period(int value); ...@@ -52,6 +56,20 @@ extern void edac_mc_reset_delay_period(int value);
extern void *edac_align_ptr(void **p, unsigned size, int n_elems); extern void *edac_align_ptr(void **p, unsigned size, int n_elems);
/*
* EDAC debugfs functions
*/
#ifdef CONFIG_EDAC_DEBUG
int edac_debugfs_init(void);
void edac_debugfs_exit(void);
#else
static inline int edac_debugfs_init(void)
{
return -ENODEV;
}
static inline void edac_debugfs_exit(void) {}
#endif
/* /*
* EDAC PCI functions * EDAC PCI functions
*/ */
......
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册