mirror of
https://git.proxmox.com/git/mirror_iproute2
synced 2025-08-09 10:00:11 +00:00
Add SKB Priority qdisc support in tc(8)
sch_skbprio is a qdisc that prioritizes packets according to their skb->priority field. Under congestion, it drops already-enqueued lower priority packets to make space available for higher priority packets. Skbprio was conceived as a solution for denial-of-service defenses that need to route packets with different priorities as a means to overcome DoS attacks. Signed-off-by: Nishanth Devarajan <ndev2021@gmail.com> Reviewed-by: Michel Machado <michel@digirati.com.br> Signed-off-by: David Ahern <dsahern@gmail.com>
This commit is contained in:
parent
c044be6b34
commit
141b55f854
70
man/man8/tc-skbprio.8
Normal file
70
man/man8/tc-skbprio.8
Normal file
@ -0,0 +1,70 @@
|
||||
.TH SKBPRIO 8 "13 August 2018" "iproute2" "Linux"
|
||||
.SH NAME
|
||||
skbprio \- SKB Priority Queue
|
||||
|
||||
.SH SYNOPSIS
|
||||
.B tc qdisc ... add skbprio
|
||||
.B [ limit
|
||||
packets
|
||||
.B ]
|
||||
|
||||
.SH DESCRIPTION
|
||||
SKB Priority Queue is a queueing discipline intended to prioritize
|
||||
the most important packets during a denial-of-service (
|
||||
.B DoS
|
||||
) attack. The priority of a packet is given by
|
||||
.B skb->priority
|
||||
, where a higher value places the packet closer to the exit of the queue. When
|
||||
the queue is full, the lowest priority packet in the queue is dropped to make
|
||||
room for the packet to be added if it has higher priority. If the packet to be
|
||||
added has lower priority than all packets in the queue, it is dropped.
|
||||
|
||||
Without SKB priority queue, queue length limits must be imposed
|
||||
on individual sub-queues, and there is no straightforward way to enforce
|
||||
a global queue length limit across all priorities. SKBprio queue enforces
|
||||
a global queue length limit while not restricting the lengths of
|
||||
individual sub-queues.
|
||||
|
||||
While SKB Priority Queue is agnostic to how
|
||||
.B skb->priority
|
||||
is assigned. A typical use case is to copy
|
||||
the 6-bit DS field of IPv4 and IPv6 packets using
|
||||
.BR tc-skbedit (8)
|
||||
. If
|
||||
.B skb->priority
|
||||
is greater or equal to 64, the priority is assumed to be 63.
|
||||
Priorities less than 64 are taken at face value.
|
||||
|
||||
SKB Priority Queue enables routers to locally decide which
|
||||
packets to drop under a DoS attack.
|
||||
Priorities should be assigned to packets such that the higher the priority,
|
||||
the more expected behavior a source shows.
|
||||
So sources have an incentive to play by the rules.
|
||||
|
||||
.SH ALGORITHM
|
||||
|
||||
Skbprio maintains 64 lists (priorities go from 0 to 63).
|
||||
When a packet is enqueued, it gets inserted at the
|
||||
.B tail
|
||||
of its priority list. When a packet needs to be sent out to the network, it is
|
||||
taken from the head of the highest priority list. When the queue is full,
|
||||
the packet at the tail of the lowest priority list is dropped to serve the
|
||||
ingress packet - if it is of higher priority, otherwise the ingress packet is
|
||||
dropped. This algorithm allocates as much bandwidth as possible to high
|
||||
priority packets, while only servicing low priority packets when
|
||||
there is enough bandwidth.
|
||||
|
||||
.SH PARAMETERS
|
||||
.TP
|
||||
limit
|
||||
Maximum queue size specified in packets. It defaults to 64.
|
||||
The range for this parameter is [0, UINT32_MAX].
|
||||
|
||||
.SH SEE ALSO
|
||||
.BR tc-prio (8),
|
||||
.BR tc-skbedit (8)
|
||||
|
||||
.SH AUTHORS
|
||||
Nishanth Devarajan <devarajn@uci.edu>, Michel Machado <michel@digirati.com.br>
|
||||
|
||||
This manpage maintained by Bert Hubert <ahu@ds9a.nl>
|
@ -12,6 +12,7 @@ TCMODULES += q_fifo.o
|
||||
TCMODULES += q_sfq.o
|
||||
TCMODULES += q_red.o
|
||||
TCMODULES += q_prio.o
|
||||
TCMODULES += q_skbprio.o
|
||||
TCMODULES += q_tbf.o
|
||||
TCMODULES += q_cbq.o
|
||||
TCMODULES += q_rr.o
|
||||
|
84
tc/q_skbprio.c
Normal file
84
tc/q_skbprio.c
Normal file
@ -0,0 +1,84 @@
|
||||
/*
|
||||
* q_skbprio.c SKB PRIORITY QUEUE.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation; either version
|
||||
* 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* Authors: Nishanth Devarajan, <ndev2021@gmail.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
#include <syslog.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "utils.h"
|
||||
#include "tc_util.h"
|
||||
|
||||
static void explain(void)
|
||||
{
|
||||
fprintf(stderr, "Usage: ... <skbprio> [ limit NUMBER ]\n");
|
||||
}
|
||||
|
||||
static int skbprio_parse_opt(struct qdisc_util *qu, int argc, char **argv,
|
||||
struct nlmsghdr *n, const char *dev)
|
||||
{
|
||||
int ok = 0;
|
||||
struct tc_skbprio_qopt opt = {};
|
||||
|
||||
while (argc > 0) {
|
||||
if (strcmp(*argv, "limit") == 0) {
|
||||
NEXT_ARG();
|
||||
if (get_size(&opt.limit, *argv)) {
|
||||
fprintf(stderr,
|
||||
"%s: Illegal \"limit\" value:\"%s\"\n",
|
||||
qu->id, *argv);
|
||||
return -1;
|
||||
}
|
||||
ok++;
|
||||
}
|
||||
else if (strcmp(*argv, "help") == 0) {
|
||||
explain();
|
||||
return -1;
|
||||
} else {
|
||||
fprintf(stderr,
|
||||
"%s: unknown parameter \"%s\"\n",
|
||||
qu->id, *argv);
|
||||
explain();
|
||||
return -1;
|
||||
}
|
||||
argc--; argv++;
|
||||
}
|
||||
|
||||
if (ok)
|
||||
addattr_l(n, 1024, TCA_OPTIONS, &opt, sizeof(opt));
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int skbprio_print_opt(struct qdisc_util *qu, FILE *f, struct rtattr *opt)
|
||||
{
|
||||
struct tc_skbprio_qopt *qopt;
|
||||
|
||||
if (opt == NULL)
|
||||
return 0;
|
||||
|
||||
if (RTA_PAYLOAD(opt) < sizeof(*qopt))
|
||||
return -1;
|
||||
qopt = RTA_DATA(opt);
|
||||
fprintf(f, "limit %u ", qopt->limit);
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct qdisc_util skbprio_qdisc_util = {
|
||||
.id = "skbprio",
|
||||
.parse_qopt = skbprio_parse_opt,
|
||||
.print_qopt = skbprio_print_opt,
|
||||
};
|
Loading…
Reference in New Issue
Block a user