[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20170217020903.6370-5-mcgrof@kernel.org>
Date: Thu, 16 Feb 2017 18:09:02 -0800
From: "Luis R. Rodriguez" <mcgrof@...nel.org>
To: johannes.berg@...el.com, luciano.coelho@...el.com,
emmanuel.grumbach@...el.com, tj@...nel.org, arjan@...ux.intel.com,
ming.lei@...onical.com, zajec5@...il.com
Cc: jeyu@...hat.com, rusty@...tcorp.com.au, pmladek@...e.com,
gregkh@...uxfoundation.org, linuxwifi@...el.com,
linux-wireless@...r.kernel.org, linux-kernel@...r.kernel.org,
"Luis R. Rodriguez" <mcgrof@...nel.org>
Subject: [RFC 4/5] iwlwifi: move opmode loading to shared routine
This helps us compartmentalize all last required opmode work
and declutter the async firmware callback.
Signed-off-by: Luis R. Rodriguez <mcgrof@...nel.org>
---
drivers/net/wireless/intel/iwlwifi/iwl-drv.c | 33 +++++++++++++++-------------
1 file changed, 18 insertions(+), 15 deletions(-)
diff --git a/drivers/net/wireless/intel/iwlwifi/iwl-drv.c b/drivers/net/wireless/intel/iwlwifi/iwl-drv.c
index 4a1937c77f90..6cfbc3c6e0d6 100644
--- a/drivers/net/wireless/intel/iwlwifi/iwl-drv.c
+++ b/drivers/net/wireless/intel/iwlwifi/iwl-drv.c
@@ -139,6 +139,8 @@ static struct iwlwifi_opmode_table {
const char *name; /* name: iwldvm, iwlmvm, etc */
const struct iwl_op_mode_ops *ops; /* pointer to op_mode ops */
struct list_head drv; /* list of devices using this op_mode */
+ bool load_requested; /* Do we need to load a driver ? */
+ struct iwl_drv *drv_req; /* Device that set load_requested */
} iwlwifi_opmode_table[] = { /* ops set when driver is initialized */
[DVM_OP_MODE] = { .name = "iwldvm", .ops = NULL },
[MVM_OP_MODE] = { .name = "iwlmvm", .ops = NULL },
@@ -1233,27 +1235,32 @@ static void _iwl_op_mode_stop(struct iwl_drv *drv)
}
}
-static void iwlwifi_try_load_op(struct iwlwifi_opmode_table *op,
- struct iwl_drv *drv)
+/* We have to unlock and then lock for the caller */
+static void iwlwifi_try_load_op(struct iwlwifi_opmode_table *op)
{
int ret = 0;
+ mutex_unlock(&iwlwifi_opmode_table_mtx);
+
ret = request_module("%s", op->name);
- if (ret)
+ if (ret) {
+ mutex_lock(&iwlwifi_opmode_table_mtx);
goto out;
+ }
mutex_lock(&iwlwifi_opmode_table_mtx);
if (!op->ops)
ret = -ENOENT;
- mutex_unlock(&iwlwifi_opmode_table_mtx);
out:
#ifdef CONFIG_IWLWIFI_OPMODE_MODULAR
if (ret)
- IWL_ERR(drv,
+ IWL_ERR(op->drv_req,
"failed to load module %s (error %d), is dynamic loading enabled?\n",
op->name, ret);
#endif
+ op->load_requested = false;
+ op->drv_req = NULL;
}
static void iwlwifi_opmode_start_drv(struct iwlwifi_opmode_table *op,
@@ -1294,6 +1301,8 @@ static void iwlwifi_opmode_dowork(void)
op = &iwlwifi_opmode_table[i];
if (op->ops)
iwlwifi_opmode_start(op);
+ else if (op->load_requested)
+ iwlwifi_try_load_op(op);
}
mutex_unlock(&iwlwifi_opmode_table_mtx);
}
@@ -1317,7 +1326,6 @@ static void iwl_req_fw_callback(const struct firmware *ucode_raw, void *context)
size_t trigger_tlv_sz[FW_DBG_TRIGGER_MAX];
u32 api_ver;
int i;
- bool load_module = false;
bool usniffer_images = false;
fw->ucode_capa.max_probe_length = IWL_DEFAULT_MAX_PROBE_LENGTH;
@@ -1515,18 +1523,13 @@ static void iwl_req_fw_callback(const struct firmware *ucode_raw, void *context)
/* add this device to the list of devices using this op_mode */
list_add_tail(&drv->list, &op->drv);
- if (!op->ops)
- load_module = true;
+ if (!op->ops) {
+ op->load_requested = true;
+ op->drv_req = drv;
+ }
mutex_unlock(&iwlwifi_opmode_table_mtx);
- /*
- * Load the module last so we don't block anything
- * else from proceeding if the module fails to load
- * or hangs loading.
- */
- if (load_module)
- iwlwifi_try_load_op(op, drv);
iwlwifi_opmode_dowork();
goto free;
--
2.11.0
Powered by blists - more mailing lists