event/octeontx2: unlink queues during port release

Unlinking queues from port should be done during port release. Doing it
during device re-configuration could result in segfault as ports array
is re-allocated based on new number of ports.

Fixes: f7ac8b66b23c ("event/octeontx2: support linking queues to ports")
Cc: stable@dpdk.org

Signed-off-by: Shijith Thotton <sthotton@marvell.com>
Signed-off-by: Pavan Nikhilesh <pbhagavatula@marvell.com>
This commit is contained in:
Shijith Thotton 2020-11-19 14:16:11 +05:30 committed by Jerin Jacob
parent 52f2fcb3e2
commit 8e6663b8c1
3 changed files with 89 additions and 47 deletions

View File

@ -689,7 +689,36 @@ sso_lf_cfg(struct otx2_sso_evdev *dev, struct otx2_mbox *mbox,
static void static void
otx2_sso_port_release(void *port) otx2_sso_port_release(void *port)
{ {
rte_free(port); struct otx2_ssogws_cookie *gws_cookie = ssogws_get_cookie(port);
struct otx2_sso_evdev *dev;
int i;
if (!gws_cookie->configured)
goto free;
dev = sso_pmd_priv(gws_cookie->event_dev);
if (dev->dual_ws) {
struct otx2_ssogws_dual *ws = port;
for (i = 0; i < dev->nb_event_queues; i++) {
sso_port_link_modify((struct otx2_ssogws *)
&ws->ws_state[0], i, false);
sso_port_link_modify((struct otx2_ssogws *)
&ws->ws_state[1], i, false);
}
memset(ws, 0, sizeof(*ws));
} else {
struct otx2_ssogws *ws = port;
for (i = 0; i < dev->nb_event_queues; i++)
sso_port_link_modify(ws, i, false);
memset(ws, 0, sizeof(*ws));
}
memset(gws_cookie, 0, sizeof(*gws_cookie));
free:
rte_free(gws_cookie);
} }
static void static void
@ -699,33 +728,6 @@ otx2_sso_queue_release(struct rte_eventdev *event_dev, uint8_t queue_id)
RTE_SET_USED(queue_id); RTE_SET_USED(queue_id);
} }
static void
sso_clr_links(const struct rte_eventdev *event_dev)
{
struct otx2_sso_evdev *dev = sso_pmd_priv(event_dev);
int i, j;
for (i = 0; i < dev->nb_event_ports; i++) {
if (dev->dual_ws) {
struct otx2_ssogws_dual *ws;
ws = event_dev->data->ports[i];
for (j = 0; j < dev->nb_event_queues; j++) {
sso_port_link_modify((struct otx2_ssogws *)
&ws->ws_state[0], j, false);
sso_port_link_modify((struct otx2_ssogws *)
&ws->ws_state[1], j, false);
}
} else {
struct otx2_ssogws *ws;
ws = event_dev->data->ports[i];
for (j = 0; j < dev->nb_event_queues; j++)
sso_port_link_modify(ws, j, false);
}
}
}
static void static void
sso_restore_links(const struct rte_eventdev *event_dev) sso_restore_links(const struct rte_eventdev *event_dev)
{ {
@ -803,6 +805,7 @@ sso_configure_dual_ports(const struct rte_eventdev *event_dev)
} }
for (i = 0; i < dev->nb_event_ports; i++) { for (i = 0; i < dev->nb_event_ports; i++) {
struct otx2_ssogws_cookie *gws_cookie;
struct otx2_ssogws_dual *ws; struct otx2_ssogws_dual *ws;
uintptr_t base; uintptr_t base;
@ -811,14 +814,20 @@ sso_configure_dual_ports(const struct rte_eventdev *event_dev)
} else { } else {
/* Allocate event port memory */ /* Allocate event port memory */
ws = rte_zmalloc_socket("otx2_sso_ws", ws = rte_zmalloc_socket("otx2_sso_ws",
sizeof(struct otx2_ssogws_dual), sizeof(struct otx2_ssogws_dual) +
RTE_CACHE_LINE_SIZE,
RTE_CACHE_LINE_SIZE, RTE_CACHE_LINE_SIZE,
event_dev->data->socket_id); event_dev->data->socket_id);
} if (ws == NULL) {
if (ws == NULL) { otx2_err("Failed to alloc memory for port=%d",
otx2_err("Failed to alloc memory for port=%d", i); i);
rc = -ENOMEM; rc = -ENOMEM;
break; break;
}
/* First cache line is reserved for cookie */
ws = (struct otx2_ssogws_dual *)
((uint8_t *)ws + RTE_CACHE_LINE_SIZE);
} }
ws->port = i; ws->port = i;
@ -830,6 +839,10 @@ sso_configure_dual_ports(const struct rte_eventdev *event_dev)
sso_set_port_ops((struct otx2_ssogws *)&ws->ws_state[1], base); sso_set_port_ops((struct otx2_ssogws *)&ws->ws_state[1], base);
vws++; vws++;
gws_cookie = ssogws_get_cookie(ws);
gws_cookie->event_dev = event_dev;
gws_cookie->configured = 1;
event_dev->data->ports[i] = ws; event_dev->data->ports[i] = ws;
} }
@ -866,19 +879,21 @@ sso_configure_ports(const struct rte_eventdev *event_dev)
} }
for (i = 0; i < nb_lf; i++) { for (i = 0; i < nb_lf; i++) {
struct otx2_ssogws_cookie *gws_cookie;
struct otx2_ssogws *ws; struct otx2_ssogws *ws;
uintptr_t base; uintptr_t base;
/* Free memory prior to re-allocation if needed */ /* Free memory prior to re-allocation if needed */
if (event_dev->data->ports[i] != NULL) { if (event_dev->data->ports[i] != NULL) {
ws = event_dev->data->ports[i]; ws = event_dev->data->ports[i];
rte_free(ws); rte_free(ssogws_get_cookie(ws));
ws = NULL; ws = NULL;
} }
/* Allocate event port memory */ /* Allocate event port memory */
ws = rte_zmalloc_socket("otx2_sso_ws", ws = rte_zmalloc_socket("otx2_sso_ws",
sizeof(struct otx2_ssogws), sizeof(struct otx2_ssogws) +
RTE_CACHE_LINE_SIZE,
RTE_CACHE_LINE_SIZE, RTE_CACHE_LINE_SIZE,
event_dev->data->socket_id); event_dev->data->socket_id);
if (ws == NULL) { if (ws == NULL) {
@ -887,10 +902,18 @@ sso_configure_ports(const struct rte_eventdev *event_dev)
break; break;
} }
/* First cache line is reserved for cookie */
ws = (struct otx2_ssogws *)
((uint8_t *)ws + RTE_CACHE_LINE_SIZE);
ws->port = i; ws->port = i;
base = dev->bar2 + (RVU_BLOCK_ADDR_SSOW << 20 | i << 12); base = dev->bar2 + (RVU_BLOCK_ADDR_SSOW << 20 | i << 12);
sso_set_port_ops(ws, base); sso_set_port_ops(ws, base);
gws_cookie = ssogws_get_cookie(ws);
gws_cookie->event_dev = event_dev;
gws_cookie->configured = 1;
event_dev->data->ports[i] = ws; event_dev->data->ports[i] = ws;
} }
@ -1099,11 +1122,8 @@ otx2_sso_configure(const struct rte_eventdev *event_dev)
return -EINVAL; return -EINVAL;
} }
if (dev->configured) { if (dev->configured)
sso_unregister_irqs(event_dev); sso_unregister_irqs(event_dev);
/* Clear any prior port-queue mapping. */
sso_clr_links(event_dev);
}
if (dev->nb_event_queues) { if (dev->nb_event_queues) {
/* Finit any previous queues. */ /* Finit any previous queues. */

View File

@ -217,6 +217,18 @@ sso_pmd_priv(const struct rte_eventdev *event_dev)
return event_dev->data->dev_private; return event_dev->data->dev_private;
} }
struct otx2_ssogws_cookie {
const struct rte_eventdev *event_dev;
bool configured;
};
static inline struct otx2_ssogws_cookie *
ssogws_get_cookie(void *ws)
{
return (struct otx2_ssogws_cookie *)
((uint8_t *)ws - RTE_CACHE_LINE_SIZE);
}
static const union mbuf_initializer mbuf_init = { static const union mbuf_initializer mbuf_init = {
.fields = { .fields = {
.data_off = RTE_PKTMBUF_HEADROOM, .data_off = RTE_PKTMBUF_HEADROOM,

View File

@ -453,9 +453,10 @@ sso_add_tx_queue_data(const struct rte_eventdev *event_dev,
struct otx2_ssogws_dual *dws; struct otx2_ssogws_dual *dws;
old_dws = event_dev->data->ports[i]; old_dws = event_dev->data->ports[i];
dws = rte_realloc_socket(old_dws, dws = rte_realloc_socket(ssogws_get_cookie(old_dws),
sizeof(struct otx2_ssogws_dual) sizeof(struct otx2_ssogws_dual)
+ (sizeof(uint64_t) * + RTE_CACHE_LINE_SIZE +
(sizeof(uint64_t) *
(dev->max_port_id + 1) * (dev->max_port_id + 1) *
RTE_MAX_QUEUES_PER_PORT), RTE_MAX_QUEUES_PER_PORT),
RTE_CACHE_LINE_SIZE, RTE_CACHE_LINE_SIZE,
@ -463,6 +464,10 @@ sso_add_tx_queue_data(const struct rte_eventdev *event_dev,
if (dws == NULL) if (dws == NULL)
return -ENOMEM; return -ENOMEM;
/* First cache line is reserved for cookie */
dws = (struct otx2_ssogws_dual *)
((uint8_t *)dws + RTE_CACHE_LINE_SIZE);
((uint64_t (*)[RTE_MAX_QUEUES_PER_PORT] ((uint64_t (*)[RTE_MAX_QUEUES_PER_PORT]
)&dws->tx_adptr_data)[eth_port_id][tx_queue_id] = )&dws->tx_adptr_data)[eth_port_id][tx_queue_id] =
(uint64_t)txq; (uint64_t)txq;
@ -472,16 +477,21 @@ sso_add_tx_queue_data(const struct rte_eventdev *event_dev,
struct otx2_ssogws *ws; struct otx2_ssogws *ws;
old_ws = event_dev->data->ports[i]; old_ws = event_dev->data->ports[i];
ws = rte_realloc_socket(old_ws, ws = rte_realloc_socket(ssogws_get_cookie(old_ws),
sizeof(struct otx2_ssogws_dual) sizeof(struct otx2_ssogws) +
+ (sizeof(uint64_t) * RTE_CACHE_LINE_SIZE +
(dev->max_port_id + 1) * (sizeof(uint64_t) *
RTE_MAX_QUEUES_PER_PORT), (dev->max_port_id + 1) *
RTE_MAX_QUEUES_PER_PORT),
RTE_CACHE_LINE_SIZE, RTE_CACHE_LINE_SIZE,
event_dev->data->socket_id); event_dev->data->socket_id);
if (ws == NULL) if (ws == NULL)
return -ENOMEM; return -ENOMEM;
/* First cache line is reserved for cookie */
ws = (struct otx2_ssogws *)
((uint8_t *)ws + RTE_CACHE_LINE_SIZE);
((uint64_t (*)[RTE_MAX_QUEUES_PER_PORT] ((uint64_t (*)[RTE_MAX_QUEUES_PER_PORT]
)&ws->tx_adptr_data)[eth_port_id][tx_queue_id] = )&ws->tx_adptr_data)[eth_port_id][tx_queue_id] =
(uint64_t)txq; (uint64_t)txq;