2022-12-13 11:10:43 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* apps/system/iptables/iptables.c
|
|
|
|
*
|
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright ownership. The
|
|
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
|
|
|
#include <nuttx/net/netfilter/ip_tables.h>
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
#include "iptables.h"
|
2022-12-13 11:10:43 +01:00
|
|
|
#include "netutils/netlib.h"
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Types
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
typedef CODE int
|
|
|
|
(*iptables_command_func_t)(FAR const struct iptables_args_s *args);
|
2022-12-13 11:10:43 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_addr2str
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Format address and mask to ip/preflen string.
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
static FAR char *iptables_addr2str(struct in_addr addr, struct in_addr msk,
|
|
|
|
FAR char *buf, size_t bufflen)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
unsigned int preflen = popcount(msk.s_addr);
|
|
|
|
if (preflen != 0)
|
|
|
|
{
|
|
|
|
snprintf(buf, bufflen, "%s/%d", inet_ntoa(addr), preflen);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
snprintf(buf, bufflen, "anywhere");
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_print_chain
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Print all rules in a chain
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
static void iptables_print_chain(FAR const struct ipt_replace *repl,
|
|
|
|
enum nf_inet_hooks hook)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
/* Format: target !prot !idev !odev !saddr !daddr match */
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
const char fmt[] = "%-12s %1s%-4s %1s%-4s %1s%-4s %1s%-18s %1s%-18s %s\n";
|
|
|
|
char src[INET_ADDRSTRLEN + 3]; /* Format: 123.123.123.123/24 */
|
|
|
|
char dst[INET_ADDRSTRLEN + 3];
|
|
|
|
|
|
|
|
FAR struct ipt_entry *entry;
|
|
|
|
FAR struct xt_entry_match *match;
|
|
|
|
FAR struct xt_entry_target *target;
|
|
|
|
FAR uint8_t *head = (FAR uint8_t *)repl->entries + repl->hook_entry[hook];
|
|
|
|
int size = repl->underflow[hook] - repl->hook_entry[hook];
|
|
|
|
|
|
|
|
/* The underflow entry contains the default rule. */
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
entry = (FAR struct ipt_entry *)(head + size);
|
|
|
|
target = IPT_TARGET(entry);
|
|
|
|
|
|
|
|
printf("Chain %s (policy %s)\n",
|
|
|
|
iptables_hook2str(hook), iptables_target2str(target));
|
|
|
|
printf(fmt, "target", "", "prot", "", "idev", "", "odev",
|
|
|
|
"", "source", "", "destination", "");
|
|
|
|
|
|
|
|
ipt_entry_for_every(entry, head, size)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
target = IPT_TARGET(entry);
|
|
|
|
match = entry->target_offset >= sizeof(struct xt_entry_match) ?
|
|
|
|
IPT_MATCH(entry) : NULL;
|
|
|
|
printf(fmt, iptables_target2str(target),
|
|
|
|
INV_FLAG_STR(entry->ip.invflags & IPT_INV_PROTO),
|
|
|
|
iptables_proto2str(entry->ip.proto),
|
|
|
|
INV_FLAG_STR(entry->ip.invflags & IPT_INV_VIA_IN),
|
|
|
|
iptables_iface2str(entry->ip.iniface),
|
|
|
|
INV_FLAG_STR(entry->ip.invflags & IPT_INV_VIA_OUT),
|
|
|
|
iptables_iface2str(entry->ip.outiface),
|
|
|
|
INV_FLAG_STR(entry->ip.invflags & IPT_INV_SRCIP),
|
|
|
|
iptables_addr2str(entry->ip.src, entry->ip.smsk, src, sizeof(src)),
|
|
|
|
INV_FLAG_STR(entry->ip.invflags & IPT_INV_DSTIP),
|
|
|
|
iptables_addr2str(entry->ip.dst, entry->ip.dmsk, dst, sizeof(dst)),
|
|
|
|
iptables_match2str(match));
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("\n");
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_list
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* List all rules in a table
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
static int iptables_list(FAR const char *table, enum nf_inet_hooks hook)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
FAR struct ipt_replace *repl = netlib_ipt_prepare(table);
|
|
|
|
unsigned int cur_hook;
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (repl == NULL)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Failed to read table %s from kernel!\n", table);
|
|
|
|
return -EIO;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
for (cur_hook = 0; cur_hook < NF_INET_NUMHOOKS; cur_hook++)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
if ((repl->valid_hooks & (1 << cur_hook)) != 0 &&
|
|
|
|
(hook == NF_INET_NUMHOOKS || hook == cur_hook))
|
|
|
|
{
|
|
|
|
iptables_print_chain(repl, cur_hook);
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
free(repl);
|
|
|
|
return OK;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_finish_command
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Do a command and commit it
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
static int iptables_finish_command(FAR const struct iptables_args_s *args,
|
|
|
|
FAR struct ipt_replace **repl,
|
|
|
|
FAR struct ipt_entry *entry)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (args->cmd)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
case COMMAND_APPEND:
|
|
|
|
ret = netlib_ipt_append(repl, entry, args->hook);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case COMMAND_INSERT:
|
|
|
|
ret = netlib_ipt_insert(repl, entry, args->hook, args->rulenum);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case COMMAND_DELETE:
|
|
|
|
ret = netlib_ipt_delete(*repl, entry, args->hook, args->rulenum);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: /* Other commands should not call into this function. */
|
|
|
|
ret = -EINVAL;
|
|
|
|
break;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
2024-04-03 10:54:39 +02:00
|
|
|
|
|
|
|
if (ret == OK)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
ret = netlib_ipt_commit(*repl);
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
return ret;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_finish_directly
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Finish command directly without preparing any entry
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
static int iptables_finish_directly(FAR const struct iptables_args_s *args)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
FAR struct ipt_replace *repl = netlib_ipt_prepare(args->table);
|
|
|
|
int ret;
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (repl == NULL)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Failed to read table '%s' from kernel!\n", args->table);
|
|
|
|
return -EIO;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
ret = iptables_finish_command(args, &repl, NULL);
|
|
|
|
|
|
|
|
free(repl);
|
|
|
|
return ret;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_nat_command
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Do a NAT command
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
#ifdef CONFIG_NET_NAT
|
|
|
|
static int iptables_nat_command(FAR const struct iptables_args_s *args)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
FAR struct ipt_replace *repl;
|
|
|
|
FAR struct ipt_entry *entry = NULL;
|
|
|
|
int ret;
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (args->outifname == NULL || args->outifname[0] == '\0')
|
|
|
|
{
|
|
|
|
printf("Table '" TABLE_NAME_NAT "' needs an out interface!\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args->target != NULL &&
|
|
|
|
strcmp(args->target, XT_MASQUERADE_TARGET))
|
|
|
|
{
|
|
|
|
printf("Only target '" XT_MASQUERADE_TARGET
|
|
|
|
"' is supported for table '" TABLE_NAME_NAT "'!\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
repl = netlib_ipt_prepare(TABLE_NAME_NAT);
|
2022-12-13 11:10:43 +01:00
|
|
|
if (repl == NULL)
|
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Failed to read table '" TABLE_NAME_NAT "' from kernel!\n");
|
2022-12-13 11:10:43 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
entry = netlib_ipt_masquerade_entry(args->outifname);
|
|
|
|
if (entry == NULL)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Failed to prepare entry for dev %s!\n", args->outifname);
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto errout_with_repl;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
ret = iptables_finish_command(args, &repl, entry);
|
|
|
|
|
|
|
|
free(entry);
|
|
|
|
errout_with_repl:
|
2022-12-13 11:10:43 +01:00
|
|
|
free(repl);
|
2024-04-03 10:54:39 +02:00
|
|
|
return ret;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
2024-04-03 10:54:39 +02:00
|
|
|
#endif
|
2022-12-13 11:10:43 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_filter_command
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Do a FILTER command
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
#ifdef CONFIG_NET_IPFILTER
|
|
|
|
static int iptables_filter_command(FAR const struct iptables_args_s *args)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
FAR struct xt_entry_match *match;
|
|
|
|
FAR struct ipt_replace *repl = netlib_ipt_prepare(XT_TABLE_NAME_FILTER);
|
2022-12-13 11:10:43 +01:00
|
|
|
FAR struct ipt_entry *entry = NULL;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (repl == NULL)
|
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Failed to read table '" XT_TABLE_NAME_FILTER
|
|
|
|
"' from kernel!\n");
|
2022-12-13 11:10:43 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
/* Get entry and fill in proto-specific details. */
|
|
|
|
|
|
|
|
if (args->sport != NULL || args->dport != NULL)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
FAR struct xt_udp *tcpudp;
|
|
|
|
|
|
|
|
if (args->protocol != IPPROTO_TCP && args->protocol != IPPROTO_UDP)
|
|
|
|
{
|
|
|
|
printf("Source/destination port is only supported for TCP/UDP!\n");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
entry = netlib_ipt_filter_entry(args->target, args->verdict,
|
|
|
|
args->protocol);
|
2022-12-13 11:10:43 +01:00
|
|
|
if (entry == NULL)
|
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
goto errout_prepare_entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
match = IPT_MATCH(entry);
|
|
|
|
tcpudp = (FAR struct xt_udp *)(match + 1);
|
|
|
|
|
|
|
|
switch (args->protocol)
|
|
|
|
{
|
|
|
|
case IPPROTO_TCP:
|
|
|
|
((FAR struct xt_tcp *)tcpudp)->invflags = args->tcpudpinv;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IPPROTO_UDP:
|
|
|
|
((FAR struct xt_udp *)tcpudp)->invflags = args->tcpudpinv;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = iptables_parse_ports(args->sport, tcpudp->spts);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to parse source port!\n");
|
|
|
|
goto errout;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
ret = iptables_parse_ports(args->dport, tcpudp->dpts);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to parse destination port!\n");
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (args->icmp_type != NULL)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
FAR struct ipt_icmp *icmp;
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (args->protocol != IPPROTO_ICMP)
|
|
|
|
{
|
|
|
|
printf("ICMP type is only supported for ICMP protocol!\n");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
entry = netlib_ipt_filter_entry(args->target, args->verdict,
|
|
|
|
args->protocol);
|
|
|
|
if (entry == NULL)
|
|
|
|
{
|
|
|
|
goto errout_prepare_entry;
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
match = IPT_MATCH(entry);
|
|
|
|
icmp = (FAR struct ipt_icmp *)(match + 1);
|
|
|
|
|
|
|
|
ret = iptables_parse_icmp(args->icmp_type);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to parse ICMP type!\n");
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
icmp->type = ret;
|
|
|
|
icmp->invflags = args->icmpinv;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
entry = netlib_ipt_filter_entry(args->target, args->verdict, 0);
|
|
|
|
if (entry == NULL)
|
|
|
|
{
|
|
|
|
goto errout_prepare_entry;
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
/* Fill in common details. */
|
|
|
|
|
|
|
|
ret = netlib_ipt_fillifname(entry, args->inifname, args->outifname);
|
|
|
|
if (ret < 0)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Failed to fill in interface names!\n");
|
|
|
|
goto errout;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (args->saddr != NULL)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
ret = iptables_parse_ip(args->saddr, &entry->ip.src, &entry->ip.smsk,
|
|
|
|
AF_INET);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to parse source address %s!\n", args->saddr);
|
|
|
|
goto errout;
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (args->daddr != NULL)
|
|
|
|
{
|
|
|
|
ret = iptables_parse_ip(args->daddr, &entry->ip.dst, &entry->ip.dmsk,
|
|
|
|
AF_INET);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to parse destination address %s!\n", args->daddr);
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
entry->ip.proto = args->protocol;
|
|
|
|
entry->ip.invflags = args->ipinv;
|
|
|
|
|
|
|
|
/* Finish command. */
|
|
|
|
|
|
|
|
ret = iptables_finish_command(args, &repl, entry);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
free(entry);
|
2022-12-13 11:10:43 +01:00
|
|
|
free(repl);
|
|
|
|
return ret;
|
2024-04-03 10:54:39 +02:00
|
|
|
|
|
|
|
errout_prepare_entry:
|
|
|
|
printf("Failed to prepare entry!\n");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto errout;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
2024-04-03 10:54:39 +02:00
|
|
|
#endif
|
2022-12-13 11:10:43 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
2024-04-03 10:54:39 +02:00
|
|
|
* Name: iptables_apply
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2024-04-03 10:54:39 +02:00
|
|
|
* Apply rules for corresponding table
|
2022-12-13 11:10:43 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
static int iptables_apply(FAR const struct iptables_args_s *args,
|
|
|
|
iptables_command_func_t command_func)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
switch (args->cmd)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
|
|
|
case COMMAND_FLUSH:
|
2024-04-03 10:54:39 +02:00
|
|
|
return netlib_ipt_flush(args->table, args->hook);
|
2022-12-13 11:10:43 +01:00
|
|
|
|
|
|
|
case COMMAND_LIST:
|
2024-04-03 10:54:39 +02:00
|
|
|
return iptables_list(args->table, args->hook);
|
|
|
|
|
|
|
|
case COMMAND_POLICY:
|
|
|
|
return netlib_ipt_policy(args->table, args->hook, args->verdict);
|
2022-12-13 11:10:43 +01:00
|
|
|
|
|
|
|
case COMMAND_DELETE:
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
/* Delete rule with rulenum can be done directly. */
|
|
|
|
|
|
|
|
if (args->rulenum > 0)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
return iptables_finish_directly(args);
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
/* Fall through. */
|
|
|
|
|
|
|
|
case COMMAND_APPEND:
|
|
|
|
case COMMAND_INSERT:
|
|
|
|
return command_func(args);
|
2022-12-13 11:10:43 +01:00
|
|
|
|
|
|
|
default:
|
|
|
|
printf("No supported command specified!\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int main(int argc, FAR char *argv[])
|
|
|
|
{
|
|
|
|
struct iptables_args_s args;
|
2024-04-03 10:54:39 +02:00
|
|
|
int ret = iptables_parse(&args, argc, argv);
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
if (ret < 0 || args.cmd == COMMAND_INVALID)
|
|
|
|
{
|
|
|
|
iptables_showusage(argv[0]);
|
|
|
|
return ret;
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
|
2024-04-03 10:54:39 +02:00
|
|
|
#ifdef CONFIG_NET_IPFILTER
|
|
|
|
if (args.table == NULL || strcmp(args.table, XT_TABLE_NAME_FILTER) == 0)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
args.table = XT_TABLE_NAME_FILTER;
|
|
|
|
ret = iptables_apply(&args, iptables_filter_command);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("iptables got error on filter: %d!\n", ret);
|
|
|
|
}
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
2024-04-03 10:54:39 +02:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_NET_NAT
|
|
|
|
if (strcmp(args.table, TABLE_NAME_NAT) == 0)
|
2022-12-13 11:10:43 +01:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
ret = iptables_apply(&args, iptables_nat_command);
|
2022-12-13 11:10:43 +01:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("iptables got error on NAT: %d!\n", ret);
|
|
|
|
}
|
2023-04-14 10:52:24 +02:00
|
|
|
}
|
|
|
|
else
|
2024-04-03 10:54:39 +02:00
|
|
|
#endif
|
2023-04-14 10:52:24 +02:00
|
|
|
{
|
2024-04-03 10:54:39 +02:00
|
|
|
printf("Unknown table: %s\n", args.table);
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|
|
|
|
|
2023-04-14 10:52:24 +02:00
|
|
|
return ret;
|
2022-12-13 11:10:43 +01:00
|
|
|
}
|