Files
openwrt-R7800-nss/package/qca-nss/nss-userspace-oss/libnl-nss/src/nss_nlipv4_api.c
Lucas Asvio e38c766dc7 package: add qca-nss support for k6.x
Before I can get to work a feed repository that work i have included
qca-nss source needed on this commit

Source author: SqTER-PL <r.napierala@asta-net.pl>
2025-08-04 18:52:30 +02:00

177 lines
4.4 KiB
C

/*
**************************************************************************
* Copyright (c) 2019-2021, The Linux Foundation. All rights reserved.
* Permission to use, copy, modify, and/or distribute this software for
* any purpose with or without fee is hereby granted, provided that the
* above copyright notice and this permission notice appear in all copies.
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
* OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
**************************************************************************
*/
#include <nss_nlbase.h>
#include <nss_nlsock_api.h>
#include <nss_nlipv4_api.h>
/*
* nss_nlipv4_sock_cb()
* NSS NL IPv4 callback
*/
int nss_nlipv4_sock_cb(struct nl_msg *msg, void *arg)
{
pid_t pid = getpid();
struct nss_nlipv4_ctx *ctx = (struct nss_nlipv4_ctx *)arg;
struct nss_nlsock_ctx *sock = &ctx->sock;
struct nss_nlipv4_rule *rule = nss_nlsock_get_data(msg);
if (!rule) {
nss_nlsock_log_error("%d:failed to get NSS NL IPv4 header\n", pid);
return NL_SKIP;
}
uint8_t cmd = nss_nlcmn_get_cmd(&rule->cm);
switch (cmd) {
case NSS_IPV4_TX_CREATE_RULE_MSG:
case NSS_IPV4_TX_DESTROY_RULE_MSG:
{
void *cb_data = nss_nlcmn_get_cb_data(&rule->cm, sock->family_id);
if (!cb_data) {
return NL_SKIP;
}
/*
* Note: The callback user can modify the CB content so it
* needs to locally save the response data for further use
* after the callback is completed
*/
struct nss_nlipv4_resp resp;
memcpy(&resp, cb_data, sizeof(struct nss_nlipv4_resp));
/*
* clear the ownership of the CB so that callback user can
* use it if needed
*/
nss_nlcmn_clr_cb_owner(&rule->cm);
if (!resp.cb) {
nss_nlsock_log_info("%d:no IPv4 response callback for cmd(%d)\n", pid, cmd);
return NL_SKIP;
}
resp.cb(sock->user_ctx, rule, resp.data);
return NL_OK;
}
case NSS_IPV4_RX_CONN_STATS_SYNC_MSG:
{
nss_nlipv4_event_t event = ctx->event;
assert(event);
event(sock->user_ctx, rule);
return NL_OK;
}
default:
nss_nlsock_log_error("%d:unsupported message cmd type(%d)\n", pid, cmd);
return NL_SKIP;
}
}
/*
* nss_nlipv4_sock_open()
* this opens the NSS IPv4 NL socket for usage
*/
int nss_nlipv4_sock_open(struct nss_nlipv4_ctx *ctx, void *user_ctx, nss_nlipv4_event_t event_cb)
{
pid_t pid = getpid();
int error;
if (!ctx) {
nss_nlsock_log_error("%d: invalid parameters passed\n", pid);
return -EINVAL;
}
memset(ctx, 0, sizeof(*ctx));
nss_nlsock_set_family(&ctx->sock, NSS_NLIPV4_FAMILY);
nss_nlsock_set_user_ctx(&ctx->sock, user_ctx);
/*
* try opening the socket with Linux
*/
error = nss_nlsock_open(&ctx->sock, nss_nlipv4_sock_cb);
if (error) {
nss_nlsock_log_error("%d:unable to open NSS IPv4 socket, error(%d)\n", pid, error);
goto fail;
}
return 0;
fail:
memset(ctx, 0, sizeof(*ctx));
return error;
}
/*
* nss_nlipv4_sock_close()
* close the NSS IPv4 NL socket
*/
void nss_nlipv4_sock_close(struct nss_nlipv4_ctx *ctx)
{
nss_nlsock_close(&ctx->sock);
}
/*
* nss_nlipv4_sock_send()
* register callback and send the IPv4 message synchronously through the socket
*/
int nss_nlipv4_sock_send(struct nss_nlipv4_ctx *ctx, struct nss_nlipv4_rule *rule, nss_nlipv4_resp_t cb, void *data)
{
int32_t family_id = ctx->sock.family_id;
struct nss_nlipv4_resp *resp;
pid_t pid = getpid();
bool has_resp = false;
int error;
if (!rule) {
nss_nlsock_log_error("%d:invalid NSS IPv4 rule\n", pid);
return -ENOMEM;
}
if (cb) {
nss_nlcmn_set_cb_owner(&rule->cm, family_id);
resp = nss_nlcmn_get_cb_data(&rule->cm, family_id);
assert(resp);
resp->data = data;
resp->cb = cb;
has_resp = true;
}
error = nss_nlsock_send(&ctx->sock, &rule->cm, rule, has_resp);
if (error) {
nss_nlsock_log_error("%d:failed to send NSS IPv4 rule, error(%d)\n", pid, error);
return error;
}
return 0;
}
/*
* nss_nlipv4_init_rule()
* init the rule message
*/
void nss_nlipv4_init_rule(struct nss_nlipv4_rule *rule, enum nss_ipv4_message_types type)
{
nss_nlipv4_rule_init(rule, type);
}