soc/fsl/qbman: Fixup qman_shutdown_fq()
When shutting down a FQ on a dedicated channel only the SW portal associated with that channel can dequeue from it. Make sure the correct portal is use. Signed-off-by: Roy Pledge <roy.pledge@nxp.com> Signed-off-by: Li Yang <leoyang.li@nxp.com>
This commit is contained in:
parent
ea2b8488ba
commit
e844168a8f
|
@ -1018,6 +1018,20 @@ static inline void put_affine_portal(void)
|
||||||
put_cpu_var(qman_affine_portal);
|
put_cpu_var(qman_affine_portal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static inline struct qman_portal *get_portal_for_channel(u16 channel)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < num_possible_cpus(); i++) {
|
||||||
|
if (affine_portals[i] &&
|
||||||
|
affine_portals[i]->config->channel == channel)
|
||||||
|
return affine_portals[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
static struct workqueue_struct *qm_portal_wq;
|
static struct workqueue_struct *qm_portal_wq;
|
||||||
|
|
||||||
int qman_dqrr_set_ithresh(struct qman_portal *portal, u8 ithresh)
|
int qman_dqrr_set_ithresh(struct qman_portal *portal, u8 ithresh)
|
||||||
|
@ -2601,7 +2615,7 @@ static int _qm_dqrr_consume_and_match(struct qm_portal *p, u32 fqid, int s,
|
||||||
|
|
||||||
int qman_shutdown_fq(u32 fqid)
|
int qman_shutdown_fq(u32 fqid)
|
||||||
{
|
{
|
||||||
struct qman_portal *p;
|
struct qman_portal *p, *channel_portal;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
union qm_mc_command *mcc;
|
union qm_mc_command *mcc;
|
||||||
union qm_mc_result *mcr;
|
union qm_mc_result *mcr;
|
||||||
|
@ -2641,17 +2655,28 @@ int qman_shutdown_fq(u32 fqid)
|
||||||
channel = qm_fqd_get_chan(&mcr->queryfq.fqd);
|
channel = qm_fqd_get_chan(&mcr->queryfq.fqd);
|
||||||
wq = qm_fqd_get_wq(&mcr->queryfq.fqd);
|
wq = qm_fqd_get_wq(&mcr->queryfq.fqd);
|
||||||
|
|
||||||
|
if (channel < qm_channel_pool1) {
|
||||||
|
channel_portal = get_portal_for_channel(channel);
|
||||||
|
if (channel_portal == NULL) {
|
||||||
|
dev_err(dev, "Can't find portal for dedicated channel 0x%x\n",
|
||||||
|
channel);
|
||||||
|
ret = -EIO;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
channel_portal = p;
|
||||||
|
|
||||||
switch (state) {
|
switch (state) {
|
||||||
case QM_MCR_NP_STATE_TEN_SCHED:
|
case QM_MCR_NP_STATE_TEN_SCHED:
|
||||||
case QM_MCR_NP_STATE_TRU_SCHED:
|
case QM_MCR_NP_STATE_TRU_SCHED:
|
||||||
case QM_MCR_NP_STATE_ACTIVE:
|
case QM_MCR_NP_STATE_ACTIVE:
|
||||||
case QM_MCR_NP_STATE_PARKED:
|
case QM_MCR_NP_STATE_PARKED:
|
||||||
orl_empty = 0;
|
orl_empty = 0;
|
||||||
mcc = qm_mc_start(&p->p);
|
mcc = qm_mc_start(&channel_portal->p);
|
||||||
qm_fqid_set(&mcc->fq, fqid);
|
qm_fqid_set(&mcc->fq, fqid);
|
||||||
qm_mc_commit(&p->p, QM_MCC_VERB_ALTER_RETIRE);
|
qm_mc_commit(&channel_portal->p, QM_MCC_VERB_ALTER_RETIRE);
|
||||||
if (!qm_mc_result_timeout(&p->p, &mcr)) {
|
if (!qm_mc_result_timeout(&channel_portal->p, &mcr)) {
|
||||||
dev_err(dev, "QUERYFQ_NP timeout\n");
|
dev_err(dev, "ALTER_RETIRE timeout\n");
|
||||||
ret = -ETIMEDOUT;
|
ret = -ETIMEDOUT;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -2659,6 +2684,9 @@ int qman_shutdown_fq(u32 fqid)
|
||||||
QM_MCR_VERB_ALTER_RETIRE);
|
QM_MCR_VERB_ALTER_RETIRE);
|
||||||
res = mcr->result; /* Make a copy as we reuse MCR below */
|
res = mcr->result; /* Make a copy as we reuse MCR below */
|
||||||
|
|
||||||
|
if (res == QM_MCR_RESULT_OK)
|
||||||
|
drain_mr_fqrni(&channel_portal->p);
|
||||||
|
|
||||||
if (res == QM_MCR_RESULT_PENDING) {
|
if (res == QM_MCR_RESULT_PENDING) {
|
||||||
/*
|
/*
|
||||||
* Need to wait for the FQRN in the message ring, which
|
* Need to wait for the FQRN in the message ring, which
|
||||||
|
@ -2688,21 +2716,25 @@ int qman_shutdown_fq(u32 fqid)
|
||||||
}
|
}
|
||||||
/* Set the sdqcr to drain this channel */
|
/* Set the sdqcr to drain this channel */
|
||||||
if (channel < qm_channel_pool1)
|
if (channel < qm_channel_pool1)
|
||||||
qm_dqrr_sdqcr_set(&p->p,
|
qm_dqrr_sdqcr_set(&channel_portal->p,
|
||||||
QM_SDQCR_TYPE_ACTIVE |
|
QM_SDQCR_TYPE_ACTIVE |
|
||||||
QM_SDQCR_CHANNELS_DEDICATED);
|
QM_SDQCR_CHANNELS_DEDICATED);
|
||||||
else
|
else
|
||||||
qm_dqrr_sdqcr_set(&p->p,
|
qm_dqrr_sdqcr_set(&channel_portal->p,
|
||||||
QM_SDQCR_TYPE_ACTIVE |
|
QM_SDQCR_TYPE_ACTIVE |
|
||||||
QM_SDQCR_CHANNELS_POOL_CONV
|
QM_SDQCR_CHANNELS_POOL_CONV
|
||||||
(channel));
|
(channel));
|
||||||
do {
|
do {
|
||||||
/* Keep draining DQRR while checking the MR*/
|
/* Keep draining DQRR while checking the MR*/
|
||||||
qm_dqrr_drain_nomatch(&p->p);
|
qm_dqrr_drain_nomatch(&channel_portal->p);
|
||||||
/* Process message ring too */
|
/* Process message ring too */
|
||||||
found_fqrn = qm_mr_drain(&p->p, FQRN);
|
found_fqrn = qm_mr_drain(&channel_portal->p,
|
||||||
|
FQRN);
|
||||||
cpu_relax();
|
cpu_relax();
|
||||||
} while (!found_fqrn);
|
} while (!found_fqrn);
|
||||||
|
/* Restore SDQCR */
|
||||||
|
qm_dqrr_sdqcr_set(&channel_portal->p,
|
||||||
|
channel_portal->sdqcr);
|
||||||
|
|
||||||
}
|
}
|
||||||
if (res != QM_MCR_RESULT_OK &&
|
if (res != QM_MCR_RESULT_OK &&
|
||||||
|
@ -2733,9 +2765,8 @@ int qman_shutdown_fq(u32 fqid)
|
||||||
* Wait for a dequeue and process the dequeues,
|
* Wait for a dequeue and process the dequeues,
|
||||||
* making sure to empty the ring completely
|
* making sure to empty the ring completely
|
||||||
*/
|
*/
|
||||||
} while (qm_dqrr_drain_wait(&p->p, fqid, FQ_EMPTY));
|
} while (!qm_dqrr_drain_wait(&p->p, fqid, FQ_EMPTY));
|
||||||
}
|
}
|
||||||
qm_dqrr_sdqcr_set(&p->p, 0);
|
|
||||||
|
|
||||||
while (!orl_empty) {
|
while (!orl_empty) {
|
||||||
/* Wait for the ORL to have been completely drained */
|
/* Wait for the ORL to have been completely drained */
|
||||||
|
|
Loading…
Reference in New Issue