From: Stephen Hemminger <stephen@networkplumber.org>
To: dev@dpdk.org
Cc: Stephen Hemminger <stephen@networkplumber.org>,
Reshma Pattan <reshma.pattan@intel.com>,
Aman Singh <aman.deep.singh@intel.com>,
Naga Harish K S V <s.v.naga.harish.k@intel.com>,
Bruce Richardson <bruce.richardson@intel.com>,
Kishore Padmanabha <kishore.padmanabha@broadcom.com>,
Ajit Khaparde <ajit.khaparde@broadcom.com>,
Nithin Dabilpuram <ndabilpuram@marvell.com>,
Kiran Kumar K <kirankumark@marvell.com>,
Sunil Kumar Kori <skori@marvell.com>,
Satha Rao <skoteshwar@marvell.com>,
Harman Kalra <hkalra@marvell.com>,
Anatoly Burakov <anatoly.burakov@intel.com>,
Cristian Dumitrescu <cristian.dumitrescu@intel.com>,
Thomas Monjalon <thomas@monjalon.net>,
Andrew Rybchenko <andrew.rybchenko@oktetlabs.ru>
Subject: [RFC] ethdev: add buffer size parameter to rte_eth_dev_get_name_by_port()
Date: Wed, 8 Apr 2026 18:36:57 -0700 [thread overview]
Message-ID: <20260409013658.965613-1-stephen@networkplumber.org> (raw)
rte_eth_dev_get_name_by_port() uses strcpy() into a caller-provided
buffer with no bounds checking. A mistaken caller that supplies a
buffer smaller than RTE_ETH_NAME_MAX_LEN can overflow the buffer.
Add a size parameter and replace strcpy() with strlcpy(), returning
-ERANGE if the name is truncated. The copy is now performed under
the ethdev shared data lock so that the name cannot be mutated by
another process between reading and copying.
The previous ABI is preserved via symbol versioning (DPDK_26) with a
thin wrapper that delegates to the new implementation with
RTE_ETH_NAME_MAX_LEN. The versioned symbol will be removed in 26.11.
Update all in-tree callers to pass sizeof(name) as the new parameter.
Signed-off-by: Stephen Hemminger <stephen@networkplumber.org>
---
app/dumpcap/main.c | 8 ++--
app/pdump/main.c | 8 ++--
app/test-pmd/config.c | 6 +--
app/test/test_event_eth_tx_adapter.c | 4 +-
app/test/test_pmd_ring_perf.c | 4 +-
app/test/test_pmd_tap.c | 6 +--
doc/guides/rel_notes/release_26_07.rst | 4 ++
drivers/net/bnxt/tf_ulp/bnxt_ulp_tf.c | 4 +-
drivers/net/cnxk/cnxk_flow.c | 10 +++--
examples/multi_process/hotplug_mp/commands.c | 2 +-
examples/pipeline/cli.c | 2 +-
lib/ethdev/meson.build | 2 +
lib/ethdev/rte_ethdev.c | 42 +++++++++++++++-----
lib/ethdev/rte_ethdev.h | 10 +++--
lib/pdump/rte_pdump.c | 2 +-
15 files changed, 75 insertions(+), 39 deletions(-)
diff --git a/app/dumpcap/main.c b/app/dumpcap/main.c
index 46a6cb251e..47483535fc 100644
--- a/app/dumpcap/main.c
+++ b/app/dumpcap/main.c
@@ -244,7 +244,7 @@ static void find_interfaces(void)
/* maybe got passed port number string as name */
intf->port = get_uint(intf->name, "port_number", UINT16_MAX);
- if (rte_eth_dev_get_name_by_port(intf->port, intf->name) < 0)
+ if (rte_eth_dev_get_name_by_port(intf->port, intf->name, sizeof(intf->name)) < 0)
rte_exit(EXIT_FAILURE, "Invalid port number %u\n",
intf->port);
}
@@ -261,7 +261,7 @@ static void set_default_interface(void)
uint16_t p;
RTE_ETH_FOREACH_DEV(p) {
- if (rte_eth_dev_get_name_by_port(p, name) < 0)
+ if (rte_eth_dev_get_name_by_port(p, name, sizeof(name)) < 0)
continue;
intf = add_interface(name);
@@ -278,7 +278,7 @@ static void dump_interfaces(void)
uint16_t p;
RTE_ETH_FOREACH_DEV(p) {
- if (rte_eth_dev_get_name_by_port(p, name) < 0)
+ if (rte_eth_dev_get_name_by_port(p, name, sizeof(name)) < 0)
continue;
printf("%u. %s\n", p, name);
}
@@ -498,7 +498,7 @@ static void statistics_loop(void)
while (!rte_atomic_load_explicit(&quit_signal, rte_memory_order_relaxed)) {
RTE_ETH_FOREACH_DEV(p) {
- if (rte_eth_dev_get_name_by_port(p, name) < 0)
+ if (rte_eth_dev_get_name_by_port(p, name, sizeof(name)) < 0)
continue;
r = rte_eth_stats_get(p, &stats);
diff --git a/app/pdump/main.c b/app/pdump/main.c
index 1e62c8adc1..1f6135b20e 100644
--- a/app/pdump/main.c
+++ b/app/pdump/main.c
@@ -539,16 +539,16 @@ cleanup_pdump_resources(void)
/* Remove the vdev(s) created */
if (pt->dir & RTE_PDUMP_FLAG_RX) {
- rte_eth_dev_get_name_by_port(pt->rx_vdev_id, name);
- rte_eal_hotplug_remove("vdev", name);
+ if (rte_eth_dev_get_name_by_port(pt->rx_vdev_id, name, sizeof(name)) == 0)
+ rte_eal_hotplug_remove("vdev", name);
}
if (pt->single_pdump_dev)
continue;
if (pt->dir & RTE_PDUMP_FLAG_TX) {
- rte_eth_dev_get_name_by_port(pt->tx_vdev_id, name);
- rte_eal_hotplug_remove("vdev", name);
+ if (rte_eth_dev_get_name_by_port(pt->tx_vdev_id, name, sizeof(name)) == 0)
+ rte_eal_hotplug_remove("vdev", name);
}
}
diff --git a/app/test-pmd/config.c b/app/test-pmd/config.c
index 32c885de0b..7b3743e00f 100644
--- a/app/test-pmd/config.c
+++ b/app/test-pmd/config.c
@@ -697,7 +697,7 @@ device_infos_display(const char *identifier)
&mac_addr) == 0)
print_ethaddr("\n\tMAC address: ",
&mac_addr);
- rte_eth_dev_get_name_by_port(port_id, name);
+ rte_eth_dev_get_name_by_port(port_id, name, sizeof(name));
printf("\n\tDevice name: %s", name);
if (rte_eth_dev_info_get(port_id, &dev_info) == 0)
device_infos_display_speeds(dev_info.speed_capa);
@@ -823,7 +823,7 @@ port_infos_display(portid_t port_id)
info_border, port_id, info_border);
if (eth_macaddr_get_print_err(port_id, &mac_addr) == 0)
print_ethaddr("MAC address: ", &mac_addr);
- rte_eth_dev_get_name_by_port(port_id, name);
+ rte_eth_dev_get_name_by_port(port_id, name, sizeof(name));
printf("\nDevice name: %s", name);
printf("\nDriver name: %s", dev_info.driver_name);
@@ -1011,7 +1011,7 @@ port_summary_display(portid_t port_id)
if (ret != 0)
return;
- rte_eth_dev_get_name_by_port(port_id, name);
+ rte_eth_dev_get_name_by_port(port_id, name, sizeof(name));
ret = eth_macaddr_get_print_err(port_id, &mac_addr);
if (ret != 0)
return;
diff --git a/app/test/test_event_eth_tx_adapter.c b/app/test/test_event_eth_tx_adapter.c
index bec298a8b8..a7cbe39367 100644
--- a/app/test/test_event_eth_tx_adapter.c
+++ b/app/test/test_event_eth_tx_adapter.c
@@ -192,8 +192,8 @@ deinit_ports(void)
for (i = 0; i < RTE_DIM(default_params.port); i++) {
rte_eth_dev_stop(default_params.port[i]);
- rte_eth_dev_get_name_by_port(default_params.port[i], name);
- rte_vdev_uninit(name);
+ if (rte_eth_dev_get_name_by_port(default_params.port[i], name, sizeof(name)) == 0)
+ rte_vdev_uninit(name);
for (j = 0; j < RTE_DIM(default_params.r[i]); j++)
rte_ring_free(default_params.r[i][j]);
}
diff --git a/app/test/test_pmd_ring_perf.c b/app/test/test_pmd_ring_perf.c
index 3636df5c73..ac60f506ab 100644
--- a/app/test/test_pmd_ring_perf.c
+++ b/app/test/test_pmd_ring_perf.c
@@ -157,8 +157,8 @@ test_ring_pmd_perf(void)
/* release port and ring resources */
if (rte_eth_dev_stop(ring_ethdev_port) != 0)
return -1;
- rte_eth_dev_get_name_by_port(ring_ethdev_port, name);
- rte_vdev_uninit(name);
+ if (rte_eth_dev_get_name_by_port(ring_ethdev_port, name, sizeof(name)) == 0)
+ rte_vdev_uninit(name);
rte_ring_free(r);
return 0;
}
diff --git a/app/test/test_pmd_tap.c b/app/test/test_pmd_tap.c
index 6d0e8b4f54..ded26bc5e3 100644
--- a/app/test/test_pmd_tap.c
+++ b/app/test/test_pmd_tap.c
@@ -594,7 +594,7 @@ test_tap_multi_queue(void)
uint16_t port;
RTE_ETH_FOREACH_DEV(port) {
char name[RTE_ETH_NAME_MAX_LEN];
- if (rte_eth_dev_get_name_by_port(port, name) == 0 &&
+ if (rte_eth_dev_get_name_by_port(port, name, sizeof(name)) == 0 &&
strstr(name, "net_tap_mq"))
goto found;
}
@@ -739,7 +739,7 @@ test_tap_setup(void)
/* Find the port IDs */
RTE_ETH_FOREACH_DEV(port_id) {
char name[RTE_ETH_NAME_MAX_LEN];
- if (rte_eth_dev_get_name_by_port(port_id, name) != 0)
+ if (rte_eth_dev_get_name_by_port(port_id, name, sizeof(name)) != 0)
continue;
if (strstr(name, "net_tap0"))
@@ -785,7 +785,7 @@ test_tap_rx_queue_setup(void)
port = -1;
RTE_ETH_FOREACH_DEV(port_id) {
char name[RTE_ETH_NAME_MAX_LEN];
- if (rte_eth_dev_get_name_by_port(port_id, name) == 0 &&
+ if (rte_eth_dev_get_name_by_port(port_id, name, sizeof(name)) == 0 &&
strstr(name, "net_tap_neg")) {
port = port_id;
break;
diff --git a/doc/guides/rel_notes/release_26_07.rst b/doc/guides/rel_notes/release_26_07.rst
index 060b26ff61..f0bf4a4c1b 100644
--- a/doc/guides/rel_notes/release_26_07.rst
+++ b/doc/guides/rel_notes/release_26_07.rst
@@ -84,6 +84,10 @@ API Changes
Also, make sure to start the actual text at the margin.
=======================================================
+* ethdev: Added ``size`` parameter to ``rte_eth_dev_get_name_by_port()``
+ to prevent buffer overflows. The old ABI version without the size parameter
+ is preserved using symbol versioning and will be removed in 26.11.
+
ABI Changes
-----------
diff --git a/drivers/net/bnxt/tf_ulp/bnxt_ulp_tf.c b/drivers/net/bnxt/tf_ulp/bnxt_ulp_tf.c
index bc347de202..bff5ea8c3f 100644
--- a/drivers/net/bnxt/tf_ulp/bnxt_ulp_tf.c
+++ b/drivers/net/bnxt/tf_ulp/bnxt_ulp_tf.c
@@ -608,7 +608,7 @@ ulp_tf_ctx_shared_session_open(struct bnxt *bp,
memset(&parms, 0, sizeof(parms));
rc = rte_eth_dev_get_name_by_port(ethdev->data->port_id,
- parms.ctrl_chan_name);
+ parms.ctrl_chan_name, sizeof(parms.ctrl_chan_name));
if (rc) {
BNXT_DRV_DBG(ERR, "Invalid port %d, rc = %d\n",
ethdev->data->port_id, rc);
@@ -786,7 +786,7 @@ ulp_tf_ctx_session_open(struct bnxt *bp,
memset(¶ms, 0, sizeof(params));
rc = rte_eth_dev_get_name_by_port(ethdev->data->port_id,
- params.ctrl_chan_name);
+ params.ctrl_chan_name, sizeof(params.ctrl_chan_name));
if (rc) {
BNXT_DRV_DBG(ERR, "Invalid port %d, rc = %d\n",
ethdev->data->port_id, rc);
diff --git a/drivers/net/cnxk/cnxk_flow.c b/drivers/net/cnxk/cnxk_flow.c
index c1c48eb7ab..eb51ce37e8 100644
--- a/drivers/net/cnxk/cnxk_flow.c
+++ b/drivers/net/cnxk/cnxk_flow.c
@@ -117,7 +117,7 @@ npc_parse_port_id_action(struct rte_eth_dev *eth_dev, const struct rte_flow_acti
port_act = (const struct rte_flow_action_port_id *)action->conf;
- rc = rte_eth_dev_get_name_by_port(port_act->id, if_name);
+ rc = rte_eth_dev_get_name_by_port(port_act->id, if_name, sizeof(if_name));
if (rc) {
plt_err("Name not found for output port id");
goto err_exit;
@@ -517,7 +517,8 @@ cnxk_map_actions(struct rte_eth_dev *eth_dev, const struct rte_flow_attr *attr,
case RTE_FLOW_ACTION_TYPE_PORT_REPRESENTOR:
in_actions[i].conf = actions->conf;
act_ethdev = (const struct rte_flow_action_ethdev *)actions->conf;
- if (rte_eth_dev_get_name_by_port(act_ethdev->port_id, if_name)) {
+ if (rte_eth_dev_get_name_by_port(act_ethdev->port_id, if_name,
+ sizeof(if_name))) {
plt_err("Name not found for output port id");
goto err_exit;
}
@@ -557,7 +558,7 @@ cnxk_map_actions(struct rte_eth_dev *eth_dev, const struct rte_flow_attr *attr,
actions->type != RTE_FLOW_ACTION_TYPE_PORT_ID ?
act_ethdev->port_id :
port_act->id,
- if_name)) {
+ if_name, sizeof(if_name))) {
plt_err("Name not found for output port id");
goto err_exit;
}
@@ -713,7 +714,8 @@ cnxk_map_pattern(struct rte_eth_dev *eth_dev, const struct rte_flow_item pattern
if (pattern->type == RTE_FLOW_ITEM_TYPE_REPRESENTED_PORT ||
pattern->type == RTE_FLOW_ITEM_TYPE_PORT_REPRESENTOR) {
rep_eth_dev = (const struct rte_flow_item_ethdev *)pattern->spec;
- if (rte_eth_dev_get_name_by_port(rep_eth_dev->port_id, if_name)) {
+ if (rte_eth_dev_get_name_by_port(rep_eth_dev->port_id, if_name,
+ sizeof(if_name))) {
plt_err("Name not found for output port id");
goto fail;
}
diff --git a/examples/multi_process/hotplug_mp/commands.c b/examples/multi_process/hotplug_mp/commands.c
index 900eb9f774..92a4f0378b 100644
--- a/examples/multi_process/hotplug_mp/commands.c
+++ b/examples/multi_process/hotplug_mp/commands.c
@@ -36,7 +36,7 @@ cmd_list_parsed(__rte_unused void *parsed_result,
cmdline_printf(cl, "list all etherdev\n");
RTE_ETH_FOREACH_DEV(port_id) {
- rte_eth_dev_get_name_by_port(port_id, dev_name);
+ rte_eth_dev_get_name_by_port(port_id, dev_name, sizeof(dev_name));
if (strlen(dev_name) > 0)
cmdline_printf(cl, "%d\t%s\n", port_id, dev_name);
else
diff --git a/examples/pipeline/cli.c b/examples/pipeline/cli.c
index 215b4061d5..51c8864077 100644
--- a/examples/pipeline/cli.c
+++ b/examples/pipeline/cli.c
@@ -396,7 +396,7 @@ ethdev_show(uint16_t port_id, char **out, size_t *out_size)
if (rte_eth_link_get(port_id, &link) != 0)
return;
- rte_eth_dev_get_name_by_port(port_id, name);
+ rte_eth_dev_get_name_by_port(port_id, name, sizeof(name));
rte_eth_stats_get(port_id, &stats);
rte_eth_macaddr_get(port_id, &addr);
rte_eth_dev_get_mtu(port_id, &mtu);
diff --git a/lib/ethdev/meson.build b/lib/ethdev/meson.build
index 8ba6c708a2..b2c4222fe6 100644
--- a/lib/ethdev/meson.build
+++ b/lib/ethdev/meson.build
@@ -62,3 +62,5 @@ endif
if get_option('buildtype').contains('debug')
cflags += ['-DRTE_FLOW_DEBUG']
endif
+
+use_function_versioning = true
diff --git a/lib/ethdev/rte_ethdev.c b/lib/ethdev/rte_ethdev.c
index 2edc7a362e..586a56bd46 100644
--- a/lib/ethdev/rte_ethdev.c
+++ b/lib/ethdev/rte_ethdev.c
@@ -721,12 +721,9 @@ rte_eth_dev_count_total(void)
return count;
}
-RTE_EXPORT_SYMBOL(rte_eth_dev_get_name_by_port)
-int
-rte_eth_dev_get_name_by_port(uint16_t port_id, char *name)
+static int
+eth_dev_get_name_by_port(uint16_t port_id, char *name, size_t size)
{
- char *tmp;
-
RTE_ETH_VALID_PORTID_OR_ERR_RET(port_id, -ENODEV);
if (name == NULL) {
@@ -735,19 +732,46 @@ rte_eth_dev_get_name_by_port(uint16_t port_id, char *name)
return -EINVAL;
}
+ if (size == 0) {
+ RTE_ETHDEV_LOG_LINE(ERR,
+ "Cannot get ethdev port %u name with zero-size buffer", port_id);
+ return -EINVAL;
+ }
+
rte_spinlock_lock(rte_mcfg_ethdev_get_lock());
- /* shouldn't check 'rte_eth_devices[i].data',
- * because it might be overwritten by VDEV PMD */
- tmp = eth_dev_shared_data->data[port_id].name;
+ /*
+ * Use the shared data name rather than rte_eth_devices[].data->name
+ * because VDEV PMDs may overwrite the per-process data pointer.
+ */
+ size_t n = strlcpy(name, eth_dev_shared_data->data[port_id].name, size);
rte_spinlock_unlock(rte_mcfg_ethdev_get_lock());
- strcpy(name, tmp);
+ if (n >= size) {
+ RTE_ETHDEV_LOG_LINE(ERR,
+ "ethdev port %u name exceeds buffer size %zu",
+ port_id, size);
+ return -ERANGE;
+ }
rte_ethdev_trace_get_name_by_port(port_id, name);
return 0;
}
+/* new ABI version: bounded copy using caller-provided size */
+RTE_DEFAULT_SYMBOL(27, int, rte_eth_dev_get_name_by_port,
+ (uint16_t port_id, char *name, size_t size))
+{
+ return eth_dev_get_name_by_port(port_id, name, size);
+}
+
+/* old ABI version: delegate to new version with max buffer size */
+RTE_VERSION_SYMBOL(26, int, rte_eth_dev_get_name_by_port,
+ (uint16_t port_id, char *name))
+{
+ return eth_dev_get_name_by_port(port_id, name, RTE_ETH_NAME_MAX_LEN);
+}
+
RTE_EXPORT_SYMBOL(rte_eth_dev_get_port_by_name)
int
rte_eth_dev_get_port_by_name(const char *name, uint16_t *port_id)
diff --git a/lib/ethdev/rte_ethdev.h b/lib/ethdev/rte_ethdev.h
index 0d8e2d0236..cdf26b97b8 100644
--- a/lib/ethdev/rte_ethdev.h
+++ b/lib/ethdev/rte_ethdev.h
@@ -5667,14 +5667,18 @@ rte_eth_dev_get_port_by_name(const char *name, uint16_t *port_id);
* @param port_id
* Port identifier of the device.
* @param name
- * Buffer of size RTE_ETH_NAME_MAX_LEN to store the name.
+ * Buffer to store the name.
+ * @param size
+ * Size of the buffer pointed to by @p name. Should be at least
+ * RTE_ETH_NAME_MAX_LEN bytes.
* @return
* - (0) if successful.
* - (-ENODEV) if *port_id* is invalid.
- * - (-EINVAL) on failure.
+ * - (-EINVAL) if *name* is NULL or *size* is 0.
+ * - (-ERANGE) if the buffer is too small for the device name.
*/
int
-rte_eth_dev_get_name_by_port(uint16_t port_id, char *name);
+rte_eth_dev_get_name_by_port(uint16_t port_id, char *name, size_t size);
/**
* Check that numbers of Rx and Tx descriptors satisfy descriptors limits from
diff --git a/lib/pdump/rte_pdump.c b/lib/pdump/rte_pdump.c
index ac94efe7ff..57c81d4322 100644
--- a/lib/pdump/rte_pdump.c
+++ b/lib/pdump/rte_pdump.c
@@ -736,7 +736,7 @@ pdump_validate_port(uint16_t port, char *name)
return -1;
}
- ret = rte_eth_dev_get_name_by_port(port, name);
+ ret = rte_eth_dev_get_name_by_port(port, name, sizeof(name));
if (ret < 0) {
PDUMP_LOG_LINE(ERR, "port %u to name mapping failed",
port);
--
2.53.0
next reply other threads:[~2026-04-09 1:37 UTC|newest]
Thread overview: 5+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-04-09 1:36 Stephen Hemminger [this message]
2026-04-09 6:59 ` [RFC] ethdev: add buffer size parameter to rte_eth_dev_get_name_by_port() Andrew Rybchenko
2026-04-09 7:58 ` Bruce Richardson
2026-04-09 9:05 ` Andrew Rybchenko
2026-04-09 15:23 ` Stephen Hemminger
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20260409013658.965613-1-stephen@networkplumber.org \
--to=stephen@networkplumber.org \
--cc=ajit.khaparde@broadcom.com \
--cc=aman.deep.singh@intel.com \
--cc=anatoly.burakov@intel.com \
--cc=andrew.rybchenko@oktetlabs.ru \
--cc=bruce.richardson@intel.com \
--cc=cristian.dumitrescu@intel.com \
--cc=dev@dpdk.org \
--cc=hkalra@marvell.com \
--cc=kirankumark@marvell.com \
--cc=kishore.padmanabha@broadcom.com \
--cc=ndabilpuram@marvell.com \
--cc=reshma.pattan@intel.com \
--cc=s.v.naga.harish.k@intel.com \
--cc=skori@marvell.com \
--cc=skoteshwar@marvell.com \
--cc=thomas@monjalon.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox