diff options
Diffstat (limited to 'drivers/edac/amd64_edac_inj.c')
| -rw-r--r-- | drivers/edac/amd64_edac_inj.c | 273 | 
1 files changed, 146 insertions, 127 deletions
diff --git a/drivers/edac/amd64_edac_inj.c b/drivers/edac/amd64_edac_inj.c index 29f1f7a612d..0d66ae68d46 100644 --- a/drivers/edac/amd64_edac_inj.c +++ b/drivers/edac/amd64_edac_inj.c @@ -1,7 +1,10 @@  #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;  	return sprintf(buf, "0x%x\n", pvt->injection.section);  } @@ -12,31 +15,33 @@ static ssize_t amd64_inject_section_show(struct mem_ctl_info *mci, char *buf)   *   * 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)  { +	struct mem_ctl_info *mci = to_mci(dev);  	struct amd64_pvt *pvt = mci->pvt_info;  	unsigned long value; -	int ret = 0; +	int ret; -	ret = strict_strtoul(data, 10, &value); -	if (ret != -EINVAL) { +	ret = kstrtoul(data, 10, &value); +	if (ret < 0) +		return ret; -		if (value > 3) { -			amd64_printk(KERN_WARNING, -				     "%s: invalid section 0x%lx\n", -				     __func__, value); -			return -EINVAL; -		} - -		pvt->injection.section = (u32) value; -		return count; +	if (value > 3) { +		amd64_warn("%s: invalid section 0x%lx\n", __func__, value); +		return -EINVAL;  	} -	return ret; + +	pvt->injection.section = (u32) value; +	return count;  } -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;  	return sprintf(buf, "0x%x\n", pvt->injection.word);  } @@ -47,31 +52,33 @@ static ssize_t amd64_inject_word_show(struct mem_ctl_info *mci, char *buf)   *   * range: 0..8   */ -static ssize_t amd64_inject_word_store(struct mem_ctl_info *mci, -					const char *data, size_t count) +static ssize_t amd64_inject_word_store(struct device *dev, +				       struct device_attribute *mattr, +				       const char *data, size_t count)  { +	struct mem_ctl_info *mci = to_mci(dev);  	struct amd64_pvt *pvt = mci->pvt_info;  	unsigned long value; -	int ret = 0; +	int ret; -	ret = strict_strtoul(data, 10, &value); -	if (ret != -EINVAL) { +	ret = kstrtoul(data, 10, &value); +	if (ret < 0) +		return ret; -		if (value > 8) { -			amd64_printk(KERN_WARNING, -				     "%s: invalid word 0x%lx\n", -				     __func__, value); -			return -EINVAL; -		} - -		pvt->injection.word = (u32) value; -		return count; +	if (value > 8) { +		amd64_warn("%s: invalid word 0x%lx\n", __func__, value); +		return -EINVAL;  	} -	return ret; + +	pvt->injection.word = (u32) value; +	return count;  } -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;  	return sprintf(buf, "0x%x\n", pvt->injection.bit_map);  } @@ -81,142 +88,154 @@ 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   * 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, -					     const char *data, size_t count) +static ssize_t amd64_inject_ecc_vector_store(struct device *dev, +				       struct device_attribute *mattr, +				       const char *data, size_t count)  { +	struct mem_ctl_info *mci = to_mci(dev);  	struct amd64_pvt *pvt = mci->pvt_info;  	unsigned long value; -	int ret = 0; - -	ret = strict_strtoul(data, 16, &value); -	if (ret != -EINVAL) { +	int ret; -		if (value & 0xFFFF0000) { -			amd64_printk(KERN_WARNING, -				     "%s: invalid EccVector: 0x%lx\n", -				     __func__, value); -			return -EINVAL; -		} +	ret = kstrtoul(data, 16, &value); +	if (ret < 0) +		return ret; -		pvt->injection.bit_map = (u32) value; -		return count; +	if (value & 0xFFFF0000) { +		amd64_warn("%s: invalid EccVector: 0x%lx\n", __func__, value); +		return -EINVAL;  	} -	return ret; + +	pvt->injection.bit_map = (u32) value; +	return count;  }  /*   * Do a DRAM ECC read. Assemble staged values in the pvt area, format into   * fields needed by the injection registers and read the NB Array Data Port.   */ -static ssize_t amd64_inject_read_store(struct mem_ctl_info *mci, -					const char *data, size_t count) +static ssize_t amd64_inject_read_store(struct device *dev, +				       struct device_attribute *mattr, +				       const char *data, size_t count)  { +	struct mem_ctl_info *mci = to_mci(dev);  	struct amd64_pvt *pvt = mci->pvt_info;  	unsigned long value;  	u32 section, word_bits; -	int ret = 0; +	int ret; -	ret = strict_strtoul(data, 10, &value); -	if (ret != -EINVAL) { +	ret = kstrtoul(data, 10, &value); +	if (ret < 0) +		return ret; -		/* Form value to choose 16-byte section of cacheline */ -		section = F10_NB_ARRAY_DRAM_ECC | -				SET_NB_ARRAY_ADDRESS(pvt->injection.section); -		pci_write_config_dword(pvt->misc_f3_ctl, -					F10_NB_ARRAY_ADDR, section); +	/* Form value to choose 16-byte section of cacheline */ +	section = F10_NB_ARRAY_DRAM | SET_NB_ARRAY_ADDR(pvt->injection.section); -		word_bits = SET_NB_DRAM_INJECTION_READ(pvt->injection.word, -						pvt->injection.bit_map); +	amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_ADDR, section); -		/* Issue 'word' and 'bit' along with the READ request */ -		pci_write_config_dword(pvt->misc_f3_ctl, -					F10_NB_ARRAY_DATA, word_bits); +	word_bits = SET_NB_DRAM_INJECTION_READ(pvt->injection); -		debugf0("section=0x%x word_bits=0x%x\n", section, word_bits); +	/* Issue 'word' and 'bit' along with the READ request */ +	amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, word_bits); -		return count; -	} -	return ret; +	edac_dbg(0, "section=0x%x word_bits=0x%x\n", section, word_bits); + +	return count;  }  /*   * Do a DRAM ECC write. Assemble staged values in the pvt area and format into   * fields needed by the injection registers.   */ -static ssize_t 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)  { +	struct mem_ctl_info *mci = to_mci(dev);  	struct amd64_pvt *pvt = mci->pvt_info; +	u32 section, word_bits, tmp;  	unsigned long value; -	u32 section, word_bits; -	int ret = 0; +	int ret; + +	ret = kstrtoul(data, 10, &value); +	if (ret < 0) +		return ret; -	ret = strict_strtoul(data, 10, &value); -	if (ret != -EINVAL) { +	/* Form value to choose 16-byte section of cacheline */ +	section = F10_NB_ARRAY_DRAM | SET_NB_ARRAY_ADDR(pvt->injection.section); -		/* Form value to choose 16-byte section of cacheline */ -		section = F10_NB_ARRAY_DRAM_ECC | -				SET_NB_ARRAY_ADDRESS(pvt->injection.section); -		pci_write_config_dword(pvt->misc_f3_ctl, -					F10_NB_ARRAY_ADDR, section); +	amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_ADDR, section); -		word_bits = SET_NB_DRAM_INJECTION_WRITE(pvt->injection.word, -						pvt->injection.bit_map); +	word_bits = SET_NB_DRAM_INJECTION_WRITE(pvt->injection); -		/* Issue 'word' and 'bit' along with the READ request */ -		pci_write_config_dword(pvt->misc_f3_ctl, -					F10_NB_ARRAY_DATA, word_bits); +	pr_notice_once("Don't forget to decrease MCE polling interval in\n" +			"/sys/bus/machinecheck/devices/machinecheck<CPUNUM>/check_interval\n" +			"so that you can get the error report faster.\n"); -		debugf0("section=0x%x word_bits=0x%x\n", section, word_bits); +	on_each_cpu(disable_caches, NULL, 1); -		return count; +	/* Issue 'word' and 'bit' along with the READ request */ +	amd64_write_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, word_bits); + + retry: +	/* wait until injection happens */ +	amd64_read_pci_cfg(pvt->F3, F10_NB_ARRAY_DATA, &tmp); +	if (tmp & F10_NB_ARR_ECC_WR_REQ) { +		cpu_relax(); +		goto retry;  	} -	return ret; + +	on_each_cpu(enable_caches, NULL, 1); + +	edac_dbg(0, "section=0x%x word_bits=0x%x\n", section, word_bits); + +	return count;  }  /*   * update NUM_INJ_ATTRS in case you add new members   */ -struct mcidev_sysfs_attribute amd64_inj_attrs[] = { - -	{ -		.attr = { -			.name = "inject_section", -			.mode = (S_IRUGO | S_IWUSR) -		}, -		.show = amd64_inject_section_show, -		.store = amd64_inject_section_store, -	}, -	{ -		.attr = { -			.name = "inject_word", -			.mode = (S_IRUGO | S_IWUSR) -		}, -		.show = amd64_inject_word_show, -		.store = amd64_inject_word_store, -	}, -	{ -		.attr = { -			.name = "inject_ecc_vector", -			.mode = (S_IRUGO | S_IWUSR) -		}, -		.show = amd64_inject_ecc_vector_show, -		.store = amd64_inject_ecc_vector_store, -	}, -	{ -		.attr = { -			.name = "inject_write", -			.mode = (S_IRUGO | S_IWUSR) -		}, -		.show = NULL, -		.store = amd64_inject_write_store, -	}, -	{ -		.attr = { -			.name = "inject_read", -			.mode = (S_IRUGO | S_IWUSR) -		}, -		.show = NULL, -		.store = amd64_inject_read_store, -	}, -}; + +static DEVICE_ATTR(inject_section, S_IRUGO | S_IWUSR, +		   amd64_inject_section_show, amd64_inject_section_store); +static DEVICE_ATTR(inject_word, S_IRUGO | S_IWUSR, +		   amd64_inject_word_show, amd64_inject_word_store); +static DEVICE_ATTR(inject_ecc_vector, S_IRUGO | S_IWUSR, +		   amd64_inject_ecc_vector_show, amd64_inject_ecc_vector_store); +static DEVICE_ATTR(inject_write, S_IWUSR, +		   NULL, amd64_inject_write_store); +static DEVICE_ATTR(inject_read,  S_IWUSR, +		   NULL, amd64_inject_read_store); + + +int amd64_create_sysfs_inject_files(struct mem_ctl_info *mci) +{ +	int rc; + +	rc = device_create_file(&mci->dev, &dev_attr_inject_section); +	if (rc < 0) +		return rc; +	rc = device_create_file(&mci->dev, &dev_attr_inject_word); +	if (rc < 0) +		return rc; +	rc = device_create_file(&mci->dev, &dev_attr_inject_ecc_vector); +	if (rc < 0) +		return rc; +	rc = device_create_file(&mci->dev, &dev_attr_inject_write); +	if (rc < 0) +		return rc; +	rc = device_create_file(&mci->dev, &dev_attr_inject_read); +	if (rc < 0) +		return rc; + +	return 0; +} + +void amd64_remove_sysfs_inject_files(struct mem_ctl_info *mci) +{ +	device_remove_file(&mci->dev, &dev_attr_inject_section); +	device_remove_file(&mci->dev, &dev_attr_inject_word); +	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); +}  | 
