1*cf84ac9aSAndroid Build Coastguard Worker /*
2*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2017 JingPiao Chen <[email protected]>
3*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2017-2018 The strace developers.
4*cf84ac9aSAndroid Build Coastguard Worker * All rights reserved.
5*cf84ac9aSAndroid Build Coastguard Worker *
6*cf84ac9aSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
7*cf84ac9aSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
8*cf84ac9aSAndroid Build Coastguard Worker * are met:
9*cf84ac9aSAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
10*cf84ac9aSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
11*cf84ac9aSAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
12*cf84ac9aSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
13*cf84ac9aSAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
14*cf84ac9aSAndroid Build Coastguard Worker * 3. The name of the author may not be used to endorse or promote products
15*cf84ac9aSAndroid Build Coastguard Worker * derived from this software without specific prior written permission.
16*cf84ac9aSAndroid Build Coastguard Worker *
17*cf84ac9aSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18*cf84ac9aSAndroid Build Coastguard Worker * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19*cf84ac9aSAndroid Build Coastguard Worker * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20*cf84ac9aSAndroid Build Coastguard Worker * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21*cf84ac9aSAndroid Build Coastguard Worker * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22*cf84ac9aSAndroid Build Coastguard Worker * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23*cf84ac9aSAndroid Build Coastguard Worker * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24*cf84ac9aSAndroid Build Coastguard Worker * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25*cf84ac9aSAndroid Build Coastguard Worker * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26*cf84ac9aSAndroid Build Coastguard Worker * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27*cf84ac9aSAndroid Build Coastguard Worker */
28*cf84ac9aSAndroid Build Coastguard Worker
29*cf84ac9aSAndroid Build Coastguard Worker #include "tests.h"
30*cf84ac9aSAndroid Build Coastguard Worker
31*cf84ac9aSAndroid Build Coastguard Worker #ifdef HAVE_LINUX_FIB_RULES_H
32*cf84ac9aSAndroid Build Coastguard Worker
33*cf84ac9aSAndroid Build Coastguard Worker # include <stdio.h>
34*cf84ac9aSAndroid Build Coastguard Worker # include <inttypes.h>
35*cf84ac9aSAndroid Build Coastguard Worker # include "test_nlattr.h"
36*cf84ac9aSAndroid Build Coastguard Worker # include <linux/fib_rules.h>
37*cf84ac9aSAndroid Build Coastguard Worker # include <linux/in.h>
38*cf84ac9aSAndroid Build Coastguard Worker # include <linux/ip.h>
39*cf84ac9aSAndroid Build Coastguard Worker # include <linux/rtnetlink.h>
40*cf84ac9aSAndroid Build Coastguard Worker
41*cf84ac9aSAndroid Build Coastguard Worker #define FRA_TUN_ID 12
42*cf84ac9aSAndroid Build Coastguard Worker #define FRA_TABLE 15
43*cf84ac9aSAndroid Build Coastguard Worker #define FRA_UID_RANGE 20
44*cf84ac9aSAndroid Build Coastguard Worker #define FRA_PROTOCOL 21
45*cf84ac9aSAndroid Build Coastguard Worker #define FRA_IP_PROTO 22
46*cf84ac9aSAndroid Build Coastguard Worker #define FRA_SPORT_RANGE 23
47*cf84ac9aSAndroid Build Coastguard Worker #define FRA_DPORT_RANGE 24
48*cf84ac9aSAndroid Build Coastguard Worker
49*cf84ac9aSAndroid Build Coastguard Worker # ifndef HAVE_STRUCT_FIB_RULE_PORT_RANGE
50*cf84ac9aSAndroid Build Coastguard Worker struct fib_rule_port_range {
51*cf84ac9aSAndroid Build Coastguard Worker uint16_t start;
52*cf84ac9aSAndroid Build Coastguard Worker uint16_t end;
53*cf84ac9aSAndroid Build Coastguard Worker };
54*cf84ac9aSAndroid Build Coastguard Worker # endif /* HAVE_STRUCT_FIB_RULE_PORT_RANGE */
55*cf84ac9aSAndroid Build Coastguard Worker
56*cf84ac9aSAndroid Build Coastguard Worker static void
init_rtmsg(struct nlmsghdr * const nlh,const unsigned int msg_len)57*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg(struct nlmsghdr *const nlh, const unsigned int msg_len)
58*cf84ac9aSAndroid Build Coastguard Worker {
59*cf84ac9aSAndroid Build Coastguard Worker SET_STRUCT(struct nlmsghdr, nlh,
60*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_len = msg_len,
61*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_type = RTM_GETRULE,
62*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_flags = NLM_F_DUMP
63*cf84ac9aSAndroid Build Coastguard Worker );
64*cf84ac9aSAndroid Build Coastguard Worker
65*cf84ac9aSAndroid Build Coastguard Worker struct rtmsg *const msg = NLMSG_DATA(nlh);
66*cf84ac9aSAndroid Build Coastguard Worker SET_STRUCT(struct rtmsg, msg,
67*cf84ac9aSAndroid Build Coastguard Worker .rtm_family = AF_UNIX,
68*cf84ac9aSAndroid Build Coastguard Worker .rtm_tos = IPTOS_LOWDELAY,
69*cf84ac9aSAndroid Build Coastguard Worker .rtm_table = RT_TABLE_UNSPEC,
70*cf84ac9aSAndroid Build Coastguard Worker .rtm_type = FR_ACT_TO_TBL,
71*cf84ac9aSAndroid Build Coastguard Worker .rtm_flags = FIB_RULE_INVERT
72*cf84ac9aSAndroid Build Coastguard Worker );
73*cf84ac9aSAndroid Build Coastguard Worker }
74*cf84ac9aSAndroid Build Coastguard Worker
75*cf84ac9aSAndroid Build Coastguard Worker static void
print_rtmsg(const unsigned int msg_len)76*cf84ac9aSAndroid Build Coastguard Worker print_rtmsg(const unsigned int msg_len)
77*cf84ac9aSAndroid Build Coastguard Worker {
78*cf84ac9aSAndroid Build Coastguard Worker printf("{len=%u, type=RTM_GETRULE, flags=NLM_F_DUMP"
79*cf84ac9aSAndroid Build Coastguard Worker ", seq=0, pid=0}, {family=AF_UNIX"
80*cf84ac9aSAndroid Build Coastguard Worker ", dst_len=0, src_len=0"
81*cf84ac9aSAndroid Build Coastguard Worker ", tos=IPTOS_LOWDELAY"
82*cf84ac9aSAndroid Build Coastguard Worker ", table=RT_TABLE_UNSPEC"
83*cf84ac9aSAndroid Build Coastguard Worker ", action=FR_ACT_TO_TBL"
84*cf84ac9aSAndroid Build Coastguard Worker ", flags=FIB_RULE_INVERT}",
85*cf84ac9aSAndroid Build Coastguard Worker msg_len);
86*cf84ac9aSAndroid Build Coastguard Worker }
87*cf84ac9aSAndroid Build Coastguard Worker
88*cf84ac9aSAndroid Build Coastguard Worker int
main(void)89*cf84ac9aSAndroid Build Coastguard Worker main(void)
90*cf84ac9aSAndroid Build Coastguard Worker {
91*cf84ac9aSAndroid Build Coastguard Worker skip_if_unavailable("/proc/self/fd/");
92*cf84ac9aSAndroid Build Coastguard Worker
93*cf84ac9aSAndroid Build Coastguard Worker const int fd = create_nl_socket(NETLINK_ROUTE);
94*cf84ac9aSAndroid Build Coastguard Worker const unsigned int hdrlen = sizeof(struct rtmsg);
95*cf84ac9aSAndroid Build Coastguard Worker void *nlh0 = midtail_alloc(NLMSG_SPACE(hdrlen), NLA_HDRLEN + 8);
96*cf84ac9aSAndroid Build Coastguard Worker
97*cf84ac9aSAndroid Build Coastguard Worker static char pattern[4096];
98*cf84ac9aSAndroid Build Coastguard Worker fill_memory_ex(pattern, sizeof(pattern), 'a', 'z' - 'a' + 1);
99*cf84ac9aSAndroid Build Coastguard Worker
100*cf84ac9aSAndroid Build Coastguard Worker const unsigned int nla_type = 0xffff & NLA_TYPE_MASK;
101*cf84ac9aSAndroid Build Coastguard Worker char nla_type_str[256];
102*cf84ac9aSAndroid Build Coastguard Worker sprintf(nla_type_str, "%#x /* FRA_??? */", nla_type);
103*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_(fd, nlh0, hdrlen,
104*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
105*cf84ac9aSAndroid Build Coastguard Worker nla_type, nla_type_str,
106*cf84ac9aSAndroid Build Coastguard Worker 4, pattern, 4,
107*cf84ac9aSAndroid Build Coastguard Worker print_quoted_hex(pattern, 4));
108*cf84ac9aSAndroid Build Coastguard Worker
109*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR(fd, nlh0, hdrlen,
110*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
111*cf84ac9aSAndroid Build Coastguard Worker FRA_DST, 4, pattern, 4,
112*cf84ac9aSAndroid Build Coastguard Worker print_quoted_hex(pattern, 4));
113*cf84ac9aSAndroid Build Coastguard Worker
114*cf84ac9aSAndroid Build Coastguard Worker const uint32_t table_id = RT_TABLE_DEFAULT;
115*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
116*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
117*cf84ac9aSAndroid Build Coastguard Worker FRA_TABLE, pattern, table_id,
118*cf84ac9aSAndroid Build Coastguard Worker printf("RT_TABLE_DEFAULT"));
119*cf84ac9aSAndroid Build Coastguard Worker
120*cf84ac9aSAndroid Build Coastguard Worker #ifdef HAVE_STRUCT_FIB_RULE_UID_RANGE
121*cf84ac9aSAndroid Build Coastguard Worker static const struct fib_rule_uid_range range = {
122*cf84ac9aSAndroid Build Coastguard Worker .start = 0xabcdedad,
123*cf84ac9aSAndroid Build Coastguard Worker .end = 0xbcdeadba
124*cf84ac9aSAndroid Build Coastguard Worker };
125*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
126*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
127*cf84ac9aSAndroid Build Coastguard Worker FRA_UID_RANGE, pattern, range,
128*cf84ac9aSAndroid Build Coastguard Worker PRINT_FIELD_U("{", range, start);
129*cf84ac9aSAndroid Build Coastguard Worker PRINT_FIELD_U(", ", range, end);
130*cf84ac9aSAndroid Build Coastguard Worker printf("}"));
131*cf84ac9aSAndroid Build Coastguard Worker #endif
132*cf84ac9aSAndroid Build Coastguard Worker #if defined HAVE_BE64TOH || defined be64toh
133*cf84ac9aSAndroid Build Coastguard Worker const uint64_t tun_id = 0xabcdcdbeedabadef;
134*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
135*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
136*cf84ac9aSAndroid Build Coastguard Worker FRA_TUN_ID, pattern, tun_id,
137*cf84ac9aSAndroid Build Coastguard Worker printf("htobe64(%" PRIu64 ")", be64toh(tun_id)));
138*cf84ac9aSAndroid Build Coastguard Worker #endif
139*cf84ac9aSAndroid Build Coastguard Worker
140*cf84ac9aSAndroid Build Coastguard Worker uint8_t proto;
141*cf84ac9aSAndroid Build Coastguard Worker
142*cf84ac9aSAndroid Build Coastguard Worker static const struct {
143*cf84ac9aSAndroid Build Coastguard Worker uint8_t arg;
144*cf84ac9aSAndroid Build Coastguard Worker const char *str;
145*cf84ac9aSAndroid Build Coastguard Worker } proto_args[] = {
146*cf84ac9aSAndroid Build Coastguard Worker { ARG_STR(RTPROT_UNSPEC) },
147*cf84ac9aSAndroid Build Coastguard Worker { 5, "0x5 /* RTPROT_??? */" },
148*cf84ac9aSAndroid Build Coastguard Worker { 17, "RTPROT_MROUTED" },
149*cf84ac9aSAndroid Build Coastguard Worker { 42, "RTPROT_BABEL" },
150*cf84ac9aSAndroid Build Coastguard Worker { 43, "0x2b /* RTPROT_??? */" },
151*cf84ac9aSAndroid Build Coastguard Worker { ARG_STR(0xde) " /* RTPROT_??? */" },
152*cf84ac9aSAndroid Build Coastguard Worker };
153*cf84ac9aSAndroid Build Coastguard Worker
154*cf84ac9aSAndroid Build Coastguard Worker for (unsigned i = 0; i < ARRAY_SIZE(proto_args); i++) {
155*cf84ac9aSAndroid Build Coastguard Worker proto = proto_args[i].arg;
156*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
157*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
158*cf84ac9aSAndroid Build Coastguard Worker FRA_PROTOCOL, pattern, proto,
159*cf84ac9aSAndroid Build Coastguard Worker printf("%s", proto_args[i].str));
160*cf84ac9aSAndroid Build Coastguard Worker }
161*cf84ac9aSAndroid Build Coastguard Worker
162*cf84ac9aSAndroid Build Coastguard Worker static const struct {
163*cf84ac9aSAndroid Build Coastguard Worker uint8_t arg;
164*cf84ac9aSAndroid Build Coastguard Worker const char *str;
165*cf84ac9aSAndroid Build Coastguard Worker } ipproto_args[] = {
166*cf84ac9aSAndroid Build Coastguard Worker { ARG_STR(IPPROTO_TCP) },
167*cf84ac9aSAndroid Build Coastguard Worker { 254, "0xfe /* IPPROTO_??? */" },
168*cf84ac9aSAndroid Build Coastguard Worker { ARG_STR(IPPROTO_RAW) },
169*cf84ac9aSAndroid Build Coastguard Worker };
170*cf84ac9aSAndroid Build Coastguard Worker
171*cf84ac9aSAndroid Build Coastguard Worker for (unsigned i = 0; i < ARRAY_SIZE(ipproto_args); i++) {
172*cf84ac9aSAndroid Build Coastguard Worker proto = ipproto_args[i].arg;
173*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
174*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
175*cf84ac9aSAndroid Build Coastguard Worker FRA_IP_PROTO, pattern, proto,
176*cf84ac9aSAndroid Build Coastguard Worker printf("%s", ipproto_args[i].str));
177*cf84ac9aSAndroid Build Coastguard Worker }
178*cf84ac9aSAndroid Build Coastguard Worker
179*cf84ac9aSAndroid Build Coastguard Worker static const struct fib_rule_port_range prange = {
180*cf84ac9aSAndroid Build Coastguard Worker .start = 0xabcd,
181*cf84ac9aSAndroid Build Coastguard Worker .end = 0xfeed,
182*cf84ac9aSAndroid Build Coastguard Worker };
183*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
184*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
185*cf84ac9aSAndroid Build Coastguard Worker FRA_SPORT_RANGE, pattern, prange,
186*cf84ac9aSAndroid Build Coastguard Worker PRINT_FIELD_U("{", prange, start);
187*cf84ac9aSAndroid Build Coastguard Worker PRINT_FIELD_U(", ", prange, end);
188*cf84ac9aSAndroid Build Coastguard Worker printf("}"));
189*cf84ac9aSAndroid Build Coastguard Worker TEST_NLATTR_OBJECT(fd, nlh0, hdrlen,
190*cf84ac9aSAndroid Build Coastguard Worker init_rtmsg, print_rtmsg,
191*cf84ac9aSAndroid Build Coastguard Worker FRA_DPORT_RANGE, pattern, prange,
192*cf84ac9aSAndroid Build Coastguard Worker PRINT_FIELD_U("{", prange, start);
193*cf84ac9aSAndroid Build Coastguard Worker PRINT_FIELD_U(", ", prange, end);
194*cf84ac9aSAndroid Build Coastguard Worker printf("}"));
195*cf84ac9aSAndroid Build Coastguard Worker
196*cf84ac9aSAndroid Build Coastguard Worker puts("+++ exited with 0 +++");
197*cf84ac9aSAndroid Build Coastguard Worker return 0;
198*cf84ac9aSAndroid Build Coastguard Worker }
199*cf84ac9aSAndroid Build Coastguard Worker
200*cf84ac9aSAndroid Build Coastguard Worker #else
201*cf84ac9aSAndroid Build Coastguard Worker
202*cf84ac9aSAndroid Build Coastguard Worker SKIP_MAIN_UNDEFINED("HAVE_LINUX_FIB_RULES_H")
203*cf84ac9aSAndroid Build Coastguard Worker
204*cf84ac9aSAndroid Build Coastguard Worker #endif
205