diff options
Diffstat (limited to 'drivers/misc')
44 files changed, 1261 insertions, 401 deletions
| diff --git a/drivers/misc/Kconfig b/drivers/misc/Kconfig index 3726eacdf65d..f417b06e11c5 100644 --- a/drivers/misc/Kconfig +++ b/drivers/misc/Kconfig @@ -513,6 +513,14 @@ config MISC_RTSX  	tristate  	default MISC_RTSX_PCI || MISC_RTSX_USB +config PVPANIC +	tristate "pvpanic device support" +	depends on HAS_IOMEM && (ACPI || OF) +	help +	  This driver provides support for the pvpanic device.  pvpanic is +	  a paravirtualized device provided by QEMU; it lets a virtual machine +	  (guest) communicate panic events to the host. +  source "drivers/misc/c2port/Kconfig"  source "drivers/misc/eeprom/Kconfig"  source "drivers/misc/cb710/Kconfig" diff --git a/drivers/misc/Makefile b/drivers/misc/Makefile index af22bbc3d00c..e39ccbbc1b3a 100644 --- a/drivers/misc/Makefile +++ b/drivers/misc/Makefile @@ -57,4 +57,5 @@ obj-$(CONFIG_ASPEED_LPC_CTRL)	+= aspeed-lpc-ctrl.o  obj-$(CONFIG_ASPEED_LPC_SNOOP)	+= aspeed-lpc-snoop.o  obj-$(CONFIG_PCI_ENDPOINT_TEST)	+= pci_endpoint_test.o  obj-$(CONFIG_OCXL)		+= ocxl/ -obj-$(CONFIG_MISC_RTSX)		+= cardreader/ +obj-y				+= cardreader/ +obj-$(CONFIG_PVPANIC)   	+= pvpanic.o diff --git a/drivers/misc/altera-stapl/altera.c b/drivers/misc/altera-stapl/altera.c index ef83a9078646..d2ed3b9728b7 100644 --- a/drivers/misc/altera-stapl/altera.c +++ b/drivers/misc/altera-stapl/altera.c @@ -2176,8 +2176,7 @@ static int altera_get_note(u8 *p, s32 program_size,  			key_ptr = &p[note_strings +  					get_unaligned_be32(  					&p[note_table + (8 * i)])]; -			if ((strncasecmp(key, key_ptr, strlen(key_ptr)) == 0) && -						(key != NULL)) { +			if (key && !strncasecmp(key, key_ptr, strlen(key_ptr))) {  				status = 0;  				value_ptr = &p[note_strings + diff --git a/drivers/misc/cardreader/Kconfig b/drivers/misc/cardreader/Kconfig index 69e815e32a8c..ed8993b5d058 100644 --- a/drivers/misc/cardreader/Kconfig +++ b/drivers/misc/cardreader/Kconfig @@ -1,3 +1,14 @@ +config MISC_ALCOR_PCI +	tristate "Alcor Micro/Alcor Link PCI-E card reader" +	depends on PCI +	select MFD_CORE +	help +	  This supports for Alcor Micro PCI-Express card reader including au6601, +	  au6621. +	  Alcor Micro card readers support access to many types of memory cards, +	  such as Memory Stick, Memory Stick Pro, Secure Digital and +	  MultiMediaCard. +  config MISC_RTSX_PCI  	tristate "Realtek PCI-E card reader"  	depends on PCI diff --git a/drivers/misc/cardreader/Makefile b/drivers/misc/cardreader/Makefile index 9fabfcc6fa7a..9882d2a1025c 100644 --- a/drivers/misc/cardreader/Makefile +++ b/drivers/misc/cardreader/Makefile @@ -1,4 +1,4 @@ -rtsx_pci-objs := rtsx_pcr.o rts5209.o rts5229.o rtl8411.o rts5227.o rts5249.o rts5260.o - +obj-$(CONFIG_MISC_ALCOR_PCI)	+= alcor_pci.o  obj-$(CONFIG_MISC_RTSX_PCI)	+= rtsx_pci.o +rtsx_pci-objs := rtsx_pcr.o rts5209.o rts5229.o rtl8411.o rts5227.o rts5249.o rts5260.o  obj-$(CONFIG_MISC_RTSX_USB)	+= rtsx_usb.o diff --git a/drivers/misc/cardreader/alcor_pci.c b/drivers/misc/cardreader/alcor_pci.c new file mode 100644 index 000000000000..bcb10fa4bc3a --- /dev/null +++ b/drivers/misc/cardreader/alcor_pci.c @@ -0,0 +1,371 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Copyright (C) 2018 Oleksij Rempel <linux@rempel-privat.de> + * + * Driver for Alcor Micro AU6601 and AU6621 controllers + */ + +#include <linux/delay.h> +#include <linux/interrupt.h> +#include <linux/io.h> +#include <linux/irq.h> +#include <linux/mfd/core.h> +#include <linux/module.h> +#include <linux/pci.h> +#include <linux/platform_device.h> +#include <linux/pm.h> + +#include <linux/alcor_pci.h> + +#define DRV_NAME_ALCOR_PCI			"alcor_pci" + +static DEFINE_IDA(alcor_pci_idr); + +static struct mfd_cell alcor_pci_cells[] = { +	[ALCOR_SD_CARD] = { +		.name = DRV_NAME_ALCOR_PCI_SDMMC, +	}, +	[ALCOR_MS_CARD] = { +		.name = DRV_NAME_ALCOR_PCI_MS, +	}, +}; + +static const struct alcor_dev_cfg alcor_cfg = { +	.dma = 0, +}; + +static const struct alcor_dev_cfg au6621_cfg = { +	.dma = 1, +}; + +static const struct pci_device_id pci_ids[] = { +	{ PCI_DEVICE(PCI_ID_ALCOR_MICRO, PCI_ID_AU6601), +		.driver_data = (kernel_ulong_t)&alcor_cfg }, +	{ PCI_DEVICE(PCI_ID_ALCOR_MICRO, PCI_ID_AU6621), +		.driver_data = (kernel_ulong_t)&au6621_cfg }, +	{ }, +}; +MODULE_DEVICE_TABLE(pci, pci_ids); + +void alcor_write8(struct alcor_pci_priv *priv, u8 val, unsigned int addr) +{ +	writeb(val, priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_write8); + +void alcor_write16(struct alcor_pci_priv *priv, u16 val, unsigned int addr) +{ +	writew(val, priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_write16); + +void alcor_write32(struct alcor_pci_priv *priv, u32 val, unsigned int addr) +{ +	writel(val, priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_write32); + +void alcor_write32be(struct alcor_pci_priv *priv, u32 val, unsigned int addr) +{ +	iowrite32be(val, priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_write32be); + +u8 alcor_read8(struct alcor_pci_priv *priv, unsigned int addr) +{ +	return readb(priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_read8); + +u32 alcor_read32(struct alcor_pci_priv *priv, unsigned int addr) +{ +	return readl(priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_read32); + +u32 alcor_read32be(struct alcor_pci_priv *priv, unsigned int addr) +{ +	return ioread32be(priv->iobase + addr); +} +EXPORT_SYMBOL_GPL(alcor_read32be); + +static int alcor_pci_find_cap_offset(struct alcor_pci_priv *priv, +				     struct pci_dev *pci) +{ +	int where; +	u8 val8; +	u32 val32; + +	where = ALCOR_CAP_START_OFFSET; +	pci_read_config_byte(pci, where, &val8); +	if (!val8) +		return 0; + +	where = (int)val8; +	while (1) { +		pci_read_config_dword(pci, where, &val32); +		if (val32 == 0xffffffff) { +			dev_dbg(priv->dev, "find_cap_offset invalid value %x.\n", +				val32); +			return 0; +		} + +		if ((val32 & 0xff) == 0x10) { +			dev_dbg(priv->dev, "pcie cap offset: %x\n", where); +			return where; +		} + +		if ((val32 & 0xff00) == 0x00) { +			dev_dbg(priv->dev, "pci_find_cap_offset invalid value %x.\n", +				val32); +			break; +		} +		where = (int)((val32 >> 8) & 0xff); +	} + +	return 0; +} + +static void alcor_pci_init_check_aspm(struct alcor_pci_priv *priv) +{ +	struct pci_dev *pci; +	int where; +	u32 val32; + +	priv->pdev_cap_off    = alcor_pci_find_cap_offset(priv, priv->pdev); +	priv->parent_cap_off = alcor_pci_find_cap_offset(priv, +							 priv->parent_pdev); + +	if ((priv->pdev_cap_off == 0) || (priv->parent_cap_off == 0)) { +		dev_dbg(priv->dev, "pci_cap_off: %x, parent_cap_off: %x\n", +			priv->pdev_cap_off, priv->parent_cap_off); +		return; +	} + +	/* link capability */ +	pci   = priv->pdev; +	where = priv->pdev_cap_off + ALCOR_PCIE_LINK_CAP_OFFSET; +	pci_read_config_dword(pci, where, &val32); +	priv->pdev_aspm_cap = (u8)(val32 >> 10) & 0x03; + +	pci   = priv->parent_pdev; +	where = priv->parent_cap_off + ALCOR_PCIE_LINK_CAP_OFFSET; +	pci_read_config_dword(pci, where, &val32); +	priv->parent_aspm_cap = (u8)(val32 >> 10) & 0x03; + +	if (priv->pdev_aspm_cap != priv->parent_aspm_cap) { +		u8 aspm_cap; + +		dev_dbg(priv->dev, "pdev_aspm_cap: %x, parent_aspm_cap: %x\n", +			priv->pdev_aspm_cap, priv->parent_aspm_cap); +		aspm_cap = priv->pdev_aspm_cap & priv->parent_aspm_cap; +		priv->pdev_aspm_cap    = aspm_cap; +		priv->parent_aspm_cap = aspm_cap; +	} + +	dev_dbg(priv->dev, "ext_config_dev_aspm: %x, pdev_aspm_cap: %x\n", +		priv->ext_config_dev_aspm, priv->pdev_aspm_cap); +	priv->ext_config_dev_aspm &= priv->pdev_aspm_cap; +} + +static void alcor_pci_aspm_ctrl(struct alcor_pci_priv *priv, u8 aspm_enable) +{ +	struct pci_dev *pci; +	u8 aspm_ctrl, i; +	int where; +	u32 val32; + +	if ((!priv->pdev_cap_off) || (!priv->parent_cap_off)) { +		dev_dbg(priv->dev, "pci_cap_off: %x, parent_cap_off: %x\n", +			priv->pdev_cap_off, priv->parent_cap_off); +		return; +	} + +	if (!priv->pdev_aspm_cap) +		return; + +	aspm_ctrl = 0; +	if (aspm_enable) { +		aspm_ctrl = priv->ext_config_dev_aspm; + +		if (!aspm_ctrl) { +			dev_dbg(priv->dev, "aspm_ctrl == 0\n"); +			return; +		} +	} + +	for (i = 0; i < 2; i++) { + +		if (i) { +			pci   = priv->parent_pdev; +			where = priv->parent_cap_off +				+ ALCOR_PCIE_LINK_CTRL_OFFSET; +		} else { +			pci   = priv->pdev; +			where = priv->pdev_cap_off +				+ ALCOR_PCIE_LINK_CTRL_OFFSET; +		} + +		pci_read_config_dword(pci, where, &val32); +		val32 &= (~0x03); +		val32 |= (aspm_ctrl & priv->pdev_aspm_cap); +		pci_write_config_byte(pci, where, (u8)val32); +	} + +} + +static inline void alcor_mask_sd_irqs(struct alcor_pci_priv *priv) +{ +	alcor_write32(priv, 0, AU6601_REG_INT_ENABLE); +} + +static inline void alcor_unmask_sd_irqs(struct alcor_pci_priv *priv) +{ +	alcor_write32(priv, AU6601_INT_CMD_MASK | AU6601_INT_DATA_MASK | +		  AU6601_INT_CARD_INSERT | AU6601_INT_CARD_REMOVE | +		  AU6601_INT_OVER_CURRENT_ERR, +		  AU6601_REG_INT_ENABLE); +} + +static inline void alcor_mask_ms_irqs(struct alcor_pci_priv *priv) +{ +	alcor_write32(priv, 0, AU6601_MS_INT_ENABLE); +} + +static inline void alcor_unmask_ms_irqs(struct alcor_pci_priv *priv) +{ +	alcor_write32(priv, 0x3d00fa, AU6601_MS_INT_ENABLE); +} + +static int alcor_pci_probe(struct pci_dev *pdev, +			   const struct pci_device_id *ent) +{ +	struct alcor_dev_cfg *cfg; +	struct alcor_pci_priv *priv; +	int ret, i, bar = 0; + +	cfg = (void *)ent->driver_data; + +	ret = pcim_enable_device(pdev); +	if (ret) +		return ret; + +	priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL); +	if (!priv) +		return -ENOMEM; + +	ret = ida_simple_get(&alcor_pci_idr, 0, 0, GFP_KERNEL); +	if (ret < 0) +		return ret; +	priv->id = ret; + +	priv->pdev = pdev; +	priv->parent_pdev = pdev->bus->self; +	priv->dev = &pdev->dev; +	priv->cfg = cfg; +	priv->irq = pdev->irq; + +	ret = pci_request_regions(pdev, DRV_NAME_ALCOR_PCI); +	if (ret) { +		dev_err(&pdev->dev, "Cannot request region\n"); +		return -ENOMEM; +	} + +	if (!(pci_resource_flags(pdev, bar) & IORESOURCE_MEM)) { +		dev_err(&pdev->dev, "BAR %d is not iomem. Aborting.\n", bar); +		ret = -ENODEV; +		goto error_release_regions; +	} + +	priv->iobase = pcim_iomap(pdev, bar, 0); +	if (!priv->iobase) { +		ret = -ENOMEM; +		goto error_release_regions; +	} + +	/* make sure irqs are disabled */ +	alcor_write32(priv, 0, AU6601_REG_INT_ENABLE); +	alcor_write32(priv, 0, AU6601_MS_INT_ENABLE); + +	ret = dma_set_mask_and_coherent(priv->dev, AU6601_SDMA_MASK); +	if (ret) { +		dev_err(priv->dev, "Failed to set DMA mask\n"); +		goto error_release_regions; +	} + +	pci_set_master(pdev); +	pci_set_drvdata(pdev, priv); +	alcor_pci_init_check_aspm(priv); + +	for (i = 0; i < ARRAY_SIZE(alcor_pci_cells); i++) { +		alcor_pci_cells[i].platform_data = priv; +		alcor_pci_cells[i].pdata_size = sizeof(*priv); +	} +	ret = mfd_add_devices(&pdev->dev, priv->id, alcor_pci_cells, +			ARRAY_SIZE(alcor_pci_cells), NULL, 0, NULL); +	if (ret < 0) +		goto error_release_regions; + +	alcor_pci_aspm_ctrl(priv, 0); + +	return 0; + +error_release_regions: +	pci_release_regions(pdev); +	return ret; +} + +static void alcor_pci_remove(struct pci_dev *pdev) +{ +	struct alcor_pci_priv *priv; + +	priv = pci_get_drvdata(pdev); + +	alcor_pci_aspm_ctrl(priv, 1); + +	mfd_remove_devices(&pdev->dev); + +	ida_simple_remove(&alcor_pci_idr, priv->id); + +	pci_release_regions(pdev); +	pci_set_drvdata(pdev, NULL); +} + +#ifdef CONFIG_PM_SLEEP +static int alcor_suspend(struct device *dev) +{ +	struct pci_dev *pdev = to_pci_dev(dev); +	struct alcor_pci_priv *priv = pci_get_drvdata(pdev); + +	alcor_pci_aspm_ctrl(priv, 1); +	return 0; +} + +static int alcor_resume(struct device *dev) +{ + +	struct pci_dev *pdev = to_pci_dev(dev); +	struct alcor_pci_priv *priv = pci_get_drvdata(pdev); + +	alcor_pci_aspm_ctrl(priv, 0); +	return 0; +} +#endif /* CONFIG_PM_SLEEP */ + +static SIMPLE_DEV_PM_OPS(alcor_pci_pm_ops, alcor_suspend, alcor_resume); + +static struct pci_driver alcor_driver = { +	.name	=	DRV_NAME_ALCOR_PCI, +	.id_table =	pci_ids, +	.probe	=	alcor_pci_probe, +	.remove =	alcor_pci_remove, +	.driver	=	{ +		.pm	= &alcor_pci_pm_ops +	}, +}; + +module_pci_driver(alcor_driver); + +MODULE_AUTHOR("Oleksij Rempel <linux@rempel-privat.de>"); +MODULE_DESCRIPTION("PCI driver for Alcor Micro AU6601 Secure Digital Host Controller Interface"); +MODULE_LICENSE("GPL"); diff --git a/drivers/misc/cardreader/rtsx_usb.c b/drivers/misc/cardreader/rtsx_usb.c index b97903ff1a72..f7a66f614085 100644 --- a/drivers/misc/cardreader/rtsx_usb.c +++ b/drivers/misc/cardreader/rtsx_usb.c @@ -723,8 +723,15 @@ static int rtsx_usb_suspend(struct usb_interface *intf, pm_message_t message)  	return 0;  } +static int rtsx_usb_resume_child(struct device *dev, void *data) +{ +	pm_request_resume(dev); +	return 0; +} +  static int rtsx_usb_resume(struct usb_interface *intf)  { +	device_for_each_child(&intf->dev, NULL, rtsx_usb_resume_child);  	return 0;  } @@ -734,6 +741,7 @@ static int rtsx_usb_reset_resume(struct usb_interface *intf)  		(struct rtsx_ucr *)usb_get_intfdata(intf);  	rtsx_usb_reset_chip(ucr); +	device_for_each_child(&intf->dev, NULL, rtsx_usb_resume_child);  	return 0;  } diff --git a/drivers/misc/cxl/pci.c b/drivers/misc/cxl/pci.c index b66d832d3233..c79ba1c699ad 100644 --- a/drivers/misc/cxl/pci.c +++ b/drivers/misc/cxl/pci.c @@ -1718,7 +1718,6 @@ int cxl_slot_is_switched(struct pci_dev *dev)  {  	struct device_node *np;  	int depth = 0; -	const __be32 *prop;  	if (!(np = pci_device_to_OF_node(dev))) {  		pr_err("cxl: np = NULL\n"); @@ -1727,8 +1726,7 @@ int cxl_slot_is_switched(struct pci_dev *dev)  	of_node_get(np);  	while (np) {  		np = of_get_next_parent(np); -		prop = of_get_property(np, "device_type", NULL); -		if (!prop || strcmp((char *)prop, "pciex")) +		if (!of_node_is_type(np, "pciex"))  			break;  		depth++;  	} diff --git a/drivers/misc/cxl/vphb.c b/drivers/misc/cxl/vphb.c index 7908633d9204..49da2f744bbf 100644 --- a/drivers/misc/cxl/vphb.c +++ b/drivers/misc/cxl/vphb.c @@ -11,17 +11,6 @@  #include <misc/cxl.h>  #include "cxl.h" -static int cxl_dma_set_mask(struct pci_dev *pdev, u64 dma_mask) -{ -	if (dma_mask < DMA_BIT_MASK(64)) { -		pr_info("%s only 64bit DMA supported on CXL", __func__); -		return -EIO; -	} - -	*(pdev->dev.dma_mask) = dma_mask; -	return 0; -} -  static int cxl_pci_probe_mode(struct pci_bus *bus)  {  	return PCI_PROBE_NORMAL; @@ -220,7 +209,6 @@ static struct pci_controller_ops cxl_pci_controller_ops =  	.reset_secondary_bus = cxl_pci_reset_secondary_bus,  	.setup_msi_irqs = cxl_setup_msi_irqs,  	.teardown_msi_irqs = cxl_teardown_msi_irqs, -	.dma_set_mask = cxl_dma_set_mask,  };  int cxl_pci_vphb_add(struct cxl_afu *afu) diff --git a/drivers/misc/eeprom/Kconfig b/drivers/misc/eeprom/Kconfig index fe7a1d27a017..a846faefa210 100644 --- a/drivers/misc/eeprom/Kconfig +++ b/drivers/misc/eeprom/Kconfig @@ -13,7 +13,7 @@ config EEPROM_AT24  	  ones like at24c64, 24lc02 or fm24c04:  	     24c00, 24c01, 24c02, spd (readonly 24c02), 24c04, 24c08, -	     24c16, 24c32, 24c64, 24c128, 24c256, 24c512, 24c1024 +	     24c16, 24c32, 24c64, 24c128, 24c256, 24c512, 24c1024, 24c2048  	  Unless you like data loss puzzles, always be sure that any chip  	  you configure as a 24c32 (32 kbit) or larger is NOT really a diff --git a/drivers/misc/eeprom/at24.c b/drivers/misc/eeprom/at24.c index 636ed7149793..ddfcf4ade7bf 100644 --- a/drivers/misc/eeprom/at24.c +++ b/drivers/misc/eeprom/at24.c @@ -156,6 +156,7 @@ AT24_CHIP_DATA(at24_data_24c128, 131072 / 8, AT24_FLAG_ADDR16);  AT24_CHIP_DATA(at24_data_24c256, 262144 / 8, AT24_FLAG_ADDR16);  AT24_CHIP_DATA(at24_data_24c512, 524288 / 8, AT24_FLAG_ADDR16);  AT24_CHIP_DATA(at24_data_24c1024, 1048576 / 8, AT24_FLAG_ADDR16); +AT24_CHIP_DATA(at24_data_24c2048, 2097152 / 8, AT24_FLAG_ADDR16);  /* identical to 24c08 ? */  AT24_CHIP_DATA(at24_data_INT3499, 8192 / 8, 0); @@ -182,6 +183,7 @@ static const struct i2c_device_id at24_ids[] = {  	{ "24c256",	(kernel_ulong_t)&at24_data_24c256 },  	{ "24c512",	(kernel_ulong_t)&at24_data_24c512 },  	{ "24c1024",	(kernel_ulong_t)&at24_data_24c1024 }, +	{ "24c2048",    (kernel_ulong_t)&at24_data_24c2048 },  	{ "at24",	0 },  	{ /* END OF LIST */ }  }; @@ -210,6 +212,7 @@ static const struct of_device_id at24_of_match[] = {  	{ .compatible = "atmel,24c256",		.data = &at24_data_24c256 },  	{ .compatible = "atmel,24c512",		.data = &at24_data_24c512 },  	{ .compatible = "atmel,24c1024",	.data = &at24_data_24c1024 }, +	{ .compatible = "atmel,24c2048",	.data = &at24_data_24c2048 },  	{ /* END OF LIST */ },  };  MODULE_DEVICE_TABLE(of, at24_of_match); diff --git a/drivers/misc/genwqe/card_debugfs.c b/drivers/misc/genwqe/card_debugfs.c index c6b82f09b3ba..7c713e01d198 100644 --- a/drivers/misc/genwqe/card_debugfs.c +++ b/drivers/misc/genwqe/card_debugfs.c @@ -33,19 +33,6 @@  #include "card_base.h"  #include "card_ddcb.h" -#define GENWQE_DEBUGFS_RO(_name, _showfn)				\ -	static int genwqe_debugfs_##_name##_open(struct inode *inode,	\ -						 struct file *file)	\ -	{								\ -		return single_open(file, _showfn, inode->i_private);	\ -	}								\ -	static const struct file_operations genwqe_##_name##_fops = {	\ -		.open = genwqe_debugfs_##_name##_open,			\ -		.read = seq_read,					\ -		.llseek = seq_lseek,					\ -		.release = single_release,				\ -	} -  static void dbg_uidn_show(struct seq_file *s, struct genwqe_reg *regs,  			  int entries)  { @@ -87,26 +74,26 @@ static int curr_dbg_uidn_show(struct seq_file *s, void *unused, int uid)  	return 0;  } -static int genwqe_curr_dbg_uid0_show(struct seq_file *s, void *unused) +static int curr_dbg_uid0_show(struct seq_file *s, void *unused)  {  	return curr_dbg_uidn_show(s, unused, 0);  } -GENWQE_DEBUGFS_RO(curr_dbg_uid0, genwqe_curr_dbg_uid0_show); +DEFINE_SHOW_ATTRIBUTE(curr_dbg_uid0); -static int genwqe_curr_dbg_uid1_show(struct seq_file *s, void *unused) +static int curr_dbg_uid1_show(struct seq_file *s, void *unused)  {  	return curr_dbg_uidn_show(s, unused, 1);  } -GENWQE_DEBUGFS_RO(curr_dbg_uid1, genwqe_curr_dbg_uid1_show); +DEFINE_SHOW_ATTRIBUTE(curr_dbg_uid1); -static int genwqe_curr_dbg_uid2_show(struct seq_file *s, void *unused) +static int curr_dbg_uid2_show(struct seq_file *s, void *unused)  {  	return curr_dbg_uidn_show(s, unused, 2);  } -GENWQE_DEBUGFS_RO(curr_dbg_uid2, genwqe_curr_dbg_uid2_show); +DEFINE_SHOW_ATTRIBUTE(curr_dbg_uid2);  static int prev_dbg_uidn_show(struct seq_file *s, void *unused, int uid)  { @@ -116,28 +103,28 @@ static int prev_dbg_uidn_show(struct seq_file *s, void *unused, int uid)  	return 0;  } -static int genwqe_prev_dbg_uid0_show(struct seq_file *s, void *unused) +static int prev_dbg_uid0_show(struct seq_file *s, void *unused)  {  	return prev_dbg_uidn_show(s, unused, 0);  } -GENWQE_DEBUGFS_RO(prev_dbg_uid0, genwqe_prev_dbg_uid0_show); +DEFINE_SHOW_ATTRIBUTE(prev_dbg_uid0); -static int genwqe_prev_dbg_uid1_show(struct seq_file *s, void *unused) +static int prev_dbg_uid1_show(struct seq_file *s, void *unused)  {  	return prev_dbg_uidn_show(s, unused, 1);  } -GENWQE_DEBUGFS_RO(prev_dbg_uid1, genwqe_prev_dbg_uid1_show); +DEFINE_SHOW_ATTRIBUTE(prev_dbg_uid1); -static int genwqe_prev_dbg_uid2_show(struct seq_file *s, void *unused) +static int prev_dbg_uid2_show(struct seq_file *s, void *unused)  {  	return prev_dbg_uidn_show(s, unused, 2);  } -GENWQE_DEBUGFS_RO(prev_dbg_uid2, genwqe_prev_dbg_uid2_show); +DEFINE_SHOW_ATTRIBUTE(prev_dbg_uid2); -static int genwqe_curr_regs_show(struct seq_file *s, void *unused) +static int curr_regs_show(struct seq_file *s, void *unused)  {  	struct genwqe_dev *cd = s->private;  	unsigned int i; @@ -164,9 +151,9 @@ static int genwqe_curr_regs_show(struct seq_file *s, void *unused)  	return 0;  } -GENWQE_DEBUGFS_RO(curr_regs, genwqe_curr_regs_show); +DEFINE_SHOW_ATTRIBUTE(curr_regs); -static int genwqe_prev_regs_show(struct seq_file *s, void *unused) +static int prev_regs_show(struct seq_file *s, void *unused)  {  	struct genwqe_dev *cd = s->private;  	unsigned int i; @@ -188,9 +175,9 @@ static int genwqe_prev_regs_show(struct seq_file *s, void *unused)  	return 0;  } -GENWQE_DEBUGFS_RO(prev_regs, genwqe_prev_regs_show); +DEFINE_SHOW_ATTRIBUTE(prev_regs); -static int genwqe_jtimer_show(struct seq_file *s, void *unused) +static int jtimer_show(struct seq_file *s, void *unused)  {  	struct genwqe_dev *cd = s->private;  	unsigned int vf_num; @@ -209,9 +196,9 @@ static int genwqe_jtimer_show(struct seq_file *s, void *unused)  	return 0;  } -GENWQE_DEBUGFS_RO(jtimer, genwqe_jtimer_show); +DEFINE_SHOW_ATTRIBUTE(jtimer); -static int genwqe_queue_working_time_show(struct seq_file *s, void *unused) +static int queue_working_time_show(struct seq_file *s, void *unused)  {  	struct genwqe_dev *cd = s->private;  	unsigned int vf_num; @@ -227,9 +214,9 @@ static int genwqe_queue_working_time_show(struct seq_file *s, void *unused)  	return 0;  } -GENWQE_DEBUGFS_RO(queue_working_time, genwqe_queue_working_time_show); +DEFINE_SHOW_ATTRIBUTE(queue_working_time); -static int genwqe_ddcb_info_show(struct seq_file *s, void *unused) +static int ddcb_info_show(struct seq_file *s, void *unused)  {  	struct genwqe_dev *cd = s->private;  	unsigned int i; @@ -300,9 +287,9 @@ static int genwqe_ddcb_info_show(struct seq_file *s, void *unused)  	return 0;  } -GENWQE_DEBUGFS_RO(ddcb_info, genwqe_ddcb_info_show); +DEFINE_SHOW_ATTRIBUTE(ddcb_info); -static int genwqe_info_show(struct seq_file *s, void *unused) +static int info_show(struct seq_file *s, void *unused)  {  	struct genwqe_dev *cd = s->private;  	u64 app_id, slu_id, bitstream = -1; @@ -335,7 +322,7 @@ static int genwqe_info_show(struct seq_file *s, void *unused)  	return 0;  } -GENWQE_DEBUGFS_RO(info, genwqe_info_show); +DEFINE_SHOW_ATTRIBUTE(info);  int genwqe_init_debugfs(struct genwqe_dev *cd)  { @@ -356,14 +343,14 @@ int genwqe_init_debugfs(struct genwqe_dev *cd)  	/* non privileged interfaces are done here */  	file = debugfs_create_file("ddcb_info", S_IRUGO, root, cd, -				   &genwqe_ddcb_info_fops); +				   &ddcb_info_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("info", S_IRUGO, root, cd, -				   &genwqe_info_fops); +				   &info_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1; @@ -396,56 +383,56 @@ int genwqe_init_debugfs(struct genwqe_dev *cd)  	}  	file = debugfs_create_file("curr_regs", S_IRUGO, root, cd, -				   &genwqe_curr_regs_fops); +				   &curr_regs_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("curr_dbg_uid0", S_IRUGO, root, cd, -				   &genwqe_curr_dbg_uid0_fops); +				   &curr_dbg_uid0_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("curr_dbg_uid1", S_IRUGO, root, cd, -				   &genwqe_curr_dbg_uid1_fops); +				   &curr_dbg_uid1_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("curr_dbg_uid2", S_IRUGO, root, cd, -				   &genwqe_curr_dbg_uid2_fops); +				   &curr_dbg_uid2_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("prev_regs", S_IRUGO, root, cd, -				   &genwqe_prev_regs_fops); +				   &prev_regs_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("prev_dbg_uid0", S_IRUGO, root, cd, -				   &genwqe_prev_dbg_uid0_fops); +				   &prev_dbg_uid0_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("prev_dbg_uid1", S_IRUGO, root, cd, -				   &genwqe_prev_dbg_uid1_fops); +				   &prev_dbg_uid1_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("prev_dbg_uid2", S_IRUGO, root, cd, -				   &genwqe_prev_dbg_uid2_fops); +				   &prev_dbg_uid2_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1; @@ -463,14 +450,14 @@ int genwqe_init_debugfs(struct genwqe_dev *cd)  	}  	file = debugfs_create_file("jobtimer", S_IRUGO, root, cd, -				   &genwqe_jtimer_fops); +				   &jtimer_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1;  	}  	file = debugfs_create_file("queue_working_time", S_IRUGO, root, cd, -				   &genwqe_queue_working_time_fops); +				   &queue_working_time_fops);  	if (!file) {  		ret = -ENOMEM;  		goto err1; diff --git a/drivers/misc/genwqe/card_utils.c b/drivers/misc/genwqe/card_utils.c index 3fcb9a2fe1c9..efe2fb72d54b 100644 --- a/drivers/misc/genwqe/card_utils.c +++ b/drivers/misc/genwqe/card_utils.c @@ -215,7 +215,7 @@ u32 genwqe_crc32(u8 *buff, size_t len, u32 init)  void *__genwqe_alloc_consistent(struct genwqe_dev *cd, size_t size,  			       dma_addr_t *dma_handle)  { -	if (get_order(size) > MAX_ORDER) +	if (get_order(size) >= MAX_ORDER)  		return NULL;  	return dma_zalloc_coherent(&cd->pci_dev->dev, size, dma_handle, diff --git a/drivers/misc/mei/Makefile b/drivers/misc/mei/Makefile index cd6825afa8e1..d9215fc4e499 100644 --- a/drivers/misc/mei/Makefile +++ b/drivers/misc/mei/Makefile @@ -9,6 +9,7 @@ mei-objs += hbm.o  mei-objs += interrupt.o  mei-objs += client.o  mei-objs += main.o +mei-objs += dma-ring.o  mei-objs += bus.o  mei-objs += bus-fixup.o  mei-$(CONFIG_DEBUG_FS) += debugfs.o diff --git a/drivers/misc/mei/client.c b/drivers/misc/mei/client.c index ebdcf0b450e2..1fc8ea0f519b 100644 --- a/drivers/misc/mei/client.c +++ b/drivers/misc/mei/client.c @@ -318,23 +318,6 @@ void mei_me_cl_rm_all(struct mei_device *dev)  }  /** - * mei_cl_cmp_id - tells if the clients are the same - * - * @cl1: host client 1 - * @cl2: host client 2 - * - * Return: true  - if the clients has same host and me ids - *         false - otherwise - */ -static inline bool mei_cl_cmp_id(const struct mei_cl *cl1, -				const struct mei_cl *cl2) -{ -	return cl1 && cl2 && -		(cl1->host_client_id == cl2->host_client_id) && -		(mei_cl_me_id(cl1) == mei_cl_me_id(cl2)); -} - -/**   * mei_io_cb_free - free mei_cb_private related memory   *   * @cb: mei callback struct @@ -418,7 +401,7 @@ static void mei_io_list_flush_cl(struct list_head *head,  	struct mei_cl_cb *cb, *next;  	list_for_each_entry_safe(cb, next, head, list) { -		if (mei_cl_cmp_id(cl, cb->cl)) +		if (cl == cb->cl)  			list_del_init(&cb->list);  	}  } @@ -435,7 +418,7 @@ static void mei_io_tx_list_free_cl(struct list_head *head,  	struct mei_cl_cb *cb, *next;  	list_for_each_entry_safe(cb, next, head, list) { -		if (mei_cl_cmp_id(cl, cb->cl)) +		if (cl == cb->cl)  			mei_tx_cb_dequeue(cb);  	}  } @@ -478,7 +461,7 @@ struct mei_cl_cb *mei_cl_alloc_cb(struct mei_cl *cl, size_t length,  	if (length == 0)  		return cb; -	cb->buf.data = kmalloc(length, GFP_KERNEL); +	cb->buf.data = kmalloc(roundup(length, MEI_SLOT_SIZE), GFP_KERNEL);  	if (!cb->buf.data) {  		mei_io_cb_free(cb);  		return NULL; @@ -1374,7 +1357,9 @@ int mei_cl_notify_request(struct mei_cl *cl,  	mutex_unlock(&dev->device_lock);  	wait_event_timeout(cl->wait, -			   cl->notify_en == request || !mei_cl_is_connected(cl), +			   cl->notify_en == request || +			   cl->status || +			   !mei_cl_is_connected(cl),  			   mei_secs_to_jiffies(MEI_CL_CONNECT_TIMEOUT));  	mutex_lock(&dev->device_lock); @@ -1573,10 +1558,13 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	struct mei_msg_hdr mei_hdr;  	size_t hdr_len = sizeof(mei_hdr);  	size_t len; -	size_t hbuf_len; +	size_t hbuf_len, dr_len;  	int hbuf_slots; +	u32 dr_slots; +	u32 dma_len;  	int rets;  	bool first_chunk; +	const void *data;  	if (WARN_ON(!cl || !cl->dev))  		return -ENODEV; @@ -1597,6 +1585,7 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	}  	len = buf->size - cb->buf_idx; +	data = buf->data + cb->buf_idx;  	hbuf_slots = mei_hbuf_empty_slots(dev);  	if (hbuf_slots < 0) {  		rets = -EOVERFLOW; @@ -1604,6 +1593,8 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	}  	hbuf_len = mei_slots2data(hbuf_slots); +	dr_slots = mei_dma_ring_empty_slots(dev); +	dr_len = mei_slots2data(dr_slots);  	mei_msg_hdr_init(&mei_hdr, cb); @@ -1614,23 +1605,33 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	if (len + hdr_len <= hbuf_len) {  		mei_hdr.length = len;  		mei_hdr.msg_complete = 1; +	} else if (dr_slots && hbuf_len >= hdr_len + sizeof(dma_len)) { +		mei_hdr.dma_ring = 1; +		if (len > dr_len) +			len = dr_len; +		else +			mei_hdr.msg_complete = 1; + +		mei_hdr.length = sizeof(dma_len); +		dma_len = len; +		data = &dma_len;  	} else if ((u32)hbuf_slots == mei_hbuf_depth(dev)) { -		mei_hdr.length = hbuf_len - hdr_len; +		len = hbuf_len - hdr_len; +		mei_hdr.length = len;  	} else {  		return 0;  	} -	cl_dbg(dev, cl, "buf: size = %zu idx = %zu\n", -			cb->buf.size, cb->buf_idx); +	if (mei_hdr.dma_ring) +		mei_dma_ring_write(dev, buf->data + cb->buf_idx, len); -	rets = mei_write_message(dev, &mei_hdr, hdr_len, -				 buf->data + cb->buf_idx, mei_hdr.length); +	rets = mei_write_message(dev, &mei_hdr, hdr_len, data, mei_hdr.length);  	if (rets)  		goto err;  	cl->status = 0;  	cl->writing_state = MEI_WRITING; -	cb->buf_idx += mei_hdr.length; +	cb->buf_idx += len;  	if (first_chunk) {  		if (mei_cl_tx_flow_ctrl_creds_reduce(cl)) { @@ -1665,11 +1666,13 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  	struct mei_msg_data *buf;  	struct mei_msg_hdr mei_hdr;  	size_t hdr_len = sizeof(mei_hdr); -	size_t len; -	size_t hbuf_len; +	size_t len, hbuf_len, dr_len;  	int hbuf_slots; +	u32 dr_slots; +	u32 dma_len;  	ssize_t rets;  	bool blocking; +	const void *data;  	if (WARN_ON(!cl || !cl->dev))  		return -ENODEV; @@ -1681,10 +1684,12 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  	buf = &cb->buf;  	len = buf->size; -	blocking = cb->blocking;  	cl_dbg(dev, cl, "len=%zd\n", len); +	blocking = cb->blocking; +	data = buf->data; +  	rets = pm_runtime_get(dev->dev);  	if (rets < 0 && rets != -EINPROGRESS) {  		pm_runtime_put_noidle(dev->dev); @@ -1721,16 +1726,32 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  	}  	hbuf_len = mei_slots2data(hbuf_slots); +	dr_slots = mei_dma_ring_empty_slots(dev); +	dr_len =  mei_slots2data(dr_slots);  	if (len + hdr_len <= hbuf_len) {  		mei_hdr.length = len;  		mei_hdr.msg_complete = 1; +	} else if (dr_slots && hbuf_len >= hdr_len + sizeof(dma_len)) { +		mei_hdr.dma_ring = 1; +		if (len > dr_len) +			len = dr_len; +		else +			mei_hdr.msg_complete = 1; + +		mei_hdr.length = sizeof(dma_len); +		dma_len = len; +		data = &dma_len;  	} else { -		mei_hdr.length = hbuf_len - hdr_len; +		len = hbuf_len - hdr_len; +		mei_hdr.length = len;  	} +	if (mei_hdr.dma_ring) +		mei_dma_ring_write(dev, buf->data, len); +  	rets = mei_write_message(dev, &mei_hdr, hdr_len, -				 buf->data, mei_hdr.length); +				 data, mei_hdr.length);  	if (rets)  		goto err; @@ -1739,7 +1760,9 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  		goto err;  	cl->writing_state = MEI_WRITING; -	cb->buf_idx = mei_hdr.length; +	cb->buf_idx = len; +	/* restore return value */ +	len = buf->size;  out:  	if (mei_hdr.msg_complete) diff --git a/drivers/misc/mei/dma-ring.c b/drivers/misc/mei/dma-ring.c new file mode 100644 index 000000000000..795641b82181 --- /dev/null +++ b/drivers/misc/mei/dma-ring.c @@ -0,0 +1,269 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Copyright(c) 2016 - 2018 Intel Corporation. All rights reserved. + */ +#include <linux/dma-mapping.h> +#include <linux/mei.h> + +#include "mei_dev.h" + +/** + * mei_dmam_dscr_alloc() - allocate a managed coherent buffer + *     for the dma descriptor + * @dev: mei_device + * @dscr: dma descriptor + * + * Return: + * * 0       - on success or zero allocation request + * * -EINVAL - if size is not power of 2 + * * -ENOMEM - of allocation has failed + */ +static int mei_dmam_dscr_alloc(struct mei_device *dev, +			       struct mei_dma_dscr *dscr) +{ +	if (!dscr->size) +		return 0; + +	if (WARN_ON(!is_power_of_2(dscr->size))) +		return -EINVAL; + +	if (dscr->vaddr) +		return 0; + +	dscr->vaddr = dmam_alloc_coherent(dev->dev, dscr->size, &dscr->daddr, +					  GFP_KERNEL); +	if (!dscr->vaddr) +		return -ENOMEM; + +	return 0; +} + +/** + * mei_dmam_dscr_free() - free a managed coherent buffer + *     from the dma descriptor + * @dev: mei_device + * @dscr: dma descriptor + */ +static void mei_dmam_dscr_free(struct mei_device *dev, +			       struct mei_dma_dscr *dscr) +{ +	if (!dscr->vaddr) +		return; + +	dmam_free_coherent(dev->dev, dscr->size, dscr->vaddr, dscr->daddr); +	dscr->vaddr = NULL; +} + +/** + * mei_dmam_ring_free() - free dma ring buffers + * @dev: mei device + */ +void mei_dmam_ring_free(struct mei_device *dev) +{ +	int i; + +	for (i = 0; i < DMA_DSCR_NUM; i++) +		mei_dmam_dscr_free(dev, &dev->dr_dscr[i]); +} + +/** + * mei_dmam_ring_alloc() - allocate dma ring buffers + * @dev: mei device + * + * Return: -ENOMEM on allocation failure 0 otherwise + */ +int mei_dmam_ring_alloc(struct mei_device *dev) +{ +	int i; + +	for (i = 0; i < DMA_DSCR_NUM; i++) +		if (mei_dmam_dscr_alloc(dev, &dev->dr_dscr[i])) +			goto err; + +	return 0; + +err: +	mei_dmam_ring_free(dev); +	return -ENOMEM; +} + +/** + * mei_dma_ring_is_allocated() - check if dma ring is allocated + * @dev: mei device + * + * Return: true if dma ring is allocated + */ +bool mei_dma_ring_is_allocated(struct mei_device *dev) +{ +	return !!dev->dr_dscr[DMA_DSCR_HOST].vaddr; +} + +static inline +struct hbm_dma_ring_ctrl *mei_dma_ring_ctrl(struct mei_device *dev) +{ +	return (struct hbm_dma_ring_ctrl *)dev->dr_dscr[DMA_DSCR_CTRL].vaddr; +} + +/** + * mei_dma_ring_reset() - reset the dma control block + * @dev: mei device + */ +void mei_dma_ring_reset(struct mei_device *dev) +{ +	struct hbm_dma_ring_ctrl *ctrl = mei_dma_ring_ctrl(dev); + +	if (!ctrl) +		return; + +	memset(ctrl, 0, sizeof(*ctrl)); +} + +/** + * mei_dma_copy_from() - copy from dma ring into buffer + * @dev: mei device + * @buf: data buffer + * @offset: offset in slots. + * @n: number of slots to copy. + */ +static size_t mei_dma_copy_from(struct mei_device *dev, unsigned char *buf, +				u32 offset, u32 n) +{ +	unsigned char *dbuf = dev->dr_dscr[DMA_DSCR_DEVICE].vaddr; + +	size_t b_offset = offset << 2; +	size_t b_n = n << 2; + +	memcpy(buf, dbuf + b_offset, b_n); + +	return b_n; +} + +/** + * mei_dma_copy_to() - copy to a buffer to the dma ring + * @dev: mei device + * @buf: data buffer + * @offset: offset in slots. + * @n: number of slots to copy. + */ +static size_t mei_dma_copy_to(struct mei_device *dev, unsigned char *buf, +			      u32 offset, u32 n) +{ +	unsigned char *hbuf = dev->dr_dscr[DMA_DSCR_HOST].vaddr; + +	size_t b_offset = offset << 2; +	size_t b_n = n << 2; + +	memcpy(hbuf + b_offset, buf, b_n); + +	return b_n; +} + +/** + * mei_dma_ring_read() - read data from the ring + * @dev: mei device + * @buf: buffer to read into: may be NULL in case of droping the data. + * @len: length to read. + */ +void mei_dma_ring_read(struct mei_device *dev, unsigned char *buf, u32 len) +{ +	struct hbm_dma_ring_ctrl *ctrl = mei_dma_ring_ctrl(dev); +	u32 dbuf_depth; +	u32 rd_idx, rem, slots; + +	if (WARN_ON(!ctrl)) +		return; + +	dev_dbg(dev->dev, "reading from dma %u bytes\n", len); + +	if (!len) +		return; + +	dbuf_depth = dev->dr_dscr[DMA_DSCR_DEVICE].size >> 2; +	rd_idx = READ_ONCE(ctrl->dbuf_rd_idx) & (dbuf_depth - 1); +	slots = mei_data2slots(len); + +	/* if buf is NULL we drop the packet by advancing the pointer.*/ +	if (!buf) +		goto out; + +	if (rd_idx + slots > dbuf_depth) { +		buf += mei_dma_copy_from(dev, buf, rd_idx, dbuf_depth - rd_idx); +		rem = slots - (dbuf_depth - rd_idx); +		rd_idx = 0; +	} else { +		rem = slots; +	} + +	mei_dma_copy_from(dev, buf, rd_idx, rem); +out: +	WRITE_ONCE(ctrl->dbuf_rd_idx, ctrl->dbuf_rd_idx + slots); +} + +static inline u32 mei_dma_ring_hbuf_depth(struct mei_device *dev) +{ +	return dev->dr_dscr[DMA_DSCR_HOST].size >> 2; +} + +/** + * mei_dma_ring_empty_slots() - calaculate number of empty slots in dma ring + * @dev: mei_device + * + * Return: number of empty slots + */ +u32 mei_dma_ring_empty_slots(struct mei_device *dev) +{ +	struct hbm_dma_ring_ctrl *ctrl = mei_dma_ring_ctrl(dev); +	u32 wr_idx, rd_idx, hbuf_depth, empty; + +	if (!mei_dma_ring_is_allocated(dev)) +		return 0; + +	if (WARN_ON(!ctrl)) +		return 0; + +	/* easier to work in slots */ +	hbuf_depth = mei_dma_ring_hbuf_depth(dev); +	rd_idx = READ_ONCE(ctrl->hbuf_rd_idx); +	wr_idx = READ_ONCE(ctrl->hbuf_wr_idx); + +	if (rd_idx > wr_idx) +		empty = rd_idx - wr_idx; +	else +		empty = hbuf_depth - (wr_idx - rd_idx); + +	return empty; +} + +/** + * mei_dma_ring_write - write data to dma ring host buffer + * + * @dev: mei_device + * @buf: data will be written + * @len: data length + */ +void mei_dma_ring_write(struct mei_device *dev, unsigned char *buf, u32 len) +{ +	struct hbm_dma_ring_ctrl *ctrl = mei_dma_ring_ctrl(dev); +	u32 hbuf_depth; +	u32 wr_idx, rem, slots; + +	if (WARN_ON(!ctrl)) +		return; + +	dev_dbg(dev->dev, "writing to dma %u bytes\n", len); +	hbuf_depth = mei_dma_ring_hbuf_depth(dev); +	wr_idx = READ_ONCE(ctrl->hbuf_wr_idx) & (hbuf_depth - 1); +	slots = mei_data2slots(len); + +	if (wr_idx + slots > hbuf_depth) { +		buf += mei_dma_copy_to(dev, buf, wr_idx, hbuf_depth - wr_idx); +		rem = slots - (hbuf_depth - wr_idx); +		wr_idx = 0; +	} else { +		rem = slots; +	} + +	mei_dma_copy_to(dev, buf, wr_idx, rem); + +	WRITE_ONCE(ctrl->hbuf_wr_idx, ctrl->hbuf_wr_idx + slots); +} diff --git a/drivers/misc/mei/hbm.c b/drivers/misc/mei/hbm.c index e56f3e72d57a..78c26cebf5d4 100644 --- a/drivers/misc/mei/hbm.c +++ b/drivers/misc/mei/hbm.c @@ -65,6 +65,7 @@ const char *mei_hbm_state_str(enum mei_hbm_state state)  	MEI_HBM_STATE(IDLE);  	MEI_HBM_STATE(STARTING);  	MEI_HBM_STATE(STARTED); +	MEI_HBM_STATE(DR_SETUP);  	MEI_HBM_STATE(ENUM_CLIENTS);  	MEI_HBM_STATE(CLIENT_PROPERTIES);  	MEI_HBM_STATE(STOPPED); @@ -296,6 +297,48 @@ int mei_hbm_start_req(struct mei_device *dev)  }  /** + * mei_hbm_dma_setup_req() - setup DMA request + * @dev: the device structure + * + * Return: 0 on success and < 0 on failure + */ +static int mei_hbm_dma_setup_req(struct mei_device *dev) +{ +	struct mei_msg_hdr mei_hdr; +	struct hbm_dma_setup_request req; +	const size_t len = sizeof(struct hbm_dma_setup_request); +	unsigned int i; +	int ret; + +	mei_hbm_hdr(&mei_hdr, len); + +	memset(&req, 0, len); +	req.hbm_cmd = MEI_HBM_DMA_SETUP_REQ_CMD; +	for (i = 0; i < DMA_DSCR_NUM; i++) { +		phys_addr_t paddr; + +		paddr = dev->dr_dscr[i].daddr; +		req.dma_dscr[i].addr_hi = upper_32_bits(paddr); +		req.dma_dscr[i].addr_lo = lower_32_bits(paddr); +		req.dma_dscr[i].size = dev->dr_dscr[i].size; +	} + +	mei_dma_ring_reset(dev); + +	ret = mei_hbm_write_message(dev, &mei_hdr, &req); +	if (ret) { +		dev_err(dev->dev, "dma setup request write failed: ret = %d.\n", +			ret); +		return ret; +	} + +	dev->hbm_state = MEI_HBM_DR_SETUP; +	dev->init_clients_timer = MEI_CLIENTS_INIT_TIMEOUT; +	mei_schedule_stall_timer(dev); +	return 0; +} + +/**   * mei_hbm_enum_clients_req - sends enumeration client request message.   *   * @dev: the device structure @@ -1044,6 +1087,7 @@ int mei_hbm_dispatch(struct mei_device *dev, struct mei_msg_hdr *hdr)  	struct hbm_host_version_response *version_res;  	struct hbm_props_response *props_res;  	struct hbm_host_enum_response *enum_res; +	struct hbm_dma_setup_response *dma_setup_res;  	struct hbm_add_client_request *add_cl_req;  	int ret; @@ -1108,14 +1152,52 @@ int mei_hbm_dispatch(struct mei_device *dev, struct mei_msg_hdr *hdr)  			return -EPROTO;  		} -		if (mei_hbm_enum_clients_req(dev)) { -			dev_err(dev->dev, "hbm: start: failed to send enumeration request\n"); -			return -EIO; +		if (dev->hbm_f_dr_supported) { +			if (mei_dmam_ring_alloc(dev)) +				dev_info(dev->dev, "running w/o dma ring\n"); +			if (mei_dma_ring_is_allocated(dev)) { +				if (mei_hbm_dma_setup_req(dev)) +					return -EIO; + +				wake_up(&dev->wait_hbm_start); +				break; +			}  		} +		dev->hbm_f_dr_supported = 0; +		mei_dmam_ring_free(dev); + +		if (mei_hbm_enum_clients_req(dev)) +			return -EIO; +  		wake_up(&dev->wait_hbm_start);  		break; +	case MEI_HBM_DMA_SETUP_RES_CMD: +		dev_dbg(dev->dev, "hbm: dma setup response: message received.\n"); + +		dev->init_clients_timer = 0; + +		if (dev->hbm_state != MEI_HBM_DR_SETUP) { +			dev_err(dev->dev, "hbm: dma setup response: state mismatch, [%d, %d]\n", +				dev->dev_state, dev->hbm_state); +			return -EPROTO; +		} + +		dma_setup_res = (struct hbm_dma_setup_response *)mei_msg; + +		if (dma_setup_res->status) { +			dev_info(dev->dev, "hbm: dma setup response: failure = %d %s\n", +				 dma_setup_res->status, +				 mei_hbm_status_str(dma_setup_res->status)); +			dev->hbm_f_dr_supported = 0; +			mei_dmam_ring_free(dev); +		} + +		if (mei_hbm_enum_clients_req(dev)) +			return -EIO; +		break; +  	case CLIENT_CONNECT_RES_CMD:  		dev_dbg(dev->dev, "hbm: client connect response: message received.\n");  		mei_hbm_cl_res(dev, cl_cmd, MEI_FOP_CONNECT); @@ -1271,8 +1353,8 @@ int mei_hbm_dispatch(struct mei_device *dev, struct mei_msg_hdr *hdr)  		break;  	default: -		BUG(); -		break; +		WARN(1, "hbm: wrong command %d\n", mei_msg->hbm_cmd); +		return -EPROTO;  	}  	return 0; diff --git a/drivers/misc/mei/hbm.h b/drivers/misc/mei/hbm.h index a2025a5083a3..0171a7e79bab 100644 --- a/drivers/misc/mei/hbm.h +++ b/drivers/misc/mei/hbm.h @@ -26,6 +26,7 @@ struct mei_cl;   *   * @MEI_HBM_IDLE : protocol not started   * @MEI_HBM_STARTING : start request message was sent + * @MEI_HBM_DR_SETUP : dma ring setup request message was sent   * @MEI_HBM_ENUM_CLIENTS : enumeration request was sent   * @MEI_HBM_CLIENT_PROPERTIES : acquiring clients properties   * @MEI_HBM_STARTED : enumeration was completed @@ -34,6 +35,7 @@ struct mei_cl;  enum mei_hbm_state {  	MEI_HBM_IDLE = 0,  	MEI_HBM_STARTING, +	MEI_HBM_DR_SETUP,  	MEI_HBM_ENUM_CLIENTS,  	MEI_HBM_CLIENT_PROPERTIES,  	MEI_HBM_STARTED, diff --git a/drivers/misc/mei/hw-me.c b/drivers/misc/mei/hw-me.c index 0759c3a668de..3fbbadfa2ae1 100644 --- a/drivers/misc/mei/hw-me.c +++ b/drivers/misc/mei/hw-me.c @@ -1471,15 +1471,21 @@ struct mei_device *mei_me_dev_init(struct pci_dev *pdev,  {  	struct mei_device *dev;  	struct mei_me_hw *hw; +	int i;  	dev = devm_kzalloc(&pdev->dev, sizeof(struct mei_device) +  			   sizeof(struct mei_me_hw), GFP_KERNEL);  	if (!dev)  		return NULL; +  	hw = to_me_hw(dev); +	for (i = 0; i < DMA_DSCR_NUM; i++) +		dev->dr_dscr[i].size = cfg->dma_size[i]; +  	mei_device_init(dev, &pdev->dev, &mei_me_hw_ops);  	hw->cfg = cfg; +  	return dev;  } diff --git a/drivers/misc/mei/hw.h b/drivers/misc/mei/hw.h index 65655925791a..2b7f7677f8cc 100644 --- a/drivers/misc/mei/hw.h +++ b/drivers/misc/mei/hw.h @@ -35,7 +35,7 @@  /*   * MEI Version   */ -#define HBM_MINOR_VERSION                   0 +#define HBM_MINOR_VERSION                   1  #define HBM_MAJOR_VERSION                   2  /* @@ -206,6 +206,7 @@ enum  mei_cl_disconnect_status {   * @dma_ring: message is on dma ring   * @internal: message is internal   * @msg_complete: last packet of the message + * @extension: extension of the header   */  struct mei_msg_hdr {  	u32 me_addr:8; @@ -215,8 +216,11 @@ struct mei_msg_hdr {  	u32 dma_ring:1;  	u32 internal:1;  	u32 msg_complete:1; +	u32 extension[0];  } __packed; +#define MEI_MSG_HDR_MAX 2 +  struct mei_bus_message {  	u8 hbm_cmd;  	u8 data[0]; @@ -512,4 +516,27 @@ struct hbm_dma_setup_response {  	u8 reserved[2];  } __packed; +/** + * struct mei_dma_ring_ctrl - dma ring control block + * + * @hbuf_wr_idx: host circular buffer write index in slots + * @reserved1: reserved for alignment + * @hbuf_rd_idx: host circular buffer read index in slots + * @reserved2: reserved for alignment + * @dbuf_wr_idx: device circular buffer write index in slots + * @reserved3: reserved for alignment + * @dbuf_rd_idx: device circular buffer read index in slots + * @reserved4: reserved for alignment + */ +struct hbm_dma_ring_ctrl { +	u32 hbuf_wr_idx; +	u32 reserved1; +	u32 hbuf_rd_idx; +	u32 reserved2; +	u32 dbuf_wr_idx; +	u32 reserved3; +	u32 dbuf_rd_idx; +	u32 reserved4; +} __packed; +  #endif diff --git a/drivers/misc/mei/init.c b/drivers/misc/mei/init.c index 4888ebc076b7..eb026e2a0537 100644 --- a/drivers/misc/mei/init.c +++ b/drivers/misc/mei/init.c @@ -151,7 +151,7 @@ int mei_reset(struct mei_device *dev)  	mei_hbm_reset(dev); -	dev->rd_msg_hdr = 0; +	memset(dev->rd_msg_hdr, 0, sizeof(dev->rd_msg_hdr));  	if (ret) {  		dev_err(dev->dev, "hw_reset failed ret = %d\n", ret); diff --git a/drivers/misc/mei/interrupt.c b/drivers/misc/mei/interrupt.c index 5a661cbdf2ae..055c2d89b310 100644 --- a/drivers/misc/mei/interrupt.c +++ b/drivers/misc/mei/interrupt.c @@ -75,6 +75,8 @@ static inline int mei_cl_hbm_equal(struct mei_cl *cl,   */  static void mei_irq_discard_msg(struct mei_device *dev, struct mei_msg_hdr *hdr)  { +	if (hdr->dma_ring) +		mei_dma_ring_read(dev, NULL, hdr->extension[0]);  	/*  	 * no need to check for size as it is guarantied  	 * that length fits into rd_msg_buf @@ -100,6 +102,7 @@ static int mei_cl_irq_read_msg(struct mei_cl *cl,  	struct mei_device *dev = cl->dev;  	struct mei_cl_cb *cb;  	size_t buf_sz; +	u32 length;  	cb = list_first_entry_or_null(&cl->rd_pending, struct mei_cl_cb, list);  	if (!cb) { @@ -119,25 +122,31 @@ static int mei_cl_irq_read_msg(struct mei_cl *cl,  		goto discard;  	} -	buf_sz = mei_hdr->length + cb->buf_idx; +	length = mei_hdr->dma_ring ? mei_hdr->extension[0] : mei_hdr->length; + +	buf_sz = length + cb->buf_idx;  	/* catch for integer overflow */  	if (buf_sz < cb->buf_idx) {  		cl_err(dev, cl, "message is too big len %d idx %zu\n", -		       mei_hdr->length, cb->buf_idx); +		       length, cb->buf_idx);  		cb->status = -EMSGSIZE;  		goto discard;  	}  	if (cb->buf.size < buf_sz) {  		cl_dbg(dev, cl, "message overflow. size %zu len %d idx %zu\n", -			cb->buf.size, mei_hdr->length, cb->buf_idx); +			cb->buf.size, length, cb->buf_idx);  		cb->status = -EMSGSIZE;  		goto discard;  	} +	if (mei_hdr->dma_ring) +		mei_dma_ring_read(dev, cb->buf.data + cb->buf_idx, length); + +	/*  for DMA read 0 length to generate an interrupt to the device */  	mei_read_slots(dev, cb->buf.data + cb->buf_idx, mei_hdr->length); -	cb->buf_idx += mei_hdr->length; +	cb->buf_idx += length;  	if (mei_hdr->msg_complete) {  		cl_dbg(dev, cl, "completed read length = %zu\n", cb->buf_idx); @@ -247,6 +256,9 @@ static inline int hdr_is_valid(u32 msg_hdr)  	if (!msg_hdr || mei_hdr->reserved)  		return -EBADMSG; +	if (mei_hdr->dma_ring && mei_hdr->length != MEI_SLOT_SIZE) +		return -EBADMSG; +  	return 0;  } @@ -267,20 +279,20 @@ int mei_irq_read_handler(struct mei_device *dev,  	struct mei_cl *cl;  	int ret; -	if (!dev->rd_msg_hdr) { -		dev->rd_msg_hdr = mei_read_hdr(dev); +	if (!dev->rd_msg_hdr[0]) { +		dev->rd_msg_hdr[0] = mei_read_hdr(dev);  		(*slots)--;  		dev_dbg(dev->dev, "slots =%08x.\n", *slots); -		ret = hdr_is_valid(dev->rd_msg_hdr); +		ret = hdr_is_valid(dev->rd_msg_hdr[0]);  		if (ret) {  			dev_err(dev->dev, "corrupted message header 0x%08X\n", -				dev->rd_msg_hdr); +				dev->rd_msg_hdr[0]);  			goto end;  		}  	} -	mei_hdr = (struct mei_msg_hdr *)&dev->rd_msg_hdr; +	mei_hdr = (struct mei_msg_hdr *)dev->rd_msg_hdr;  	dev_dbg(dev->dev, MEI_HDR_FMT, MEI_HDR_PRM(mei_hdr));  	if (mei_slots2data(*slots) < mei_hdr->length) { @@ -291,6 +303,12 @@ int mei_irq_read_handler(struct mei_device *dev,  		goto end;  	} +	if (mei_hdr->dma_ring) { +		dev->rd_msg_hdr[1] = mei_read_hdr(dev); +		(*slots)--; +		mei_hdr->length = 0; +	} +  	/*  HBM message */  	if (hdr_is_hbm(mei_hdr)) {  		ret = mei_hbm_dispatch(dev, mei_hdr); @@ -324,7 +342,7 @@ int mei_irq_read_handler(struct mei_device *dev,  			goto reset_slots;  		}  		dev_err(dev->dev, "no destination client found 0x%08X\n", -				dev->rd_msg_hdr); +				dev->rd_msg_hdr[0]);  		ret = -EBADMSG;  		goto end;  	} @@ -334,9 +352,8 @@ int mei_irq_read_handler(struct mei_device *dev,  reset_slots:  	/* reset the number of slots and header */ +	memset(dev->rd_msg_hdr, 0, sizeof(dev->rd_msg_hdr));  	*slots = mei_count_full_read_slots(dev); -	dev->rd_msg_hdr = 0; -  	if (*slots == -EOVERFLOW) {  		/* overflow - reset */  		dev_err(dev->dev, "resetting due to slots overflow.\n"); diff --git a/drivers/misc/mei/mei_dev.h b/drivers/misc/mei/mei_dev.h index 377397e1b5a5..685b78ce30a5 100644 --- a/drivers/misc/mei/mei_dev.h +++ b/drivers/misc/mei/mei_dev.h @@ -122,6 +122,19 @@ struct mei_msg_data {  	unsigned char *data;  }; +/** + * struct mei_dma_dscr - dma address descriptor + * + * @vaddr: dma buffer virtual address + * @daddr: dma buffer physical address + * @size : dma buffer size + */ +struct mei_dma_dscr { +	void *vaddr; +	dma_addr_t daddr; +	size_t size; +}; +  /* Maximum number of processed FW status registers */  #define MEI_FW_STATUS_MAX 6  /* Minimal  buffer for FW status string (8 bytes in dw + space or '\0') */ @@ -409,6 +422,7 @@ struct mei_fw_version {   * @rd_msg_hdr  : read message header storage   *   * @hbuf_is_ready : query if the host host/write buffer is ready + * @dr_dscr: DMA ring descriptors: TX, RX, and CTRL   *   * @version     : HBM protocol version in use   * @hbm_f_pg_supported  : hbm feature pgi protocol @@ -483,11 +497,13 @@ struct mei_device {  #endif /* CONFIG_PM */  	unsigned char rd_msg_buf[MEI_RD_MSG_BUF_SIZE]; -	u32 rd_msg_hdr; +	u32 rd_msg_hdr[MEI_MSG_HDR_MAX];  	/* write buffer */  	bool hbuf_is_ready; +	struct mei_dma_dscr dr_dscr[DMA_DSCR_NUM]; +  	struct hbm_version version;  	unsigned int hbm_f_pg_supported:1;  	unsigned int hbm_f_dc_supported:1; @@ -578,6 +594,14 @@ int mei_restart(struct mei_device *dev);  void mei_stop(struct mei_device *dev);  void mei_cancel_work(struct mei_device *dev); +int mei_dmam_ring_alloc(struct mei_device *dev); +void mei_dmam_ring_free(struct mei_device *dev); +bool mei_dma_ring_is_allocated(struct mei_device *dev); +void mei_dma_ring_reset(struct mei_device *dev); +void mei_dma_ring_read(struct mei_device *dev, unsigned char *buf, u32 len); +void mei_dma_ring_write(struct mei_device *dev, unsigned char *buf, u32 len); +u32 mei_dma_ring_empty_slots(struct mei_device *dev); +  /*   *  MEI interrupt functions prototype   */ diff --git a/drivers/misc/mei/pci-me.c b/drivers/misc/mei/pci-me.c index ea4e152270a3..73ace2d59dea 100644 --- a/drivers/misc/mei/pci-me.c +++ b/drivers/misc/mei/pci-me.c @@ -98,9 +98,9 @@ static const struct pci_device_id mei_me_pci_tbl[] = {  	{MEI_PCI_DEVICE(MEI_DEV_ID_KBP, MEI_ME_PCH8_CFG)},  	{MEI_PCI_DEVICE(MEI_DEV_ID_KBP_2, MEI_ME_PCH8_CFG)}, -	{MEI_PCI_DEVICE(MEI_DEV_ID_CNP_LP, MEI_ME_PCH8_CFG)}, +	{MEI_PCI_DEVICE(MEI_DEV_ID_CNP_LP, MEI_ME_PCH12_CFG)},  	{MEI_PCI_DEVICE(MEI_DEV_ID_CNP_LP_4, MEI_ME_PCH8_CFG)}, -	{MEI_PCI_DEVICE(MEI_DEV_ID_CNP_H, MEI_ME_PCH8_CFG)}, +	{MEI_PCI_DEVICE(MEI_DEV_ID_CNP_H, MEI_ME_PCH12_CFG)},  	{MEI_PCI_DEVICE(MEI_DEV_ID_CNP_H_4, MEI_ME_PCH8_CFG)},  	/* required last entry */ diff --git a/drivers/misc/mic/card/mic_debugfs.c b/drivers/misc/mic/card/mic_debugfs.c index 421b3d7911df..7a4140874888 100644 --- a/drivers/misc/mic/card/mic_debugfs.c +++ b/drivers/misc/mic/card/mic_debugfs.c @@ -37,9 +37,9 @@  static struct dentry *mic_dbg;  /** - * mic_intr_test - Send interrupts to host. + * mic_intr_show - Send interrupts to host.   */ -static int mic_intr_test(struct seq_file *s, void *unused) +static int mic_intr_show(struct seq_file *s, void *unused)  {  	struct mic_driver *mdrv = s->private;  	struct mic_device *mdev = &mdrv->mdev; @@ -56,23 +56,7 @@ static int mic_intr_test(struct seq_file *s, void *unused)  	return 0;  } -static int mic_intr_test_open(struct inode *inode, struct file *file) -{ -	return single_open(file, mic_intr_test, inode->i_private); -} - -static int mic_intr_test_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations intr_test_ops = { -	.owner   = THIS_MODULE, -	.open    = mic_intr_test_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = mic_intr_test_release -}; +DEFINE_SHOW_ATTRIBUTE(mic_intr);  /**   * mic_create_card_debug_dir - Initialize MIC debugfs entries. @@ -91,7 +75,7 @@ void __init mic_create_card_debug_dir(struct mic_driver *mdrv)  	}  	d = debugfs_create_file("intr_test", 0444, mdrv->dbg_dir, -		mdrv, &intr_test_ops); +		mdrv, &mic_intr_fops);  	if (!d) {  		dev_err(mdrv->dev, diff --git a/drivers/misc/mic/cosm/cosm_debugfs.c b/drivers/misc/mic/cosm/cosm_debugfs.c index 216cb3cd2fe3..71c216d0504d 100644 --- a/drivers/misc/mic/cosm/cosm_debugfs.c +++ b/drivers/misc/mic/cosm/cosm_debugfs.c @@ -28,12 +28,12 @@  static struct dentry *cosm_dbg;  /** - * cosm_log_buf_show - Display MIC kernel log buffer + * log_buf_show - Display MIC kernel log buffer   *   * log_buf addr/len is read from System.map by user space   * and populated in sysfs entries.   */ -static int cosm_log_buf_show(struct seq_file *s, void *unused) +static int log_buf_show(struct seq_file *s, void *unused)  {  	void __iomem *log_buf_va;  	int __iomem *log_buf_len_va; @@ -78,26 +78,15 @@ done:  	return 0;  } -static int cosm_log_buf_open(struct inode *inode, struct file *file) -{ -	return single_open(file, cosm_log_buf_show, inode->i_private); -} - -static const struct file_operations log_buf_ops = { -	.owner   = THIS_MODULE, -	.open    = cosm_log_buf_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = single_release -}; +DEFINE_SHOW_ATTRIBUTE(log_buf);  /** - * cosm_force_reset_show - Force MIC reset + * force_reset_show - Force MIC reset   *   * Invokes the force_reset COSM bus op instead of the standard reset   * op in case a force reset of the MIC device is required   */ -static int cosm_force_reset_show(struct seq_file *s, void *pos) +static int force_reset_show(struct seq_file *s, void *pos)  {  	struct cosm_device *cdev = s->private; @@ -105,18 +94,7 @@ static int cosm_force_reset_show(struct seq_file *s, void *pos)  	return 0;  } -static int cosm_force_reset_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, cosm_force_reset_show, inode->i_private); -} - -static const struct file_operations force_reset_ops = { -	.owner   = THIS_MODULE, -	.open    = cosm_force_reset_debug_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = single_release -}; +DEFINE_SHOW_ATTRIBUTE(force_reset);  void cosm_create_debug_dir(struct cosm_device *cdev)  { @@ -130,9 +108,10 @@ void cosm_create_debug_dir(struct cosm_device *cdev)  	if (!cdev->dbg_dir)  		return; -	debugfs_create_file("log_buf", 0444, cdev->dbg_dir, cdev, &log_buf_ops); +	debugfs_create_file("log_buf", 0444, cdev->dbg_dir, cdev, +			    &log_buf_fops);  	debugfs_create_file("force_reset", 0444, cdev->dbg_dir, cdev, -			    &force_reset_ops); +			    &force_reset_fops);  }  void cosm_delete_debug_dir(struct cosm_device *cdev) diff --git a/drivers/misc/mic/host/mic_boot.c b/drivers/misc/mic/host/mic_boot.c index c327985c9523..6479435ac96b 100644 --- a/drivers/misc/mic/host/mic_boot.c +++ b/drivers/misc/mic/host/mic_boot.c @@ -149,7 +149,7 @@ static void *__mic_dma_alloc(struct device *dev, size_t size,  	struct scif_hw_dev *scdev = dev_get_drvdata(dev);  	struct mic_device *mdev = scdev_to_mdev(scdev);  	dma_addr_t tmp; -	void *va = kmalloc(size, gfp); +	void *va = kmalloc(size, gfp | __GFP_ZERO);  	if (va) {  		tmp = mic_map_single(mdev, va, size); diff --git a/drivers/misc/mic/host/mic_debugfs.c b/drivers/misc/mic/host/mic_debugfs.c index 0a9daba8bb5d..c6e3c764699f 100644 --- a/drivers/misc/mic/host/mic_debugfs.c +++ b/drivers/misc/mic/host/mic_debugfs.c @@ -54,23 +54,7 @@ static int mic_smpt_show(struct seq_file *s, void *pos)  	return 0;  } -static int mic_smpt_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, mic_smpt_show, inode->i_private); -} - -static int mic_smpt_debug_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations smpt_file_ops = { -	.owner   = THIS_MODULE, -	.open    = mic_smpt_debug_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = mic_smpt_debug_release -}; +DEFINE_SHOW_ATTRIBUTE(mic_smpt);  static int mic_post_code_show(struct seq_file *s, void *pos)  { @@ -81,23 +65,7 @@ static int mic_post_code_show(struct seq_file *s, void *pos)  	return 0;  } -static int mic_post_code_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, mic_post_code_show, inode->i_private); -} - -static int mic_post_code_debug_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations post_code_ops = { -	.owner   = THIS_MODULE, -	.open    = mic_post_code_debug_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = mic_post_code_debug_release -}; +DEFINE_SHOW_ATTRIBUTE(mic_post_code);  static int mic_msi_irq_info_show(struct seq_file *s, void *pos)  { @@ -143,24 +111,7 @@ static int mic_msi_irq_info_show(struct seq_file *s, void *pos)  	return 0;  } -static int mic_msi_irq_info_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, mic_msi_irq_info_show, inode->i_private); -} - -static int -mic_msi_irq_info_debug_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations msi_irq_info_ops = { -	.owner   = THIS_MODULE, -	.open    = mic_msi_irq_info_debug_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = mic_msi_irq_info_debug_release -}; +DEFINE_SHOW_ATTRIBUTE(mic_msi_irq_info);  /**   * mic_create_debug_dir - Initialize MIC debugfs entries. @@ -177,13 +128,14 @@ void mic_create_debug_dir(struct mic_device *mdev)  	if (!mdev->dbg_dir)  		return; -	debugfs_create_file("smpt", 0444, mdev->dbg_dir, mdev, &smpt_file_ops); +	debugfs_create_file("smpt", 0444, mdev->dbg_dir, mdev, +			    &mic_smpt_fops);  	debugfs_create_file("post_code", 0444, mdev->dbg_dir, mdev, -			    &post_code_ops); +			    &mic_post_code_fops);  	debugfs_create_file("msi_irq_info", 0444, mdev->dbg_dir, mdev, -			    &msi_irq_info_ops); +			    &mic_msi_irq_info_fops);  }  /** diff --git a/drivers/misc/mic/scif/scif_debugfs.c b/drivers/misc/mic/scif/scif_debugfs.c index 6884dad97e17..cca5e980c710 100644 --- a/drivers/misc/mic/scif/scif_debugfs.c +++ b/drivers/misc/mic/scif/scif_debugfs.c @@ -24,7 +24,7 @@  /* Debugfs parent dir */  static struct dentry *scif_dbg; -static int scif_dev_test(struct seq_file *s, void *unused) +static int scif_dev_show(struct seq_file *s, void *unused)  {  	int node; @@ -44,23 +44,7 @@ static int scif_dev_test(struct seq_file *s, void *unused)  	return 0;  } -static int scif_dev_test_open(struct inode *inode, struct file *file) -{ -	return single_open(file, scif_dev_test, inode->i_private); -} - -static int scif_dev_test_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations scif_dev_ops = { -	.owner   = THIS_MODULE, -	.open    = scif_dev_test_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = scif_dev_test_release -}; +DEFINE_SHOW_ATTRIBUTE(scif_dev);  static void scif_display_window(struct scif_window *window, struct seq_file *s)  { @@ -104,7 +88,7 @@ static void scif_display_all_windows(struct list_head *head, struct seq_file *s)  	}  } -static int scif_rma_test(struct seq_file *s, void *unused) +static int scif_rma_show(struct seq_file *s, void *unused)  {  	struct scif_endpt *ep;  	struct list_head *pos; @@ -123,23 +107,7 @@ static int scif_rma_test(struct seq_file *s, void *unused)  	return 0;  } -static int scif_rma_test_open(struct inode *inode, struct file *file) -{ -	return single_open(file, scif_rma_test, inode->i_private); -} - -static int scif_rma_test_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations scif_rma_ops = { -	.owner   = THIS_MODULE, -	.open    = scif_rma_test_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = scif_rma_test_release -}; +DEFINE_SHOW_ATTRIBUTE(scif_rma);  void __init scif_init_debugfs(void)  { @@ -150,8 +118,8 @@ void __init scif_init_debugfs(void)  		return;  	} -	debugfs_create_file("scif_dev", 0444, scif_dbg, NULL, &scif_dev_ops); -	debugfs_create_file("scif_rma", 0444, scif_dbg, NULL, &scif_rma_ops); +	debugfs_create_file("scif_dev", 0444, scif_dbg, NULL, &scif_dev_fops); +	debugfs_create_file("scif_rma", 0444, scif_dbg, NULL, &scif_rma_fops);  	debugfs_create_u8("en_msg_log", 0666, scif_dbg, &scif_info.en_msg_log);  	debugfs_create_u8("p2p_enable", 0666, scif_dbg, &scif_info.p2p_enable);  } diff --git a/drivers/misc/mic/scif/scif_dma.c b/drivers/misc/mic/scif/scif_dma.c index 18b8ed57c4ac..e0d97044d0e9 100644 --- a/drivers/misc/mic/scif/scif_dma.c +++ b/drivers/misc/mic/scif/scif_dma.c @@ -201,23 +201,18 @@ static void scif_mmu_notifier_release(struct mmu_notifier *mn,  }  static int scif_mmu_notifier_invalidate_range_start(struct mmu_notifier *mn, -						     struct mm_struct *mm, -						     unsigned long start, -						     unsigned long end, -						     bool blockable) +					const struct mmu_notifier_range *range)  {  	struct scif_mmu_notif	*mmn;  	mmn = container_of(mn, struct scif_mmu_notif, ep_mmu_notifier); -	scif_rma_destroy_tcw(mmn, start, end - start); +	scif_rma_destroy_tcw(mmn, range->start, range->end - range->start);  	return 0;  }  static void scif_mmu_notifier_invalidate_range_end(struct mmu_notifier *mn, -						   struct mm_struct *mm, -						   unsigned long start, -						   unsigned long end) +			const struct mmu_notifier_range *range)  {  	/*  	 * Nothing to do here, everything needed was done in diff --git a/drivers/misc/mic/scif/scif_fence.c b/drivers/misc/mic/scif/scif_fence.c index 7bb929f05d85..2e7ce6ae9dd2 100644 --- a/drivers/misc/mic/scif/scif_fence.c +++ b/drivers/misc/mic/scif/scif_fence.c @@ -195,10 +195,11 @@ static inline void *scif_get_local_va(off_t off, struct scif_window *window)  static void scif_prog_signal_cb(void *arg)  { -	struct scif_status *status = arg; +	struct scif_cb_arg *cb_arg = arg; -	dma_pool_free(status->ep->remote_dev->signal_pool, status, -		      status->src_dma_addr); +	dma_pool_free(cb_arg->ep->remote_dev->signal_pool, cb_arg->status, +		      cb_arg->src_dma_addr); +	kfree(cb_arg);  }  static int _scif_prog_signal(scif_epd_t epd, dma_addr_t dst, u64 val) @@ -209,6 +210,7 @@ static int _scif_prog_signal(scif_epd_t epd, dma_addr_t dst, u64 val)  	bool x100 = !is_dma_copy_aligned(chan->device, 1, 1, 1);  	struct dma_async_tx_descriptor *tx;  	struct scif_status *status = NULL; +	struct scif_cb_arg *cb_arg = NULL;  	dma_addr_t src;  	dma_cookie_t cookie;  	int err; @@ -257,8 +259,16 @@ static int _scif_prog_signal(scif_epd_t epd, dma_addr_t dst, u64 val)  		goto dma_fail;  	}  	if (!x100) { +		cb_arg = kmalloc(sizeof(*cb_arg), GFP_KERNEL); +		if (!cb_arg) { +			err = -ENOMEM; +			goto dma_fail; +		} +		cb_arg->src_dma_addr = src; +		cb_arg->status = status; +		cb_arg->ep = ep;  		tx->callback = scif_prog_signal_cb; -		tx->callback_param = status; +		tx->callback_param = cb_arg;  	}  	cookie = tx->tx_submit(tx);  	if (dma_submit_error(cookie)) { @@ -270,9 +280,11 @@ static int _scif_prog_signal(scif_epd_t epd, dma_addr_t dst, u64 val)  	dma_async_issue_pending(chan);  	return 0;  dma_fail: -	if (!x100) +	if (!x100) {  		dma_pool_free(ep->remote_dev->signal_pool, status,  			      src - offsetof(struct scif_status, val)); +		kfree(cb_arg); +	}  alloc_fail:  	return err;  } diff --git a/drivers/misc/mic/scif/scif_rma.c b/drivers/misc/mic/scif/scif_rma.c index c824329f7012..749321eb91ae 100644 --- a/drivers/misc/mic/scif/scif_rma.c +++ b/drivers/misc/mic/scif/scif_rma.c @@ -15,7 +15,7 @@   * Intel SCIF driver.   *   */ -#include <linux/dma_remapping.h> +#include <linux/intel-iommu.h>  #include <linux/pagemap.h>  #include <linux/sched/mm.h>  #include <linux/sched/signal.h> @@ -416,7 +416,7 @@ static int scif_create_remote_lookup(struct scif_dev *remote_dev,  		if (err)  			goto error_window;  		err = scif_map_page(&window->num_pages_lookup.lookup[j], -				    vmalloc_dma_phys ? +				    vmalloc_num_pages ?  				    vmalloc_to_page(&window->num_pages[i]) :  				    virt_to_page(&window->num_pages[i]),  				    remote_dev); diff --git a/drivers/misc/mic/scif/scif_rma.h b/drivers/misc/mic/scif/scif_rma.h index fa6722279196..964dd0fc3657 100644 --- a/drivers/misc/mic/scif/scif_rma.h +++ b/drivers/misc/mic/scif/scif_rma.h @@ -53,7 +53,7 @@  #ifndef SCIF_RMA_H  #define SCIF_RMA_H -#include <linux/dma_remapping.h> +#include <linux/intel-iommu.h>  #include <linux/mmu_notifier.h>  #include "../bus/scif_bus.h" @@ -206,6 +206,19 @@ struct scif_status {  };  /* + * struct scif_cb_arg - Stores the argument of the callback func + * + * @src_dma_addr: Source buffer DMA address + * @status: DMA status + * @ep: SCIF endpoint + */ +struct scif_cb_arg { +	dma_addr_t src_dma_addr; +	struct scif_status *status; +	struct scif_endpt *ep; +}; + +/*   * struct scif_window - Registration Window for Self and Remote   *   * @nr_pages: Number of pages which is defined as a s64 instead of an int diff --git a/drivers/misc/mic/vop/vop_debugfs.c b/drivers/misc/mic/vop/vop_debugfs.c index ab43884e5cd7..2ccef52aca23 100644 --- a/drivers/misc/mic/vop/vop_debugfs.c +++ b/drivers/misc/mic/vop/vop_debugfs.c @@ -101,23 +101,7 @@ static int vop_dp_show(struct seq_file *s, void *pos)  	return 0;  } -static int vop_dp_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, vop_dp_show, inode->i_private); -} - -static int vop_dp_debug_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations dp_ops = { -	.owner   = THIS_MODULE, -	.open    = vop_dp_debug_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = vop_dp_debug_release -}; +DEFINE_SHOW_ATTRIBUTE(vop_dp);  static int vop_vdev_info_show(struct seq_file *s, void *unused)  { @@ -194,23 +178,7 @@ static int vop_vdev_info_show(struct seq_file *s, void *unused)  	return 0;  } -static int vop_vdev_info_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, vop_vdev_info_show, inode->i_private); -} - -static int vop_vdev_info_debug_release(struct inode *inode, struct file *file) -{ -	return single_release(inode, file); -} - -static const struct file_operations vdev_info_ops = { -	.owner   = THIS_MODULE, -	.open    = vop_vdev_info_debug_open, -	.read    = seq_read, -	.llseek  = seq_lseek, -	.release = vop_vdev_info_debug_release -}; +DEFINE_SHOW_ATTRIBUTE(vop_vdev_info);  void vop_init_debugfs(struct vop_info *vi)  { @@ -222,8 +190,8 @@ void vop_init_debugfs(struct vop_info *vi)  		pr_err("can't create debugfs dir vop\n");  		return;  	} -	debugfs_create_file("dp", 0444, vi->dbg, vi, &dp_ops); -	debugfs_create_file("vdev_info", 0444, vi->dbg, vi, &vdev_info_ops); +	debugfs_create_file("dp", 0444, vi->dbg, vi, &vop_dp_fops); +	debugfs_create_file("vdev_info", 0444, vi->dbg, vi, &vop_vdev_info_fops);  }  void vop_exit_debugfs(struct vop_info *vi) diff --git a/drivers/misc/mic/vop/vop_main.c b/drivers/misc/mic/vop/vop_main.c index 3633202e18f4..6b212c8b78e7 100644 --- a/drivers/misc/mic/vop/vop_main.c +++ b/drivers/misc/mic/vop/vop_main.c @@ -129,6 +129,16 @@ static u64 vop_get_features(struct virtio_device *vdev)  	return features;  } +static void vop_transport_features(struct virtio_device *vdev) +{ +	/* +	 * Packed ring isn't enabled on virtio_vop for now, +	 * because virtio_vop uses vring_new_virtqueue() which +	 * creates virtio rings on preallocated memory. +	 */ +	__virtio_clear_bit(vdev, VIRTIO_F_RING_PACKED); +} +  static int vop_finalize_features(struct virtio_device *vdev)  {  	unsigned int i, bits; @@ -141,6 +151,9 @@ static int vop_finalize_features(struct virtio_device *vdev)  	/* Give virtio_ring a chance to accept features. */  	vring_transport_features(vdev); +	/* Give virtio_vop a chance to accept features. */ +	vop_transport_features(vdev); +  	memset_io(out_features, 0, feature_len);  	bits = min_t(unsigned, feature_len,  		     sizeof(vdev->features)) * 8; diff --git a/drivers/misc/ocxl/afu_irq.c b/drivers/misc/ocxl/afu_irq.c index e70cfa24577f..11ab996657a2 100644 --- a/drivers/misc/ocxl/afu_irq.c +++ b/drivers/misc/ocxl/afu_irq.c @@ -2,7 +2,6 @@  // Copyright 2017 IBM Corp.  #include <linux/interrupt.h>  #include <linux/eventfd.h> -#include <asm/pnv-ocxl.h>  #include "ocxl_internal.h"  #include "trace.h" diff --git a/drivers/misc/ocxl/config.c b/drivers/misc/ocxl/config.c index 57a6bb1fd3c9..8f2c5d8bd2ee 100644 --- a/drivers/misc/ocxl/config.c +++ b/drivers/misc/ocxl/config.c @@ -318,7 +318,7 @@ static int read_afu_name(struct pci_dev *dev, struct ocxl_fn_config *fn,  		if (rc)  			return rc;  		ptr = (u32 *) &afu->name[i]; -		*ptr = val; +		*ptr = le32_to_cpu((__force __le32) val);  	}  	afu->name[OCXL_AFU_NAME_SZ - 1] = '\0'; /* play safe */  	return 0; diff --git a/drivers/misc/ocxl/link.c b/drivers/misc/ocxl/link.c index 31695a078485..d50b861d7e57 100644 --- a/drivers/misc/ocxl/link.c +++ b/drivers/misc/ocxl/link.c @@ -273,9 +273,9 @@ static int setup_xsl_irq(struct pci_dev *dev, struct link *link)  	spa->irq_name = kasprintf(GFP_KERNEL, "ocxl-xsl-%x-%x-%x",  				link->domain, link->bus, link->dev);  	if (!spa->irq_name) { -		unmap_irq_registers(spa);  		dev_err(&dev->dev, "Can't allocate name for xsl interrupt\n"); -		return -ENOMEM; +		rc = -ENOMEM; +		goto err_xsl;  	}  	/*  	 * At some point, we'll need to look into allowing a higher @@ -283,11 +283,10 @@ static int setup_xsl_irq(struct pci_dev *dev, struct link *link)  	 */  	spa->virq = irq_create_mapping(NULL, hwirq);  	if (!spa->virq) { -		kfree(spa->irq_name); -		unmap_irq_registers(spa);  		dev_err(&dev->dev,  			"irq_create_mapping failed for translation interrupt\n"); -		return -EINVAL; +		rc = -EINVAL; +		goto err_name;  	}  	dev_dbg(&dev->dev, "hwirq %d mapped to virq %d\n", hwirq, spa->virq); @@ -295,15 +294,21 @@ static int setup_xsl_irq(struct pci_dev *dev, struct link *link)  	rc = request_irq(spa->virq, xsl_fault_handler, 0, spa->irq_name,  			link);  	if (rc) { -		irq_dispose_mapping(spa->virq); -		kfree(spa->irq_name); -		unmap_irq_registers(spa);  		dev_err(&dev->dev,  			"request_irq failed for translation interrupt: %d\n",  			rc); -		return -EINVAL; +		rc = -EINVAL; +		goto err_mapping;  	}  	return 0; + +err_mapping: +	irq_dispose_mapping(spa->virq); +err_name: +	kfree(spa->irq_name); +err_xsl: +	unmap_irq_registers(spa); +	return rc;  }  static void release_xsl_irq(struct link *link) @@ -566,7 +571,7 @@ int ocxl_link_update_pe(void *link_handle, int pasid, __u16 tid)  	mutex_lock(&spa->spa_lock); -	pe->tid = tid; +	pe->tid = cpu_to_be32(tid);  	/*  	 * The barrier makes sure the PE is updated diff --git a/drivers/misc/pvpanic.c b/drivers/misc/pvpanic.c new file mode 100644 index 000000000000..595ac065b401 --- /dev/null +++ b/drivers/misc/pvpanic.c @@ -0,0 +1,192 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + *  Pvpanic Device Support + * + *  Copyright (C) 2013 Fujitsu. + *  Copyright (C) 2018 ZTE. + */ + +#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt + +#include <linux/acpi.h> +#include <linux/kernel.h> +#include <linux/module.h> +#include <linux/of.h> +#include <linux/of_address.h> +#include <linux/platform_device.h> +#include <linux/types.h> + +static void __iomem *base; + +#define PVPANIC_PANICKED        (1 << 0) + +MODULE_AUTHOR("Hu Tao <hutao@cn.fujitsu.com>"); +MODULE_DESCRIPTION("pvpanic device driver"); +MODULE_LICENSE("GPL"); + +static void +pvpanic_send_event(unsigned int event) +{ +	iowrite8(event, base); +} + +static int +pvpanic_panic_notify(struct notifier_block *nb, unsigned long code, +		     void *unused) +{ +	pvpanic_send_event(PVPANIC_PANICKED); +	return NOTIFY_DONE; +} + +static struct notifier_block pvpanic_panic_nb = { +	.notifier_call = pvpanic_panic_notify, +	.priority = 1, /* let this called before broken drm_fb_helper */ +}; + +#ifdef CONFIG_ACPI +static int pvpanic_add(struct acpi_device *device); +static int pvpanic_remove(struct acpi_device *device); + +static const struct acpi_device_id pvpanic_device_ids[] = { +	{ "QEMU0001", 0 }, +	{ "", 0 } +}; +MODULE_DEVICE_TABLE(acpi, pvpanic_device_ids); + +static struct acpi_driver pvpanic_driver = { +	.name =		"pvpanic", +	.class =	"QEMU", +	.ids =		pvpanic_device_ids, +	.ops =		{ +				.add =		pvpanic_add, +				.remove =	pvpanic_remove, +			}, +	.owner =	THIS_MODULE, +}; + +static acpi_status +pvpanic_walk_resources(struct acpi_resource *res, void *context) +{ +	struct resource r; + +	if (acpi_dev_resource_io(res, &r)) { +		base = ioport_map(r.start, resource_size(&r)); +		return AE_OK; +	} else if (acpi_dev_resource_memory(res, &r)) { +		base = ioremap(r.start, resource_size(&r)); +		return AE_OK; +	} + +	return AE_ERROR; +} + +static int pvpanic_add(struct acpi_device *device) +{ +	int ret; + +	ret = acpi_bus_get_status(device); +	if (ret < 0) +		return ret; + +	if (!device->status.enabled || !device->status.functional) +		return -ENODEV; + +	acpi_walk_resources(device->handle, METHOD_NAME__CRS, +			    pvpanic_walk_resources, NULL); + +	if (!base) +		return -ENODEV; + +	atomic_notifier_chain_register(&panic_notifier_list, +				       &pvpanic_panic_nb); + +	return 0; +} + +static int pvpanic_remove(struct acpi_device *device) +{ + +	atomic_notifier_chain_unregister(&panic_notifier_list, +					 &pvpanic_panic_nb); +	iounmap(base); + +	return 0; +} + +static int pvpanic_register_acpi_driver(void) +{ +	return acpi_bus_register_driver(&pvpanic_driver); +} + +static void pvpanic_unregister_acpi_driver(void) +{ +	acpi_bus_unregister_driver(&pvpanic_driver); +} +#else +static int pvpanic_register_acpi_driver(void) +{ +	return -ENODEV; +} + +static void pvpanic_unregister_acpi_driver(void) {} +#endif + +static int pvpanic_mmio_probe(struct platform_device *pdev) +{ +	struct resource *mem; + +	mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); +	if (!mem) +		return -EINVAL; + +	base = devm_ioremap_resource(&pdev->dev, mem); +	if (IS_ERR(base)) +		return PTR_ERR(base); + +	atomic_notifier_chain_register(&panic_notifier_list, +				       &pvpanic_panic_nb); + +	return 0; +} + +static int pvpanic_mmio_remove(struct platform_device *pdev) +{ + +	atomic_notifier_chain_unregister(&panic_notifier_list, +					 &pvpanic_panic_nb); + +	return 0; +} + +static const struct of_device_id pvpanic_mmio_match[] = { +	{ .compatible = "qemu,pvpanic-mmio", }, +	{} +}; + +static struct platform_driver pvpanic_mmio_driver = { +	.driver = { +		.name = "pvpanic-mmio", +		.of_match_table = pvpanic_mmio_match, +	}, +	.probe = pvpanic_mmio_probe, +	.remove = pvpanic_mmio_remove, +}; + +static int __init pvpanic_mmio_init(void) +{ +	if (acpi_disabled) +		return platform_driver_register(&pvpanic_mmio_driver); +	else +		return pvpanic_register_acpi_driver(); +} + +static void __exit pvpanic_mmio_exit(void) +{ +	if (acpi_disabled) +		platform_driver_unregister(&pvpanic_mmio_driver); +	else +		pvpanic_unregister_acpi_driver(); +} + +module_init(pvpanic_mmio_init); +module_exit(pvpanic_mmio_exit); diff --git a/drivers/misc/sgi-gru/grutlbpurge.c b/drivers/misc/sgi-gru/grutlbpurge.c index 03b49d52092e..ca2032afe035 100644 --- a/drivers/misc/sgi-gru/grutlbpurge.c +++ b/drivers/misc/sgi-gru/grutlbpurge.c @@ -220,9 +220,7 @@ void gru_flush_all_tlb(struct gru_state *gru)   * MMUOPS notifier callout functions   */  static int gru_invalidate_range_start(struct mmu_notifier *mn, -				       struct mm_struct *mm, -				       unsigned long start, unsigned long end, -				       bool blockable) +			const struct mmu_notifier_range *range)  {  	struct gru_mm_struct *gms = container_of(mn, struct gru_mm_struct,  						 ms_notifier); @@ -230,15 +228,14 @@ static int gru_invalidate_range_start(struct mmu_notifier *mn,  	STAT(mmu_invalidate_range);  	atomic_inc(&gms->ms_range_active);  	gru_dbg(grudev, "gms %p, start 0x%lx, end 0x%lx, act %d\n", gms, -		start, end, atomic_read(&gms->ms_range_active)); -	gru_flush_tlb_range(gms, start, end - start); +		range->start, range->end, atomic_read(&gms->ms_range_active)); +	gru_flush_tlb_range(gms, range->start, range->end - range->start);  	return 0;  }  static void gru_invalidate_range_end(struct mmu_notifier *mn, -				     struct mm_struct *mm, unsigned long start, -				     unsigned long end) +			const struct mmu_notifier_range *range)  {  	struct gru_mm_struct *gms = container_of(mn, struct gru_mm_struct,  						 ms_notifier); @@ -247,7 +244,8 @@ static void gru_invalidate_range_end(struct mmu_notifier *mn,  	(void)atomic_dec_and_test(&gms->ms_range_active);  	wake_up_all(&gms->ms_wait_queue); -	gru_dbg(grudev, "gms %p, start 0x%lx, end 0x%lx\n", gms, start, end); +	gru_dbg(grudev, "gms %p, start 0x%lx, end 0x%lx\n", +		gms, range->start, range->end);  }  static void gru_release(struct mmu_notifier *mn, struct mm_struct *mm) diff --git a/drivers/misc/ti-st/st_kim.c b/drivers/misc/ti-st/st_kim.c index 1874ac922166..e7cfdbd1f66d 100644 --- a/drivers/misc/ti-st/st_kim.c +++ b/drivers/misc/ti-st/st_kim.c @@ -211,7 +211,7 @@ static void kim_int_recv(struct kim_data_s *kim_gdata,  static long read_local_version(struct kim_data_s *kim_gdata, char *bts_scr_name)  {  	unsigned short version = 0, chip = 0, min_ver = 0, maj_ver = 0; -	const char read_ver_cmd[] = { 0x01, 0x01, 0x10, 0x00 }; +	static const char read_ver_cmd[] = { 0x01, 0x01, 0x10, 0x00 };  	long timeout;  	pr_debug("%s", __func__); @@ -564,7 +564,7 @@ long st_kim_stop(void *kim_data)  /* functions called from subsystems */  /* called when debugfs entry is read from */ -static int show_version(struct seq_file *s, void *unused) +static int version_show(struct seq_file *s, void *unused)  {  	struct kim_data_s *kim_gdata = (struct kim_data_s *)s->private;  	seq_printf(s, "%04X %d.%d.%d\n", kim_gdata->version.full, @@ -573,7 +573,7 @@ static int show_version(struct seq_file *s, void *unused)  	return 0;  } -static int show_list(struct seq_file *s, void *unused) +static int list_show(struct seq_file *s, void *unused)  {  	struct kim_data_s *kim_gdata = (struct kim_data_s *)s->private;  	kim_st_list_protocols(kim_gdata->core_data, s); @@ -688,30 +688,8 @@ err:  	*core_data = NULL;  } -static int kim_version_open(struct inode *i, struct file *f) -{ -	return single_open(f, show_version, i->i_private); -} - -static int kim_list_open(struct inode *i, struct file *f) -{ -	return single_open(f, show_list, i->i_private); -} - -static const struct file_operations version_debugfs_fops = { -	/* version info */ -	.open = kim_version_open, -	.read = seq_read, -	.llseek = seq_lseek, -	.release = single_release, -}; -static const struct file_operations list_debugfs_fops = { -	/* protocols info */ -	.open = kim_list_open, -	.read = seq_read, -	.llseek = seq_lseek, -	.release = single_release, -}; +DEFINE_SHOW_ATTRIBUTE(version); +DEFINE_SHOW_ATTRIBUTE(list);  /**********************************************************************/  /* functions called from platform device driver subsystem @@ -789,9 +767,9 @@ static int kim_probe(struct platform_device *pdev)  	}  	debugfs_create_file("version", S_IRUGO, kim_debugfs_dir, -				kim_gdata, &version_debugfs_fops); +				kim_gdata, &version_fops);  	debugfs_create_file("protocols", S_IRUGO, kim_debugfs_dir, -				kim_gdata, &list_debugfs_fops); +				kim_gdata, &list_fops);  	return 0;  err_sysfs_group: diff --git a/drivers/misc/vexpress-syscfg.c b/drivers/misc/vexpress-syscfg.c index 6c3591cdf855..a3c6c773d9dc 100644 --- a/drivers/misc/vexpress-syscfg.c +++ b/drivers/misc/vexpress-syscfg.c @@ -61,7 +61,7 @@ static int vexpress_syscfg_exec(struct vexpress_syscfg_func *func,  	int tries;  	long timeout; -	if (WARN_ON(index > func->num_templates)) +	if (WARN_ON(index >= func->num_templates))  		return -EINVAL;  	command = readl(syscfg->base + SYS_CFGCTRL); diff --git a/drivers/misc/vmw_balloon.c b/drivers/misc/vmw_balloon.c index 9b0b3fa4f836..f8240b87df22 100644 --- a/drivers/misc/vmw_balloon.c +++ b/drivers/misc/vmw_balloon.c @@ -570,7 +570,7 @@ static int vmballoon_send_get_target(struct vmballoon *b)  	unsigned long status;  	unsigned long limit; -	limit = totalram_pages; +	limit = totalram_pages();  	/* Ensure limit fits in 32-bits */  	if (limit != (u32)limit) @@ -1470,18 +1470,7 @@ static int vmballoon_debug_show(struct seq_file *f, void *offset)  	return 0;  } -static int vmballoon_debug_open(struct inode *inode, struct file *file) -{ -	return single_open(file, vmballoon_debug_show, inode->i_private); -} - -static const struct file_operations vmballoon_debug_fops = { -	.owner		= THIS_MODULE, -	.open		= vmballoon_debug_open, -	.read		= seq_read, -	.llseek		= seq_lseek, -	.release	= single_release, -}; +DEFINE_SHOW_ATTRIBUTE(vmballoon_debug);  static int __init vmballoon_debugfs_init(struct vmballoon *b)  { diff --git a/drivers/misc/vmw_vmci/vmci_host.c b/drivers/misc/vmw_vmci/vmci_host.c index edfffc9699ba..997f92543dd4 100644 --- a/drivers/misc/vmw_vmci/vmci_host.c +++ b/drivers/misc/vmw_vmci/vmci_host.c @@ -236,7 +236,7 @@ static int vmci_host_setup_notify(struct vmci_ctx *context,  	 * about the size.  	 */  	BUILD_BUG_ON(sizeof(bool) != sizeof(u8)); -	if (!access_ok(VERIFY_WRITE, (void __user *)uva, sizeof(u8))) +	if (!access_ok((void __user *)uva, sizeof(u8)))  		return VMCI_ERROR_GENERIC;  	/* @@ -750,19 +750,10 @@ static int vmci_host_do_ctx_set_cpt_state(struct vmci_host_dev *vmci_host_dev,  	if (copy_from_user(&set_info, uptr, sizeof(set_info)))  		return -EFAULT; -	cpt_buf = kmalloc(set_info.buf_size, GFP_KERNEL); -	if (!cpt_buf) { -		vmci_ioctl_err( -			"cannot allocate memory to set cpt state (type=%d)\n", -			set_info.cpt_type); -		return -ENOMEM; -	} - -	if (copy_from_user(cpt_buf, (void __user *)(uintptr_t)set_info.cpt_buf, -			   set_info.buf_size)) { -		retval = -EFAULT; -		goto out; -	} +	cpt_buf = memdup_user((void __user *)(uintptr_t)set_info.cpt_buf, +				set_info.buf_size); +	if (IS_ERR(cpt_buf)) +		return PTR_ERR(cpt_buf);  	cid = vmci_ctx_get_id(vmci_host_dev->context);  	set_info.result = vmci_ctx_set_chkpt_state(cid, set_info.cpt_type, @@ -770,7 +761,6 @@ static int vmci_host_do_ctx_set_cpt_state(struct vmci_host_dev *vmci_host_dev,  	retval = copy_to_user(uptr, &set_info, sizeof(set_info)) ? -EFAULT : 0; -out:  	kfree(cpt_buf);  	return retval;  } | 
