Re: [patch 3/7] x86/pci/ce4100: Properly lock accessor functions

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



[+cc linux-pci]

On Thu, Mar 16, 2017 at 10:50:05PM +0100, Thomas Gleixner wrote:
> x86 wants to get rid of the global pci_lock protecting the config space
> accessors so ECAM mode can operate completely lockless, but the CE4100 pci
> code relies on that to protect the simulation registers.

s/pci/PCI/

> Restructure the code so it uses the x86 specific pci_config_lock to
> serialize the inner workings of the CE4100 PCI magic. That allows to remove
> the global locking via pci_lock later.
> 
> Signed-off-by: Thomas Gleixner <tglx@xxxxxxxxxxxxx>
> ---
>  arch/x86/pci/ce4100.c |   87 +++++++++++++++++++++++++++-----------------------
>  1 file changed, 48 insertions(+), 39 deletions(-)
> 
> --- a/arch/x86/pci/ce4100.c
> +++ b/arch/x86/pci/ce4100.c
> @@ -65,6 +65,9 @@ struct sim_reg_op {
>  { PCI_DEVFN(device, func), offset, init_op, read_op, write_op,\
>  	{0, SIZE_TO_MASK(size)} },
>  
> +/*
> + * All read/write functions are called with pci_config_lock held.
> + */
>  static void reg_init(struct sim_dev_reg *reg)
>  {
>  	pci_direct_conf1.read(0, 1, reg->dev_func, reg->reg, 4,
> @@ -73,21 +76,13 @@ static void reg_init(struct sim_dev_reg
>  
>  static void reg_read(struct sim_dev_reg *reg, u32 *value)
>  {
> -	unsigned long flags;
> -
> -	raw_spin_lock_irqsave(&pci_config_lock, flags);
>  	*value = reg->sim_reg.value;
> -	raw_spin_unlock_irqrestore(&pci_config_lock, flags);
>  }
>  
>  static void reg_write(struct sim_dev_reg *reg, u32 value)
>  {
> -	unsigned long flags;
> -
> -	raw_spin_lock_irqsave(&pci_config_lock, flags);
>  	reg->sim_reg.value = (value & reg->sim_reg.mask) |
>  		(reg->sim_reg.value & ~reg->sim_reg.mask);
> -	raw_spin_unlock_irqrestore(&pci_config_lock, flags);
>  }
>  
>  static void sata_reg_init(struct sim_dev_reg *reg)
> @@ -117,12 +112,8 @@ static void sata_revid_read(struct sim_d
>  
>  static void reg_noirq_read(struct sim_dev_reg *reg, u32 *value)
>  {
> -	unsigned long flags;
> -
> -	raw_spin_lock_irqsave(&pci_config_lock, flags);
>  	/* force interrupt pin value to 0 */
>  	*value = reg->sim_reg.value & 0xfff00ff;
> -	raw_spin_unlock_irqrestore(&pci_config_lock, flags);
>  }
>  
>  static struct sim_dev_reg bus1_fixups[] = {
> @@ -265,24 +256,33 @@ int bridge_read(unsigned int devfn, int
>  	return retval;
>  }
>  
> -static int ce4100_conf_read(unsigned int seg, unsigned int bus,
> -			    unsigned int devfn, int reg, int len, u32 *value)
> +static int ce4100_bus1_read(unsigned int devfn, int reg, int len, u32 *value)
>  {
> +	unsigned long flags;
>  	int i;
>  
> -	WARN_ON(seg);
> -	if (bus == 1) {
> -		for (i = 0; i < ARRAY_SIZE(bus1_fixups); i++) {
> -			if (bus1_fixups[i].dev_func == devfn &&
> -			    bus1_fixups[i].reg == (reg & ~3) &&
> -			    bus1_fixups[i].read) {
> -				bus1_fixups[i].read(&(bus1_fixups[i]),
> -						    value);
> -				extract_bytes(value, reg, len);
> -				return 0;
> -			}
> +	for (i = 0; i < ARRAY_SIZE(bus1_fixups); i++) {
> +		if (bus1_fixups[i].dev_func == devfn &&
> +		    bus1_fixups[i].reg == (reg & ~3) &&
> +		    bus1_fixups[i].read) {
> +
> +			raw_spin_lock_irqsave(&pci_config_lock, flags);
> +			bus1_fixups[i].read(&(bus1_fixups[i]), value);
> +			raw_spin_unlock_irqrestore(&pci_config_lock, flags);
> +			extract_bytes(value, reg, len);
> +			return 0;
>  		}
>  	}
> +	return -1;
> +}
> +
> +static int ce4100_conf_read(unsigned int seg, unsigned int bus,
> +			    unsigned int devfn, int reg, int len, u32 *value)
> +{
> +	WARN_ON(seg);
> +
> +	if (bus == 1 && !ce4100_bus1_read(devfn, reg, len, value))
> +		return 0;
>  
>  	if (bus == 0 && (PCI_DEVFN(1, 0) == devfn) &&
>  	    !bridge_read(devfn, reg, len, value))
> @@ -291,23 +291,32 @@ static int ce4100_conf_read(unsigned int
>  	return pci_direct_conf1.read(seg, bus, devfn, reg, len, value);
>  }
>  
> -static int ce4100_conf_write(unsigned int seg, unsigned int bus,
> -			     unsigned int devfn, int reg, int len, u32 value)
> +static int ce4100_bus1_write(unsigned int devfn, int reg, int len, u32 value)
>  {
> +	unsigned long flags;
>  	int i;
>  
> -	WARN_ON(seg);
> -	if (bus == 1) {
> -		for (i = 0; i < ARRAY_SIZE(bus1_fixups); i++) {
> -			if (bus1_fixups[i].dev_func == devfn &&
> -			    bus1_fixups[i].reg == (reg & ~3) &&
> -			    bus1_fixups[i].write) {
> -				bus1_fixups[i].write(&(bus1_fixups[i]),
> -						     value);
> -				return 0;
> -			}
> +	for (i = 0; i < ARRAY_SIZE(bus1_fixups); i++) {
> +		if (bus1_fixups[i].dev_func == devfn &&
> +		    bus1_fixups[i].reg == (reg & ~3) &&
> +		    bus1_fixups[i].write) {
> +
> +			raw_spin_lock_irqsave(&pci_config_lock, flags);
> +			bus1_fixups[i].write(&(bus1_fixups[i]), value);
> +			raw_spin_unlock_irqrestore(&pci_config_lock, flags);
> +			return 0;
>  		}
>  	}
> +	return -1;
> +}
> +
> +static int ce4100_conf_write(unsigned int seg, unsigned int bus,
> +			     unsigned int devfn, int reg, int len, u32 value)
> +{
> +	WARN_ON(seg);
> +
> +	if (bus == 1 && !ce4100_bus1_write(devfn, reg, len, value))
> +		return 0;
>  
>  	/* Discard writes to A/V bridge BAR. */
>  	if (bus == 0 && PCI_DEVFN(1, 0) == devfn &&
> @@ -318,8 +327,8 @@ static int ce4100_conf_write(unsigned in
>  }
>  
>  static const struct pci_raw_ops ce4100_pci_conf = {
> -	.read =	ce4100_conf_read,
> -	.write = ce4100_conf_write,
> +	.read	= ce4100_conf_read,
> +	.write	= ce4100_conf_write,
>  };
>  
>  int __init ce4100_pci_init(void)
> 
> 



[Index of Archives]     [DMA Engine]     [Linux Coverity]     [Linux USB]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Greybus]

  Powered by Linux