Commit 0dcfe41e authored by Dave Jiang's avatar Dave Jiang Committed by Vinod Koul

dmanegine: idxd: cleanup all device related bits after disabling device

The previous state cleanup patch only performed wq state cleanups. This
does not go far enough as when device is disabled or reset, the state
for groups and engines must also be cleaned up. Add additional state
cleanup beyond wq cleanup. Tie those cleanups directly to device
disable and reset, and wq disable and reset.

Fixes: da32b28c ("dmaengine: idxd: cleanup workqueue config after disabling")
Signed-off-by: default avatarDave Jiang <dave.jiang@intel.com>
Link: https://lore.kernel.org/r/162285154108.2096632.5572805472362321307.stgit@djiang5-desk3.ch.intel.comSigned-off-by: default avatarVinod Koul <vkoul@kernel.org>
parent 53b50458
...@@ -15,6 +15,8 @@ ...@@ -15,6 +15,8 @@
static void idxd_cmd_exec(struct idxd_device *idxd, int cmd_code, u32 operand, static void idxd_cmd_exec(struct idxd_device *idxd, int cmd_code, u32 operand,
u32 *status); u32 *status);
static void idxd_device_wqs_clear_state(struct idxd_device *idxd);
static void idxd_wq_disable_cleanup(struct idxd_wq *wq);
/* Interrupt control bits */ /* Interrupt control bits */
void idxd_mask_msix_vector(struct idxd_device *idxd, int vec_id) void idxd_mask_msix_vector(struct idxd_device *idxd, int vec_id)
...@@ -234,7 +236,7 @@ int idxd_wq_enable(struct idxd_wq *wq) ...@@ -234,7 +236,7 @@ int idxd_wq_enable(struct idxd_wq *wq)
return 0; return 0;
} }
int idxd_wq_disable(struct idxd_wq *wq) int idxd_wq_disable(struct idxd_wq *wq, bool reset_config)
{ {
struct idxd_device *idxd = wq->idxd; struct idxd_device *idxd = wq->idxd;
struct device *dev = &idxd->pdev->dev; struct device *dev = &idxd->pdev->dev;
...@@ -255,6 +257,8 @@ int idxd_wq_disable(struct idxd_wq *wq) ...@@ -255,6 +257,8 @@ int idxd_wq_disable(struct idxd_wq *wq)
return -ENXIO; return -ENXIO;
} }
if (reset_config)
idxd_wq_disable_cleanup(wq);
wq->state = IDXD_WQ_DISABLED; wq->state = IDXD_WQ_DISABLED;
dev_dbg(dev, "WQ %d disabled\n", wq->id); dev_dbg(dev, "WQ %d disabled\n", wq->id);
return 0; return 0;
...@@ -289,6 +293,7 @@ void idxd_wq_reset(struct idxd_wq *wq) ...@@ -289,6 +293,7 @@ void idxd_wq_reset(struct idxd_wq *wq)
operand = BIT(wq->id % 16) | ((wq->id / 16) << 16); operand = BIT(wq->id % 16) | ((wq->id / 16) << 16);
idxd_cmd_exec(idxd, IDXD_CMD_RESET_WQ, operand, NULL); idxd_cmd_exec(idxd, IDXD_CMD_RESET_WQ, operand, NULL);
idxd_wq_disable_cleanup(wq);
wq->state = IDXD_WQ_DISABLED; wq->state = IDXD_WQ_DISABLED;
} }
...@@ -337,7 +342,7 @@ int idxd_wq_set_pasid(struct idxd_wq *wq, int pasid) ...@@ -337,7 +342,7 @@ int idxd_wq_set_pasid(struct idxd_wq *wq, int pasid)
unsigned int offset; unsigned int offset;
unsigned long flags; unsigned long flags;
rc = idxd_wq_disable(wq); rc = idxd_wq_disable(wq, false);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -364,7 +369,7 @@ int idxd_wq_disable_pasid(struct idxd_wq *wq) ...@@ -364,7 +369,7 @@ int idxd_wq_disable_pasid(struct idxd_wq *wq)
unsigned int offset; unsigned int offset;
unsigned long flags; unsigned long flags;
rc = idxd_wq_disable(wq); rc = idxd_wq_disable(wq, false);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -383,11 +388,11 @@ int idxd_wq_disable_pasid(struct idxd_wq *wq) ...@@ -383,11 +388,11 @@ int idxd_wq_disable_pasid(struct idxd_wq *wq)
return 0; return 0;
} }
void idxd_wq_disable_cleanup(struct idxd_wq *wq) static void idxd_wq_disable_cleanup(struct idxd_wq *wq)
{ {
struct idxd_device *idxd = wq->idxd; struct idxd_device *idxd = wq->idxd;
lockdep_assert_held(&idxd->dev_lock); lockdep_assert_held(&wq->wq_lock);
memset(wq->wqcfg, 0, idxd->wqcfg_size); memset(wq->wqcfg, 0, idxd->wqcfg_size);
wq->type = IDXD_WQT_NONE; wq->type = IDXD_WQT_NONE;
wq->size = 0; wq->size = 0;
...@@ -548,22 +553,6 @@ int idxd_device_enable(struct idxd_device *idxd) ...@@ -548,22 +553,6 @@ int idxd_device_enable(struct idxd_device *idxd)
return 0; return 0;
} }
void idxd_device_wqs_clear_state(struct idxd_device *idxd)
{
int i;
lockdep_assert_held(&idxd->dev_lock);
for (i = 0; i < idxd->max_wqs; i++) {
struct idxd_wq *wq = idxd->wqs[i];
if (wq->state == IDXD_WQ_ENABLED) {
idxd_wq_disable_cleanup(wq);
wq->state = IDXD_WQ_DISABLED;
}
}
}
int idxd_device_disable(struct idxd_device *idxd) int idxd_device_disable(struct idxd_device *idxd)
{ {
struct device *dev = &idxd->pdev->dev; struct device *dev = &idxd->pdev->dev;
...@@ -585,7 +574,7 @@ int idxd_device_disable(struct idxd_device *idxd) ...@@ -585,7 +574,7 @@ int idxd_device_disable(struct idxd_device *idxd)
} }
spin_lock_irqsave(&idxd->dev_lock, flags); spin_lock_irqsave(&idxd->dev_lock, flags);
idxd_device_wqs_clear_state(idxd); idxd_device_clear_state(idxd);
idxd->state = IDXD_DEV_CONF_READY; idxd->state = IDXD_DEV_CONF_READY;
spin_unlock_irqrestore(&idxd->dev_lock, flags); spin_unlock_irqrestore(&idxd->dev_lock, flags);
return 0; return 0;
...@@ -597,7 +586,7 @@ void idxd_device_reset(struct idxd_device *idxd) ...@@ -597,7 +586,7 @@ void idxd_device_reset(struct idxd_device *idxd)
idxd_cmd_exec(idxd, IDXD_CMD_RESET_DEVICE, 0, NULL); idxd_cmd_exec(idxd, IDXD_CMD_RESET_DEVICE, 0, NULL);
spin_lock_irqsave(&idxd->dev_lock, flags); spin_lock_irqsave(&idxd->dev_lock, flags);
idxd_device_wqs_clear_state(idxd); idxd_device_clear_state(idxd);
idxd->state = IDXD_DEV_CONF_READY; idxd->state = IDXD_DEV_CONF_READY;
spin_unlock_irqrestore(&idxd->dev_lock, flags); spin_unlock_irqrestore(&idxd->dev_lock, flags);
} }
...@@ -685,6 +674,59 @@ int idxd_device_release_int_handle(struct idxd_device *idxd, int handle, ...@@ -685,6 +674,59 @@ int idxd_device_release_int_handle(struct idxd_device *idxd, int handle,
} }
/* Device configuration bits */ /* Device configuration bits */
static void idxd_engines_clear_state(struct idxd_device *idxd)
{
struct idxd_engine *engine;
int i;
lockdep_assert_held(&idxd->dev_lock);
for (i = 0; i < idxd->max_engines; i++) {
engine = idxd->engines[i];
engine->group = NULL;
}
}
static void idxd_groups_clear_state(struct idxd_device *idxd)
{
struct idxd_group *group;
int i;
lockdep_assert_held(&idxd->dev_lock);
for (i = 0; i < idxd->max_groups; i++) {
group = idxd->groups[i];
memset(&group->grpcfg, 0, sizeof(group->grpcfg));
group->num_engines = 0;
group->num_wqs = 0;
group->use_token_limit = false;
group->tokens_allowed = 0;
group->tokens_reserved = 0;
group->tc_a = -1;
group->tc_b = -1;
}
}
static void idxd_device_wqs_clear_state(struct idxd_device *idxd)
{
int i;
lockdep_assert_held(&idxd->dev_lock);
for (i = 0; i < idxd->max_wqs; i++) {
struct idxd_wq *wq = idxd->wqs[i];
if (wq->state == IDXD_WQ_ENABLED) {
idxd_wq_disable_cleanup(wq);
wq->state = IDXD_WQ_DISABLED;
}
}
}
void idxd_device_clear_state(struct idxd_device *idxd)
{
idxd_groups_clear_state(idxd);
idxd_engines_clear_state(idxd);
idxd_device_wqs_clear_state(idxd);
}
void idxd_msix_perm_setup(struct idxd_device *idxd) void idxd_msix_perm_setup(struct idxd_device *idxd)
{ {
union msix_perm mperm; union msix_perm mperm;
......
...@@ -420,9 +420,8 @@ int idxd_device_init_reset(struct idxd_device *idxd); ...@@ -420,9 +420,8 @@ int idxd_device_init_reset(struct idxd_device *idxd);
int idxd_device_enable(struct idxd_device *idxd); int idxd_device_enable(struct idxd_device *idxd);
int idxd_device_disable(struct idxd_device *idxd); int idxd_device_disable(struct idxd_device *idxd);
void idxd_device_reset(struct idxd_device *idxd); void idxd_device_reset(struct idxd_device *idxd);
void idxd_device_cleanup(struct idxd_device *idxd); void idxd_device_clear_state(struct idxd_device *idxd);
int idxd_device_config(struct idxd_device *idxd); int idxd_device_config(struct idxd_device *idxd);
void idxd_device_wqs_clear_state(struct idxd_device *idxd);
void idxd_device_drain_pasid(struct idxd_device *idxd, int pasid); void idxd_device_drain_pasid(struct idxd_device *idxd, int pasid);
int idxd_device_load_config(struct idxd_device *idxd); int idxd_device_load_config(struct idxd_device *idxd);
int idxd_device_request_int_handle(struct idxd_device *idxd, int idx, int *handle, int idxd_device_request_int_handle(struct idxd_device *idxd, int idx, int *handle,
...@@ -435,12 +434,11 @@ void idxd_wqs_unmap_portal(struct idxd_device *idxd); ...@@ -435,12 +434,11 @@ void idxd_wqs_unmap_portal(struct idxd_device *idxd);
int idxd_wq_alloc_resources(struct idxd_wq *wq); int idxd_wq_alloc_resources(struct idxd_wq *wq);
void idxd_wq_free_resources(struct idxd_wq *wq); void idxd_wq_free_resources(struct idxd_wq *wq);
int idxd_wq_enable(struct idxd_wq *wq); int idxd_wq_enable(struct idxd_wq *wq);
int idxd_wq_disable(struct idxd_wq *wq); int idxd_wq_disable(struct idxd_wq *wq, bool reset_config);
void idxd_wq_drain(struct idxd_wq *wq); void idxd_wq_drain(struct idxd_wq *wq);
void idxd_wq_reset(struct idxd_wq *wq); void idxd_wq_reset(struct idxd_wq *wq);
int idxd_wq_map_portal(struct idxd_wq *wq); int idxd_wq_map_portal(struct idxd_wq *wq);
void idxd_wq_unmap_portal(struct idxd_wq *wq); void idxd_wq_unmap_portal(struct idxd_wq *wq);
void idxd_wq_disable_cleanup(struct idxd_wq *wq);
int idxd_wq_set_pasid(struct idxd_wq *wq, int pasid); int idxd_wq_set_pasid(struct idxd_wq *wq, int pasid);
int idxd_wq_disable_pasid(struct idxd_wq *wq); int idxd_wq_disable_pasid(struct idxd_wq *wq);
void idxd_wq_quiesce(struct idxd_wq *wq); void idxd_wq_quiesce(struct idxd_wq *wq);
......
...@@ -59,7 +59,7 @@ static void idxd_device_reinit(struct work_struct *work) ...@@ -59,7 +59,7 @@ static void idxd_device_reinit(struct work_struct *work)
return; return;
out: out:
idxd_device_wqs_clear_state(idxd); idxd_device_clear_state(idxd);
} }
static void idxd_device_fault_work(struct work_struct *work) static void idxd_device_fault_work(struct work_struct *work)
...@@ -192,7 +192,7 @@ static int process_misc_interrupts(struct idxd_device *idxd, u32 cause) ...@@ -192,7 +192,7 @@ static int process_misc_interrupts(struct idxd_device *idxd, u32 cause)
spin_lock_bh(&idxd->dev_lock); spin_lock_bh(&idxd->dev_lock);
idxd_wqs_quiesce(idxd); idxd_wqs_quiesce(idxd);
idxd_wqs_unmap_portal(idxd); idxd_wqs_unmap_portal(idxd);
idxd_device_wqs_clear_state(idxd); idxd_device_clear_state(idxd);
dev_err(&idxd->pdev->dev, dev_err(&idxd->pdev->dev,
"idxd halted, need %s.\n", "idxd halted, need %s.\n",
gensts.reset_type == IDXD_DEVICE_RESET_FLR ? gensts.reset_type == IDXD_DEVICE_RESET_FLR ?
......
...@@ -129,7 +129,7 @@ static int enable_wq(struct idxd_wq *wq) ...@@ -129,7 +129,7 @@ static int enable_wq(struct idxd_wq *wq)
rc = idxd_wq_map_portal(wq); rc = idxd_wq_map_portal(wq);
if (rc < 0) { if (rc < 0) {
dev_warn(dev, "wq portal mapping failed: %d\n", rc); dev_warn(dev, "wq portal mapping failed: %d\n", rc);
rc = idxd_wq_disable(wq); rc = idxd_wq_disable(wq, false);
if (rc < 0) if (rc < 0)
dev_warn(dev, "IDXD wq disable failed\n"); dev_warn(dev, "IDXD wq disable failed\n");
mutex_unlock(&wq->wq_lock); mutex_unlock(&wq->wq_lock);
...@@ -262,8 +262,6 @@ static void disable_wq(struct idxd_wq *wq) ...@@ -262,8 +262,6 @@ static void disable_wq(struct idxd_wq *wq)
static int idxd_config_bus_remove(struct device *dev) static int idxd_config_bus_remove(struct device *dev)
{ {
int rc;
dev_dbg(dev, "%s called for %s\n", __func__, dev_name(dev)); dev_dbg(dev, "%s called for %s\n", __func__, dev_name(dev));
/* disable workqueue here */ /* disable workqueue here */
...@@ -288,22 +286,12 @@ static int idxd_config_bus_remove(struct device *dev) ...@@ -288,22 +286,12 @@ static int idxd_config_bus_remove(struct device *dev)
} }
idxd_unregister_dma_device(idxd); idxd_unregister_dma_device(idxd);
rc = idxd_device_disable(idxd); idxd_device_disable(idxd);
if (test_bit(IDXD_FLAG_CONFIGURABLE, &idxd->flags)) { if (test_bit(IDXD_FLAG_CONFIGURABLE, &idxd->flags))
for (i = 0; i < idxd->max_wqs; i++) { idxd_device_reset(idxd);
struct idxd_wq *wq = idxd->wqs[i];
mutex_lock(&wq->wq_lock);
idxd_wq_disable_cleanup(wq);
mutex_unlock(&wq->wq_lock);
}
}
module_put(THIS_MODULE); module_put(THIS_MODULE);
if (rc < 0)
dev_warn(dev, "Device disable failed\n");
else
dev_info(dev, "Device %s disabled\n", dev_name(dev));
dev_info(dev, "Device %s disabled\n", dev_name(dev));
} }
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment