Re: [PATCH] PCI: change device runtime PM settings for probe and remove

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

 



On Tuesday, June 08, 2010, Alan Stern wrote:
> This patch (as1388) changes the way the PCI core handles runtime PM
> settings when probing or unbinding drivers.  Now the core will make
> sure the device is enabled for runtime PM, with a usage count >= 1,
> when a driver is probed.  It does the same when calling a driver's
> remove method.
> 
> If the driver wants to use runtime PM, all it has to do is call
> pm_runtime_pu_noidle() near the end of its probe routine (to cancel
> the core's usage increment) and pm_runtime_get_noresume() near the
> start of its remove routine (to restore the usage count).  It does not
> need to mess around with setting the runtime state to enabled,
> disabled, active, or suspended.
> 
> The patch updates e1000e and r8169, the only PCI drivers that already
> use the existing runtime PM interface.
> 
> Signed-off-by: Alan Stern <stern@xxxxxxxxxxxxxxxxxxx>
> CC: Rafael J. Wysocki <rjw@xxxxxxx>

Acked-by: Rafael J. Wysocki <rjw@xxxxxxx>

Jesse, please queue up for 2.6.36.

Rafael


> ---
> 
> Index: usb-2.6/drivers/pci/pci-driver.c
> ===================================================================
> --- usb-2.6.orig/drivers/pci/pci-driver.c
> +++ usb-2.6/drivers/pci/pci-driver.c
> @@ -289,8 +289,26 @@ struct drv_dev_and_id {
>  static long local_pci_probe(void *_ddi)
>  {
>  	struct drv_dev_and_id *ddi = _ddi;
> +	struct device *dev = &ddi->dev->dev;
> +	int rc;
>  
> -	return ddi->drv->probe(ddi->dev, ddi->id);
> +	/* Unbound PCI devices are always set to disabled and suspended.
> +	 * During probe, the device is set to enabled and active and the
> +	 * usage count is incremented.  If the driver supports runtime PM,
> +	 * it should call pm_runtime_put_noidle() in its probe routine and
> +	 * pm_runtime_get_noresume() in its remove routine.
> +	 */
> +	pm_runtime_get_noresume(dev);
> +	pm_runtime_set_active(dev);
> +	pm_runtime_enable(dev);
> +
> +	rc = ddi->drv->probe(ddi->dev, ddi->id);
> +	if (rc) {
> +		pm_runtime_disable(dev);
> +		pm_runtime_set_suspended(dev);
> +		pm_runtime_put_noidle(dev);
> +	}
> +	return rc;
>  }
>  
>  static int pci_call_probe(struct pci_driver *drv, struct pci_dev *dev,
> @@ -369,11 +387,19 @@ static int pci_device_remove(struct devi
>  	struct pci_driver * drv = pci_dev->driver;
>  
>  	if (drv) {
> -		if (drv->remove)
> +		if (drv->remove) {
> +			pm_runtime_get_sync(dev);
>  			drv->remove(pci_dev);
> +			pm_runtime_put_noidle(dev);
> +		}
>  		pci_dev->driver = NULL;
>  	}
>  
> +	/* Undo the runtime PM settings in local_pci_probe() */
> +	pm_runtime_disable(dev);
> +	pm_runtime_set_suspended(dev);
> +	pm_runtime_put_noidle(dev);
> +
>  	/*
>  	 * If the device is still on, set the power state as "unknown",
>  	 * since it might change by the next time we load the driver.
> Index: usb-2.6/drivers/net/e1000e/netdev.c
> ===================================================================
> --- usb-2.6.orig/drivers/net/e1000e/netdev.c
> +++ usb-2.6/drivers/net/e1000e/netdev.c
> @@ -5721,11 +5721,8 @@ static int __devinit e1000_probe(struct 
>  
>  	e1000_print_device_info(adapter);
>  
> -	if (pci_dev_run_wake(pdev)) {
> -		pm_runtime_set_active(&pdev->dev);
> -		pm_runtime_enable(&pdev->dev);
> -	}
> -	pm_schedule_suspend(&pdev->dev, MSEC_PER_SEC);
> +	if (pci_dev_run_wake(pdev))
> +		pm_runtime_put_noidle(&pdev->dev);
>  
>  	return 0;
>  
> @@ -5771,8 +5768,6 @@ static void __devexit e1000_remove(struc
>  	struct e1000_adapter *adapter = netdev_priv(netdev);
>  	bool down = test_bit(__E1000_DOWN, &adapter->state);
>  
> -	pm_runtime_get_sync(&pdev->dev);
> -
>  	/*
>  	 * flush_scheduled work may reschedule our watchdog task, so
>  	 * explicitly disable watchdog tasks from being rescheduled
> @@ -5797,11 +5792,8 @@ static void __devexit e1000_remove(struc
>  		clear_bit(__E1000_DOWN, &adapter->state);
>  	unregister_netdev(netdev);
>  
> -	if (pci_dev_run_wake(pdev)) {
> -		pm_runtime_disable(&pdev->dev);
> -		pm_runtime_set_suspended(&pdev->dev);
> -	}
> -	pm_runtime_put_noidle(&pdev->dev);
> +	if (pci_dev_run_wake(pdev))
> +		pm_runtime_get_noresume(&pdev->dev);
>  
>  	/*
>  	 * Release control of h/w to f/w.  If f/w is AMT enabled, this
> Index: usb-2.6/drivers/net/r8169.c
> ===================================================================
> --- usb-2.6.orig/drivers/net/r8169.c
> +++ usb-2.6/drivers/net/r8169.c
> @@ -3208,11 +3208,8 @@ rtl8169_init_one(struct pci_dev *pdev, c
>  
>  	device_set_wakeup_enable(&pdev->dev, tp->features & RTL_FEATURE_WOL);
>  
> -	if (pci_dev_run_wake(pdev)) {
> -		pm_runtime_set_active(&pdev->dev);
> -		pm_runtime_enable(&pdev->dev);
> -	}
> -	pm_runtime_idle(&pdev->dev);
> +	if (pci_dev_run_wake(pdev))
> +		pm_runtime_put_noidle(&pdev->dev);
>  
>  out:
>  	return rc;
> @@ -3235,17 +3232,12 @@ static void __devexit rtl8169_remove_one
>  	struct net_device *dev = pci_get_drvdata(pdev);
>  	struct rtl8169_private *tp = netdev_priv(dev);
>  
> -	pm_runtime_get_sync(&pdev->dev);
> -
>  	flush_scheduled_work();
>  
>  	unregister_netdev(dev);
>  
> -	if (pci_dev_run_wake(pdev)) {
> -		pm_runtime_disable(&pdev->dev);
> -		pm_runtime_set_suspended(&pdev->dev);
> -	}
> -	pm_runtime_put_noidle(&pdev->dev);
> +	if (pci_dev_run_wake(pdev))
> +		pm_runtime_get_noresume(&pdev->dev);
>  
>  	/* restore original MAC address */
>  	rtl_rar_set(tp, dev->perm_addr);
_______________________________________________
linux-pm mailing list
linux-pm@xxxxxxxxxxxxxxxxxxxxxxxxxx
https://lists.linux-foundation.org/mailman/listinfo/linux-pm


[Index of Archives]     [Linux ACPI]     [Netdev]     [Ethernet Bridging]     [Linux Wireless]     [CPU Freq]     [Kernel Newbies]     [Fedora Kernel]     [Security]     [Linux for Hams]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux RAID]     [Linux Admin]     [Samba]

  Powered by Linux