mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	wil6210: run-time PM when interface down
Allow run-time suspend when interface is down, keep card alive when interface is up. If driver is in wmi only or debug_fw mode run-time PM won't suspend. Signed-off-by: Lazar Alexei <qca_ailizaro@qca.qualcomm.com> Signed-off-by: Maya Erez <qca_merez@qca.qualcomm.com> Signed-off-by: Kalle Valo <kvalo@qca.qualcomm.com>
This commit is contained in:
		
							parent
							
								
									8ed05ed06f
								
							
						
					
					
						commit
						94162666cd
					
				| @ -242,12 +242,19 @@ static void wil_print_ring(struct seq_file *s, const char *prefix, | |||||||
| static int wil_mbox_debugfs_show(struct seq_file *s, void *data) | static int wil_mbox_debugfs_show(struct seq_file *s, void *data) | ||||||
| { | { | ||||||
| 	struct wil6210_priv *wil = s->private; | 	struct wil6210_priv *wil = s->private; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	ret = wil_pm_runtime_get(wil); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		return ret; | ||||||
| 
 | 
 | ||||||
| 	wil_print_ring(s, "tx", wil->csr + HOST_MBOX + | 	wil_print_ring(s, "tx", wil->csr + HOST_MBOX + | ||||||
| 		       offsetof(struct wil6210_mbox_ctl, tx)); | 		       offsetof(struct wil6210_mbox_ctl, tx)); | ||||||
| 	wil_print_ring(s, "rx", wil->csr + HOST_MBOX + | 	wil_print_ring(s, "rx", wil->csr + HOST_MBOX + | ||||||
| 		       offsetof(struct wil6210_mbox_ctl, rx)); | 		       offsetof(struct wil6210_mbox_ctl, rx)); | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -265,15 +272,37 @@ static const struct file_operations fops_mbox = { | |||||||
| 
 | 
 | ||||||
| static int wil_debugfs_iomem_x32_set(void *data, u64 val) | static int wil_debugfs_iomem_x32_set(void *data, u64 val) | ||||||
| { | { | ||||||
| 	writel(val, (void __iomem *)data); | 	struct wil_debugfs_iomem_data *d = (struct | ||||||
|  | 					    wil_debugfs_iomem_data *)data; | ||||||
|  | 	struct wil6210_priv *wil = d->wil; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	ret = wil_pm_runtime_get(wil); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
|  | 	writel(val, (void __iomem *)d->offset); | ||||||
| 	wmb(); /* make sure write propagated to HW */ | 	wmb(); /* make sure write propagated to HW */ | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int wil_debugfs_iomem_x32_get(void *data, u64 *val) | static int wil_debugfs_iomem_x32_get(void *data, u64 *val) | ||||||
| { | { | ||||||
| 	*val = readl((void __iomem *)data); | 	struct wil_debugfs_iomem_data *d = (struct | ||||||
|  | 					    wil_debugfs_iomem_data *)data; | ||||||
|  | 	struct wil6210_priv *wil = d->wil; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	ret = wil_pm_runtime_get(wil); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
|  | 	*val = readl((void __iomem *)d->offset); | ||||||
|  | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| @ -284,10 +313,21 @@ DEFINE_SIMPLE_ATTRIBUTE(fops_iomem_x32, wil_debugfs_iomem_x32_get, | |||||||
| static struct dentry *wil_debugfs_create_iomem_x32(const char *name, | static struct dentry *wil_debugfs_create_iomem_x32(const char *name, | ||||||
| 						   umode_t mode, | 						   umode_t mode, | ||||||
| 						   struct dentry *parent, | 						   struct dentry *parent, | ||||||
| 						   void *value) | 						   void *value, | ||||||
|  | 						   struct wil6210_priv *wil) | ||||||
| { | { | ||||||
| 	return debugfs_create_file(name, mode, parent, value, | 	struct dentry *file; | ||||||
| 				   &fops_iomem_x32); | 	struct wil_debugfs_iomem_data *data = &wil->dbg_data.data_arr[ | ||||||
|  | 					      wil->dbg_data.iomem_data_count]; | ||||||
|  | 
 | ||||||
|  | 	data->wil = wil; | ||||||
|  | 	data->offset = value; | ||||||
|  | 
 | ||||||
|  | 	file = debugfs_create_file(name, mode, parent, data, &fops_iomem_x32); | ||||||
|  | 	if (!IS_ERR_OR_NULL(file)) | ||||||
|  | 		wil->dbg_data.iomem_data_count++; | ||||||
|  | 
 | ||||||
|  | 	return file; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int wil_debugfs_ulong_set(void *data, u64 val) | static int wil_debugfs_ulong_set(void *data, u64 val) | ||||||
| @ -346,7 +386,8 @@ static void wil6210_debugfs_init_offset(struct wil6210_priv *wil, | |||||||
| 		case doff_io32: | 		case doff_io32: | ||||||
| 			f = wil_debugfs_create_iomem_x32(tbl[i].name, | 			f = wil_debugfs_create_iomem_x32(tbl[i].name, | ||||||
| 							 tbl[i].mode, dbg, | 							 tbl[i].mode, dbg, | ||||||
| 							 base + tbl[i].off); | 							 base + tbl[i].off, | ||||||
|  | 							 wil); | ||||||
| 			break; | 			break; | ||||||
| 		case doff_u8: | 		case doff_u8: | ||||||
| 			f = debugfs_create_u8(tbl[i].name, tbl[i].mode, dbg, | 			f = debugfs_create_u8(tbl[i].name, tbl[i].mode, dbg, | ||||||
| @ -475,13 +516,22 @@ static int wil6210_debugfs_create_ITR_CNT(struct wil6210_priv *wil, | |||||||
| static int wil_memread_debugfs_show(struct seq_file *s, void *data) | static int wil_memread_debugfs_show(struct seq_file *s, void *data) | ||||||
| { | { | ||||||
| 	struct wil6210_priv *wil = s->private; | 	struct wil6210_priv *wil = s->private; | ||||||
| 	void __iomem *a = wmi_buffer(wil, cpu_to_le32(mem_addr)); | 	void __iomem *a; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	ret = wil_pm_runtime_get(wil); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
|  | 	a = wmi_buffer(wil, cpu_to_le32(mem_addr)); | ||||||
| 
 | 
 | ||||||
| 	if (a) | 	if (a) | ||||||
| 		seq_printf(s, "[0x%08x] = 0x%08x\n", mem_addr, readl(a)); | 		seq_printf(s, "[0x%08x] = 0x%08x\n", mem_addr, readl(a)); | ||||||
| 	else | 	else | ||||||
| 		seq_printf(s, "[0x%08x] = INVALID\n", mem_addr); | 		seq_printf(s, "[0x%08x] = INVALID\n", mem_addr); | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -502,10 +552,12 @@ static ssize_t wil_read_file_ioblob(struct file *file, char __user *user_buf, | |||||||
| { | { | ||||||
| 	enum { max_count = 4096 }; | 	enum { max_count = 4096 }; | ||||||
| 	struct wil_blob_wrapper *wil_blob = file->private_data; | 	struct wil_blob_wrapper *wil_blob = file->private_data; | ||||||
|  | 	struct wil6210_priv *wil = wil_blob->wil; | ||||||
| 	loff_t pos = *ppos; | 	loff_t pos = *ppos; | ||||||
| 	size_t available = wil_blob->blob.size; | 	size_t available = wil_blob->blob.size; | ||||||
| 	void *buf; | 	void *buf; | ||||||
| 	size_t ret; | 	size_t ret; | ||||||
|  | 	int rc; | ||||||
| 
 | 
 | ||||||
| 	if (test_bit(wil_status_suspending, wil_blob->wil->status) || | 	if (test_bit(wil_status_suspending, wil_blob->wil->status) || | ||||||
| 	    test_bit(wil_status_suspended, wil_blob->wil->status)) | 	    test_bit(wil_status_suspended, wil_blob->wil->status)) | ||||||
| @ -526,10 +578,19 @@ static ssize_t wil_read_file_ioblob(struct file *file, char __user *user_buf, | |||||||
| 	if (!buf) | 	if (!buf) | ||||||
| 		return -ENOMEM; | 		return -ENOMEM; | ||||||
| 
 | 
 | ||||||
|  | 	rc = wil_pm_runtime_get(wil); | ||||||
|  | 	if (rc < 0) { | ||||||
|  | 		kfree(buf); | ||||||
|  | 		return rc; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	wil_memcpy_fromio_32(buf, (const void __iomem *) | 	wil_memcpy_fromio_32(buf, (const void __iomem *) | ||||||
| 			     wil_blob->blob.data + pos, count); | 			     wil_blob->blob.data + pos, count); | ||||||
| 
 | 
 | ||||||
| 	ret = copy_to_user(user_buf, buf, count); | 	ret = copy_to_user(user_buf, buf, count); | ||||||
|  | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
| 	kfree(buf); | 	kfree(buf); | ||||||
| 	if (ret == count) | 	if (ret == count) | ||||||
| 		return -EFAULT; | 		return -EFAULT; | ||||||
| @ -1736,14 +1797,31 @@ static const struct dbg_off dbg_statics[] = { | |||||||
| 	{}, | 	{}, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | static const int dbg_off_count = 4 * (ARRAY_SIZE(isr_off) - 1) + | ||||||
|  | 				ARRAY_SIZE(dbg_wil_regs) - 1 + | ||||||
|  | 				ARRAY_SIZE(pseudo_isr_off) - 1 + | ||||||
|  | 				ARRAY_SIZE(lgc_itr_cnt_off) - 1 + | ||||||
|  | 				ARRAY_SIZE(tx_itr_cnt_off) - 1 + | ||||||
|  | 				ARRAY_SIZE(rx_itr_cnt_off) - 1; | ||||||
|  | 
 | ||||||
| int wil6210_debugfs_init(struct wil6210_priv *wil) | int wil6210_debugfs_init(struct wil6210_priv *wil) | ||||||
| { | { | ||||||
| 	struct dentry *dbg = wil->debug = debugfs_create_dir(WIL_NAME, | 	struct dentry *dbg = wil->debug = debugfs_create_dir(WIL_NAME, | ||||||
| 			wil_to_wiphy(wil)->debugfsdir); | 			wil_to_wiphy(wil)->debugfsdir); | ||||||
| 
 |  | ||||||
| 	if (IS_ERR_OR_NULL(dbg)) | 	if (IS_ERR_OR_NULL(dbg)) | ||||||
| 		return -ENODEV; | 		return -ENODEV; | ||||||
| 
 | 
 | ||||||
|  | 	wil->dbg_data.data_arr = kcalloc(dbg_off_count, | ||||||
|  | 					 sizeof(struct wil_debugfs_iomem_data), | ||||||
|  | 					 GFP_KERNEL); | ||||||
|  | 	if (!wil->dbg_data.data_arr) { | ||||||
|  | 		debugfs_remove_recursive(dbg); | ||||||
|  | 		wil->debug = NULL; | ||||||
|  | 		return -ENOMEM; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wil->dbg_data.iomem_data_count = 0; | ||||||
|  | 
 | ||||||
| 	wil_pmc_init(wil); | 	wil_pmc_init(wil); | ||||||
| 
 | 
 | ||||||
| 	wil6210_debugfs_init_files(wil, dbg); | 	wil6210_debugfs_init_files(wil, dbg); | ||||||
| @ -1768,6 +1846,8 @@ void wil6210_debugfs_remove(struct wil6210_priv *wil) | |||||||
| 	debugfs_remove_recursive(wil->debug); | 	debugfs_remove_recursive(wil->debug); | ||||||
| 	wil->debug = NULL; | 	wil->debug = NULL; | ||||||
| 
 | 
 | ||||||
|  | 	kfree(wil->dbg_data.data_arr); | ||||||
|  | 
 | ||||||
| 	/* free pmc memory without sending command to fw, as it will
 | 	/* free pmc memory without sending command to fw, as it will
 | ||||||
| 	 * be reset on the way down anyway | 	 * be reset on the way down anyway | ||||||
| 	 */ | 	 */ | ||||||
|  | |||||||
| @ -47,9 +47,14 @@ static int wil_ethtoolops_get_coalesce(struct net_device *ndev, | |||||||
| 	struct wil6210_priv *wil = ndev_to_wil(ndev); | 	struct wil6210_priv *wil = ndev_to_wil(ndev); | ||||||
| 	u32 tx_itr_en, tx_itr_val = 0; | 	u32 tx_itr_en, tx_itr_val = 0; | ||||||
| 	u32 rx_itr_en, rx_itr_val = 0; | 	u32 rx_itr_en, rx_itr_val = 0; | ||||||
|  | 	int ret; | ||||||
| 
 | 
 | ||||||
| 	wil_dbg_misc(wil, "ethtoolops_get_coalesce\n"); | 	wil_dbg_misc(wil, "ethtoolops_get_coalesce\n"); | ||||||
| 
 | 
 | ||||||
|  | 	ret = wil_pm_runtime_get(wil); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
| 	tx_itr_en = wil_r(wil, RGF_DMA_ITR_TX_CNT_CTL); | 	tx_itr_en = wil_r(wil, RGF_DMA_ITR_TX_CNT_CTL); | ||||||
| 	if (tx_itr_en & BIT_DMA_ITR_TX_CNT_CTL_EN) | 	if (tx_itr_en & BIT_DMA_ITR_TX_CNT_CTL_EN) | ||||||
| 		tx_itr_val = wil_r(wil, RGF_DMA_ITR_TX_CNT_TRSH); | 		tx_itr_val = wil_r(wil, RGF_DMA_ITR_TX_CNT_TRSH); | ||||||
| @ -58,6 +63,8 @@ static int wil_ethtoolops_get_coalesce(struct net_device *ndev, | |||||||
| 	if (rx_itr_en & BIT_DMA_ITR_RX_CNT_CTL_EN) | 	if (rx_itr_en & BIT_DMA_ITR_RX_CNT_CTL_EN) | ||||||
| 		rx_itr_val = wil_r(wil, RGF_DMA_ITR_RX_CNT_TRSH); | 		rx_itr_val = wil_r(wil, RGF_DMA_ITR_RX_CNT_TRSH); | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
| 	cp->tx_coalesce_usecs = tx_itr_val; | 	cp->tx_coalesce_usecs = tx_itr_val; | ||||||
| 	cp->rx_coalesce_usecs = rx_itr_val; | 	cp->rx_coalesce_usecs = rx_itr_val; | ||||||
| 	return 0; | 	return 0; | ||||||
| @ -67,6 +74,7 @@ static int wil_ethtoolops_set_coalesce(struct net_device *ndev, | |||||||
| 				       struct ethtool_coalesce *cp) | 				       struct ethtool_coalesce *cp) | ||||||
| { | { | ||||||
| 	struct wil6210_priv *wil = ndev_to_wil(ndev); | 	struct wil6210_priv *wil = ndev_to_wil(ndev); | ||||||
|  | 	int ret; | ||||||
| 
 | 
 | ||||||
| 	wil_dbg_misc(wil, "ethtoolops_set_coalesce: rx %d usec, tx %d usec\n", | 	wil_dbg_misc(wil, "ethtoolops_set_coalesce: rx %d usec, tx %d usec\n", | ||||||
| 		     cp->rx_coalesce_usecs, cp->tx_coalesce_usecs); | 		     cp->rx_coalesce_usecs, cp->tx_coalesce_usecs); | ||||||
| @ -86,8 +94,15 @@ static int wil_ethtoolops_set_coalesce(struct net_device *ndev, | |||||||
| 
 | 
 | ||||||
| 	wil->tx_max_burst_duration = cp->tx_coalesce_usecs; | 	wil->tx_max_burst_duration = cp->tx_coalesce_usecs; | ||||||
| 	wil->rx_max_burst_duration = cp->rx_coalesce_usecs; | 	wil->rx_max_burst_duration = cp->rx_coalesce_usecs; | ||||||
|  | 
 | ||||||
|  | 	ret = wil_pm_runtime_get(wil); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
| 	wil_configure_interrupt_moderation(wil); | 	wil_configure_interrupt_moderation(wil); | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| 
 | 
 | ||||||
| out_bad: | out_bad: | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ | |||||||
| static int wil_open(struct net_device *ndev) | static int wil_open(struct net_device *ndev) | ||||||
| { | { | ||||||
| 	struct wil6210_priv *wil = ndev_to_wil(ndev); | 	struct wil6210_priv *wil = ndev_to_wil(ndev); | ||||||
|  | 	int rc; | ||||||
| 
 | 
 | ||||||
| 	wil_dbg_misc(wil, "open\n"); | 	wil_dbg_misc(wil, "open\n"); | ||||||
| 
 | 
 | ||||||
| @ -30,16 +31,29 @@ static int wil_open(struct net_device *ndev) | |||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return wil_up(wil); | 	rc = wil_pm_runtime_get(wil); | ||||||
|  | 	if (rc < 0) | ||||||
|  | 		return rc; | ||||||
|  | 
 | ||||||
|  | 	rc = wil_up(wil); | ||||||
|  | 	if (rc) | ||||||
|  | 		wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
|  | 	return rc; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int wil_stop(struct net_device *ndev) | static int wil_stop(struct net_device *ndev) | ||||||
| { | { | ||||||
| 	struct wil6210_priv *wil = ndev_to_wil(ndev); | 	struct wil6210_priv *wil = ndev_to_wil(ndev); | ||||||
|  | 	int rc; | ||||||
| 
 | 
 | ||||||
| 	wil_dbg_misc(wil, "stop\n"); | 	wil_dbg_misc(wil, "stop\n"); | ||||||
| 
 | 
 | ||||||
| 	return wil_down(wil); | 	rc = wil_down(wil); | ||||||
|  | 	if (!rc) | ||||||
|  | 		wil_pm_runtime_put(wil); | ||||||
|  | 
 | ||||||
|  | 	return rc; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static const struct net_device_ops wil_netdev_ops = { | static const struct net_device_ops wil_netdev_ops = { | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ | |||||||
| #include <linux/suspend.h> | #include <linux/suspend.h> | ||||||
| #include "wil6210.h" | #include "wil6210.h" | ||||||
| #include <linux/rtnetlink.h> | #include <linux/rtnetlink.h> | ||||||
|  | #include <linux/pm_runtime.h> | ||||||
| 
 | 
 | ||||||
| static bool use_msi = true; | static bool use_msi = true; | ||||||
| module_param(use_msi, bool, 0444); | module_param(use_msi, bool, 0444); | ||||||
| @ -31,10 +32,8 @@ module_param(ftm_mode, bool, 0444); | |||||||
| MODULE_PARM_DESC(ftm_mode, " Set factory test mode, default - false"); | MODULE_PARM_DESC(ftm_mode, " Set factory test mode, default - false"); | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_PM | #ifdef CONFIG_PM | ||||||
| #ifdef CONFIG_PM_SLEEP |  | ||||||
| static int wil6210_pm_notify(struct notifier_block *notify_block, | static int wil6210_pm_notify(struct notifier_block *notify_block, | ||||||
| 			     unsigned long mode, void *unused); | 			     unsigned long mode, void *unused); | ||||||
| #endif /* CONFIG_PM_SLEEP */ |  | ||||||
| #endif /* CONFIG_PM */ | #endif /* CONFIG_PM */ | ||||||
| 
 | 
 | ||||||
| static | static | ||||||
| @ -320,7 +319,6 @@ static int wil_pcie_probe(struct pci_dev *pdev, const struct pci_device_id *id) | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_PM | #ifdef CONFIG_PM | ||||||
| #ifdef CONFIG_PM_SLEEP |  | ||||||
| 	wil->pm_notify.notifier_call = wil6210_pm_notify; | 	wil->pm_notify.notifier_call = wil6210_pm_notify; | ||||||
| 	rc = register_pm_notifier(&wil->pm_notify); | 	rc = register_pm_notifier(&wil->pm_notify); | ||||||
| 	if (rc) | 	if (rc) | ||||||
| @ -328,11 +326,11 @@ static int wil_pcie_probe(struct pci_dev *pdev, const struct pci_device_id *id) | |||||||
| 		 * be prevented in a later phase if needed | 		 * be prevented in a later phase if needed | ||||||
| 		 */ | 		 */ | ||||||
| 		wil_err(wil, "register_pm_notifier failed: %d\n", rc); | 		wil_err(wil, "register_pm_notifier failed: %d\n", rc); | ||||||
| #endif /* CONFIG_PM_SLEEP */ |  | ||||||
| #endif /* CONFIG_PM */ | #endif /* CONFIG_PM */ | ||||||
| 
 | 
 | ||||||
| 	wil6210_debugfs_init(wil); | 	wil6210_debugfs_init(wil); | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_allow(wil); | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| 
 | 
 | ||||||
| @ -360,11 +358,11 @@ static void wil_pcie_remove(struct pci_dev *pdev) | |||||||
| 	wil_dbg_misc(wil, "pcie_remove\n"); | 	wil_dbg_misc(wil, "pcie_remove\n"); | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_PM | #ifdef CONFIG_PM | ||||||
| #ifdef CONFIG_PM_SLEEP |  | ||||||
| 	unregister_pm_notifier(&wil->pm_notify); | 	unregister_pm_notifier(&wil->pm_notify); | ||||||
| #endif /* CONFIG_PM_SLEEP */ |  | ||||||
| #endif /* CONFIG_PM */ | #endif /* CONFIG_PM */ | ||||||
| 
 | 
 | ||||||
|  | 	wil_pm_runtime_forbid(wil); | ||||||
|  | 
 | ||||||
| 	wil6210_debugfs_remove(wil); | 	wil6210_debugfs_remove(wil); | ||||||
| 	rtnl_lock(); | 	rtnl_lock(); | ||||||
| 	wil_p2p_wdev_free(wil); | 	wil_p2p_wdev_free(wil); | ||||||
| @ -386,7 +384,6 @@ static const struct pci_device_id wil6210_pcie_ids[] = { | |||||||
| MODULE_DEVICE_TABLE(pci, wil6210_pcie_ids); | MODULE_DEVICE_TABLE(pci, wil6210_pcie_ids); | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_PM | #ifdef CONFIG_PM | ||||||
| #ifdef CONFIG_PM_SLEEP |  | ||||||
| 
 | 
 | ||||||
| static int wil6210_suspend(struct device *dev, bool is_runtime) | static int wil6210_suspend(struct device *dev, bool is_runtime) | ||||||
| { | { | ||||||
| @ -490,12 +487,43 @@ static int wil6210_pm_resume(struct device *dev) | |||||||
| { | { | ||||||
| 	return wil6210_resume(dev, false); | 	return wil6210_resume(dev, false); | ||||||
| } | } | ||||||
| #endif /* CONFIG_PM_SLEEP */ |  | ||||||
| 
 | 
 | ||||||
|  | static int wil6210_pm_runtime_idle(struct device *dev) | ||||||
|  | { | ||||||
|  | 	struct pci_dev *pdev = to_pci_dev(dev); | ||||||
|  | 	struct wil6210_priv *wil = pci_get_drvdata(pdev); | ||||||
|  | 
 | ||||||
|  | 	wil_dbg_pm(wil, "Runtime idle\n"); | ||||||
|  | 
 | ||||||
|  | 	return wil_can_suspend(wil, true); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int wil6210_pm_runtime_resume(struct device *dev) | ||||||
|  | { | ||||||
|  | 	return wil6210_resume(dev, true); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int wil6210_pm_runtime_suspend(struct device *dev) | ||||||
|  | { | ||||||
|  | 	struct pci_dev *pdev = to_pci_dev(dev); | ||||||
|  | 	struct wil6210_priv *wil = pci_get_drvdata(pdev); | ||||||
|  | 
 | ||||||
|  | 	if (test_bit(wil_status_suspended, wil->status)) { | ||||||
|  | 		wil_dbg_pm(wil, "trying to suspend while suspended\n"); | ||||||
|  | 		return 1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return wil6210_suspend(dev, true); | ||||||
|  | } | ||||||
| #endif /* CONFIG_PM */ | #endif /* CONFIG_PM */ | ||||||
| 
 | 
 | ||||||
| static const struct dev_pm_ops wil6210_pm_ops = { | static const struct dev_pm_ops wil6210_pm_ops = { | ||||||
|  | #ifdef CONFIG_PM | ||||||
| 	SET_SYSTEM_SLEEP_PM_OPS(wil6210_pm_suspend, wil6210_pm_resume) | 	SET_SYSTEM_SLEEP_PM_OPS(wil6210_pm_suspend, wil6210_pm_resume) | ||||||
|  | 	SET_RUNTIME_PM_OPS(wil6210_pm_runtime_suspend, | ||||||
|  | 			   wil6210_pm_runtime_resume, | ||||||
|  | 			   wil6210_pm_runtime_idle) | ||||||
|  | #endif /* CONFIG_PM */ | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static struct pci_driver wil6210_driver = { | static struct pci_driver wil6210_driver = { | ||||||
|  | |||||||
| @ -16,15 +16,30 @@ | |||||||
| 
 | 
 | ||||||
| #include "wil6210.h" | #include "wil6210.h" | ||||||
| #include <linux/jiffies.h> | #include <linux/jiffies.h> | ||||||
|  | #include <linux/pm_runtime.h> | ||||||
|  | 
 | ||||||
|  | #define WIL6210_AUTOSUSPEND_DELAY_MS (1000) | ||||||
| 
 | 
 | ||||||
| int wil_can_suspend(struct wil6210_priv *wil, bool is_runtime) | int wil_can_suspend(struct wil6210_priv *wil, bool is_runtime) | ||||||
| { | { | ||||||
| 	int rc = 0; | 	int rc = 0; | ||||||
| 	struct wireless_dev *wdev = wil->wdev; | 	struct wireless_dev *wdev = wil->wdev; | ||||||
| 	struct net_device *ndev = wil_to_ndev(wil); | 	struct net_device *ndev = wil_to_ndev(wil); | ||||||
|  | 	bool wmi_only = test_bit(WMI_FW_CAPABILITY_WMI_ONLY, | ||||||
|  | 				 wil->fw_capabilities); | ||||||
| 
 | 
 | ||||||
| 	wil_dbg_pm(wil, "can_suspend: %s\n", is_runtime ? "runtime" : "system"); | 	wil_dbg_pm(wil, "can_suspend: %s\n", is_runtime ? "runtime" : "system"); | ||||||
| 
 | 
 | ||||||
|  | 	if (wmi_only || debug_fw) { | ||||||
|  | 		wil_dbg_pm(wil, "Deny any suspend - %s mode\n", | ||||||
|  | 			   wmi_only ? "wmi_only" : "debug_fw"); | ||||||
|  | 		rc = -EBUSY; | ||||||
|  | 		goto out; | ||||||
|  | 	} | ||||||
|  | 	if (is_runtime && !wil->platform_ops.suspend) { | ||||||
|  | 		rc = -EBUSY; | ||||||
|  | 		goto out; | ||||||
|  | 	} | ||||||
| 	if (!(ndev->flags & IFF_UP)) { | 	if (!(ndev->flags & IFF_UP)) { | ||||||
| 		/* can always sleep when down */ | 		/* can always sleep when down */ | ||||||
| 		wil_dbg_pm(wil, "Interface is down\n"); | 		wil_dbg_pm(wil, "Interface is down\n"); | ||||||
| @ -44,6 +59,10 @@ int wil_can_suspend(struct wil6210_priv *wil, bool is_runtime) | |||||||
| 	/* interface is running */ | 	/* interface is running */ | ||||||
| 	switch (wdev->iftype) { | 	switch (wdev->iftype) { | ||||||
| 	case NL80211_IFTYPE_MONITOR: | 	case NL80211_IFTYPE_MONITOR: | ||||||
|  | 		wil_dbg_pm(wil, "Sniffer\n"); | ||||||
|  | 		rc = -EBUSY; | ||||||
|  | 		goto out; | ||||||
|  | 	/* for STA-like interface, don't runtime suspend */ | ||||||
| 	case NL80211_IFTYPE_STATION: | 	case NL80211_IFTYPE_STATION: | ||||||
| 	case NL80211_IFTYPE_P2P_CLIENT: | 	case NL80211_IFTYPE_P2P_CLIENT: | ||||||
| 		if (test_bit(wil_status_fwconnecting, wil->status)) { | 		if (test_bit(wil_status_fwconnecting, wil->status)) { | ||||||
| @ -51,6 +70,12 @@ int wil_can_suspend(struct wil6210_priv *wil, bool is_runtime) | |||||||
| 			rc = -EBUSY; | 			rc = -EBUSY; | ||||||
| 			goto out; | 			goto out; | ||||||
| 		} | 		} | ||||||
|  | 		/* Runtime pm not supported in case the interface is up */ | ||||||
|  | 		if (is_runtime) { | ||||||
|  | 			wil_dbg_pm(wil, "STA-like interface\n"); | ||||||
|  | 			rc = -EBUSY; | ||||||
|  | 			goto out; | ||||||
|  | 		} | ||||||
| 		break; | 		break; | ||||||
| 	/* AP-like interface - can't suspend */ | 	/* AP-like interface - can't suspend */ | ||||||
| 	default: | 	default: | ||||||
| @ -348,3 +373,44 @@ out: | |||||||
| 		   is_runtime ? "runtime" : "system", rc, suspend_time_usec); | 		   is_runtime ? "runtime" : "system", rc, suspend_time_usec); | ||||||
| 	return rc; | 	return rc; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | void wil_pm_runtime_allow(struct wil6210_priv *wil) | ||||||
|  | { | ||||||
|  | 	struct device *dev = wil_to_dev(wil); | ||||||
|  | 
 | ||||||
|  | 	pm_runtime_put_noidle(dev); | ||||||
|  | 	pm_runtime_set_autosuspend_delay(dev, WIL6210_AUTOSUSPEND_DELAY_MS); | ||||||
|  | 	pm_runtime_use_autosuspend(dev); | ||||||
|  | 	pm_runtime_allow(dev); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wil_pm_runtime_forbid(struct wil6210_priv *wil) | ||||||
|  | { | ||||||
|  | 	struct device *dev = wil_to_dev(wil); | ||||||
|  | 
 | ||||||
|  | 	pm_runtime_forbid(dev); | ||||||
|  | 	pm_runtime_get_noresume(dev); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int wil_pm_runtime_get(struct wil6210_priv *wil) | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  | 	struct device *dev = wil_to_dev(wil); | ||||||
|  | 
 | ||||||
|  | 	rc = pm_runtime_get_sync(dev); | ||||||
|  | 	if (rc < 0) { | ||||||
|  | 		wil_err(wil, "pm_runtime_get_sync() failed, rc = %d\n", rc); | ||||||
|  | 		pm_runtime_put_noidle(dev); | ||||||
|  | 		return rc; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wil_pm_runtime_put(struct wil6210_priv *wil) | ||||||
|  | { | ||||||
|  | 	struct device *dev = wil_to_dev(wil); | ||||||
|  | 
 | ||||||
|  | 	pm_runtime_mark_last_busy(dev); | ||||||
|  | 	pm_runtime_put_autosuspend(dev); | ||||||
|  | } | ||||||
|  | |||||||
| @ -616,6 +616,16 @@ struct blink_on_off_time { | |||||||
| 	u32 off_ms; | 	u32 off_ms; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | struct wil_debugfs_iomem_data { | ||||||
|  | 	void *offset; | ||||||
|  | 	struct wil6210_priv *wil; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct wil_debugfs_data { | ||||||
|  | 	struct wil_debugfs_iomem_data *data_arr; | ||||||
|  | 	int iomem_data_count; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| extern struct blink_on_off_time led_blink_time[WIL_LED_TIME_LAST]; | extern struct blink_on_off_time led_blink_time[WIL_LED_TIME_LAST]; | ||||||
| extern u8 led_id; | extern u8 led_id; | ||||||
| extern u8 led_polarity; | extern u8 led_polarity; | ||||||
| @ -708,6 +718,7 @@ struct wil6210_priv { | |||||||
| 	u8 abft_len; | 	u8 abft_len; | ||||||
| 	u8 wakeup_trigger; | 	u8 wakeup_trigger; | ||||||
| 	struct wil_suspend_stats suspend_stats; | 	struct wil_suspend_stats suspend_stats; | ||||||
|  | 	struct wil_debugfs_data dbg_data; | ||||||
| 
 | 
 | ||||||
| 	void *platform_handle; | 	void *platform_handle; | ||||||
| 	struct wil_platform_ops platform_ops; | 	struct wil_platform_ops platform_ops; | ||||||
| @ -732,9 +743,7 @@ struct wil6210_priv { | |||||||
| 	int fw_calib_result; | 	int fw_calib_result; | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_PM | #ifdef CONFIG_PM | ||||||
| #ifdef CONFIG_PM_SLEEP |  | ||||||
| 	struct notifier_block pm_notify; | 	struct notifier_block pm_notify; | ||||||
| #endif /* CONFIG_PM_SLEEP */ |  | ||||||
| #endif /* CONFIG_PM */ | #endif /* CONFIG_PM */ | ||||||
| 
 | 
 | ||||||
| 	bool suspend_resp_rcvd; | 	bool suspend_resp_rcvd; | ||||||
| @ -999,6 +1008,11 @@ int wil_request_firmware(struct wil6210_priv *wil, const char *name, | |||||||
| 			 bool load); | 			 bool load); | ||||||
| bool wil_fw_verify_file_exists(struct wil6210_priv *wil, const char *name); | bool wil_fw_verify_file_exists(struct wil6210_priv *wil, const char *name); | ||||||
| 
 | 
 | ||||||
|  | void wil_pm_runtime_allow(struct wil6210_priv *wil); | ||||||
|  | void wil_pm_runtime_forbid(struct wil6210_priv *wil); | ||||||
|  | int wil_pm_runtime_get(struct wil6210_priv *wil); | ||||||
|  | void wil_pm_runtime_put(struct wil6210_priv *wil); | ||||||
|  | 
 | ||||||
| int wil_can_suspend(struct wil6210_priv *wil, bool is_runtime); | int wil_can_suspend(struct wil6210_priv *wil, bool is_runtime); | ||||||
| int wil_suspend(struct wil6210_priv *wil, bool is_runtime); | int wil_suspend(struct wil6210_priv *wil, bool is_runtime); | ||||||
| int wil_resume(struct wil6210_priv *wil, bool is_runtime); | int wil_resume(struct wil6210_priv *wil, bool is_runtime); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Lazar Alexei
						Lazar Alexei