1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
|
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <linux/ip.h>
#ifndef IPTOS_NORMALSVC
# define IPTOS_NORMALSVC 0
#endif
struct tos_value_mask {
uint8_t value, mask;
};
static const struct tos_symbol_info {
unsigned char value;
const char *name;
} tos_symbol_names[] = {
{IPTOS_LOWDELAY, "Minimize-Delay"},
{IPTOS_THROUGHPUT, "Maximize-Throughput"},
{IPTOS_RELIABILITY, "Maximize-Reliability"},
{IPTOS_MINCOST, "Minimize-Cost"},
{IPTOS_NORMALSVC, "Normal-Service"},
{ .name = NULL }
};
/*
* tos_parse_numeric - parse sth. like "15/255"
*
* @s: input string
* @info: accompanying structure
* @bits: number of bits that are allowed
* (8 for IPv4 TOS field, 4 for IPv6 Priority Field)
*/
static bool tos_parse_numeric(const char *str, struct tos_value_mask *tvm,
unsigned int bits)
{
const unsigned int max = (1 << bits) - 1;
unsigned int value;
char *end;
xtables_strtoui(str, &end, &value, 0, max);
tvm->value = value;
tvm->mask = max;
if (*end == '/') {
const char *p = end + 1;
if (!xtables_strtoui(p, &end, &value, 0, max))
xtables_error(PARAMETER_PROBLEM, "Illegal value: \"%s\"",
str);
tvm->mask = value;
}
if (*end != '\0')
xtables_error(PARAMETER_PROBLEM, "Illegal value: \"%s\"", str);
return true;
}
static bool tos_parse_symbolic(const char *str, struct tos_value_mask *tvm,
unsigned int def_mask)
{
const unsigned int max = UINT8_MAX;
const struct tos_symbol_info *symbol;
char *tmp;
if (xtables_strtoui(str, &tmp, NULL, 0, max))
return tos_parse_numeric(str, tvm, max);
/* Do not consider ECN bits */
tvm->mask = def_mask;
for (symbol = tos_symbol_names; symbol->name != NULL; ++symbol)
if (strcasecmp(str, symbol->name) == 0) {
tvm->value = symbol->value;
return true;
}
xtables_error(PARAMETER_PROBLEM, "Symbolic name \"%s\" is unknown", str);
return false;
}
static bool tos_try_print_symbolic(const char *prefix,
u_int8_t value, u_int8_t mask)
{
const struct tos_symbol_info *symbol;
if (mask != 0x3F)
return false;
for (symbol = tos_symbol_names; symbol->name != NULL; ++symbol)
if (value == symbol->value) {
printf("%s%s ", prefix, symbol->name);
return true;
}
return false;
}
|