mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/herbert/cryptodev-2.6.git
synced 2026-04-28 21:46:02 -04:00
selftests: drv-net: ncdevmem: save IDs of flow rules we added
In prep for more selective resetting of ntuple filters try to save the rule IDs to a table. Acked-by: Stanislav Fomichev <sdf@fomichev.me> Link: https://patch.msgid.link/20250825180447.2252977-3-kuba@kernel.org Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
@@ -39,6 +39,7 @@
|
||||
#define __EXPORTED_HEADERS__
|
||||
|
||||
#include <linux/uio.h>
|
||||
#include <stdarg.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
@@ -97,6 +98,10 @@ static unsigned int dmabuf_id;
|
||||
static uint32_t tx_dmabuf_id;
|
||||
static int waittime_ms = 500;
|
||||
|
||||
/* System state loaded by current_config_load() */
|
||||
#define MAX_FLOWS 8
|
||||
static int ntuple_ids[MAX_FLOWS] = { -1, -1, -1, -1, -1, -1, -1, -1, };
|
||||
|
||||
struct memory_buffer {
|
||||
int fd;
|
||||
size_t size;
|
||||
@@ -281,6 +286,85 @@ int validate_buffer(void *line, size_t size)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
__run_command(char *out, size_t outlen, const char *cmd, va_list args)
|
||||
{
|
||||
char command[256];
|
||||
FILE *fp;
|
||||
|
||||
vsnprintf(command, sizeof(command), cmd, args);
|
||||
|
||||
fprintf(stderr, "Running: %s\n", command);
|
||||
fp = popen(command, "r");
|
||||
if (!fp)
|
||||
return -1;
|
||||
if (out) {
|
||||
size_t len;
|
||||
|
||||
if (!fgets(out, outlen, fp))
|
||||
return -1;
|
||||
|
||||
/* Remove trailing newline if present */
|
||||
len = strlen(out);
|
||||
if (len && out[len - 1] == '\n')
|
||||
out[len - 1] = '\0';
|
||||
}
|
||||
return pclose(fp);
|
||||
}
|
||||
|
||||
static int run_command(const char *cmd, ...)
|
||||
{
|
||||
va_list args;
|
||||
int ret;
|
||||
|
||||
va_start(args, cmd);
|
||||
ret = __run_command(NULL, 0, cmd, args);
|
||||
va_end(args);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int ethtool_add_flow(const char *format, ...)
|
||||
{
|
||||
char local_output[256], cmd[256];
|
||||
const char *id_start;
|
||||
int flow_idx, ret;
|
||||
char *endptr;
|
||||
long flow_id;
|
||||
va_list args;
|
||||
|
||||
for (flow_idx = 0; flow_idx < MAX_FLOWS; flow_idx++)
|
||||
if (ntuple_ids[flow_idx] == -1)
|
||||
break;
|
||||
if (flow_idx == MAX_FLOWS) {
|
||||
fprintf(stderr, "Error: too many flows\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
snprintf(cmd, sizeof(cmd), "ethtool -N %s %s", ifname, format);
|
||||
|
||||
va_start(args, format);
|
||||
ret = __run_command(local_output, sizeof(local_output), cmd, args);
|
||||
va_end(args);
|
||||
|
||||
if (ret != 0)
|
||||
return ret;
|
||||
|
||||
/* Extract the ID from the output */
|
||||
id_start = strstr(local_output, "Added rule with ID ");
|
||||
if (!id_start)
|
||||
return -1;
|
||||
id_start += strlen("Added rule with ID ");
|
||||
|
||||
flow_id = strtol(id_start, &endptr, 10);
|
||||
if (endptr == id_start || flow_id < 0 || flow_id > INT_MAX)
|
||||
return -1;
|
||||
|
||||
fprintf(stderr, "Added flow rule with ID %ld\n", flow_id);
|
||||
ntuple_ids[flow_idx] = flow_id;
|
||||
return flow_id;
|
||||
}
|
||||
|
||||
static int rxq_num(int ifindex)
|
||||
{
|
||||
struct ethtool_channels_get_req *req;
|
||||
@@ -308,28 +392,17 @@ static int rxq_num(int ifindex)
|
||||
return num;
|
||||
}
|
||||
|
||||
#define run_command(cmd, ...) \
|
||||
({ \
|
||||
char command[256]; \
|
||||
memset(command, 0, sizeof(command)); \
|
||||
snprintf(command, sizeof(command), cmd, ##__VA_ARGS__); \
|
||||
fprintf(stderr, "Running: %s\n", command); \
|
||||
system(command); \
|
||||
})
|
||||
|
||||
static void reset_flow_steering(void)
|
||||
{
|
||||
/* Depending on the NIC, toggling ntuple off and on might not
|
||||
* be allowed. Additionally, attempting to delete existing filters
|
||||
* will fail if no filters are present. Therefore, do not enforce
|
||||
* the exit status.
|
||||
*/
|
||||
int i;
|
||||
|
||||
run_command("ethtool -K %s ntuple off >&2", ifname);
|
||||
run_command("ethtool -K %s ntuple on >&2", ifname);
|
||||
run_command(
|
||||
"ethtool -n %s | grep 'Filter:' | awk '{print $2}' | xargs -n1 ethtool -N %s delete >&2",
|
||||
ifname, ifname);
|
||||
for (i = 0; i < MAX_FLOWS; i++) {
|
||||
if (ntuple_ids[i] == -1)
|
||||
continue;
|
||||
run_command("ethtool -N %s delete %d",
|
||||
ifname, ntuple_ids[i]);
|
||||
ntuple_ids[i] = -1;
|
||||
}
|
||||
}
|
||||
|
||||
static const char *tcp_data_split_str(int val)
|
||||
@@ -480,6 +553,7 @@ static int configure_flow_steering(struct sockaddr_in6 *server_sin)
|
||||
const char *type = "tcp6";
|
||||
const char *server_addr;
|
||||
char buf[40];
|
||||
int flow_id;
|
||||
|
||||
inet_ntop(AF_INET6, &server_sin->sin6_addr, buf, sizeof(buf));
|
||||
server_addr = buf;
|
||||
@@ -490,23 +564,22 @@ static int configure_flow_steering(struct sockaddr_in6 *server_sin)
|
||||
}
|
||||
|
||||
/* Try configure 5-tuple */
|
||||
if (run_command("ethtool -N %s flow-type %s %s %s dst-ip %s %s %s dst-port %s queue %d >&2",
|
||||
ifname,
|
||||
type,
|
||||
client_ip ? "src-ip" : "",
|
||||
client_ip ?: "",
|
||||
server_addr,
|
||||
client_ip ? "src-port" : "",
|
||||
client_ip ? port : "",
|
||||
port, start_queue))
|
||||
flow_id = ethtool_add_flow("flow-type %s %s %s dst-ip %s %s %s dst-port %s queue %d",
|
||||
type,
|
||||
client_ip ? "src-ip" : "",
|
||||
client_ip ?: "",
|
||||
server_addr,
|
||||
client_ip ? "src-port" : "",
|
||||
client_ip ? port : "",
|
||||
port, start_queue);
|
||||
if (flow_id < 0) {
|
||||
/* If that fails, try configure 3-tuple */
|
||||
if (run_command("ethtool -N %s flow-type %s dst-ip %s dst-port %s queue %d >&2",
|
||||
ifname,
|
||||
type,
|
||||
server_addr,
|
||||
port, start_queue))
|
||||
flow_id = ethtool_add_flow("flow-type %s dst-ip %s dst-port %s queue %d",
|
||||
type, server_addr, port, start_queue);
|
||||
if (flow_id < 0)
|
||||
/* If that fails, return error */
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -682,8 +755,6 @@ static int do_server(struct memory_buffer *mem)
|
||||
return -1;
|
||||
}
|
||||
|
||||
reset_flow_steering();
|
||||
|
||||
if (configure_headersplit(1)) {
|
||||
pr_err("Failed to enable TCP header split");
|
||||
return -1;
|
||||
|
||||
Reference in New Issue
Block a user