1*cf84ac9aSAndroid Build Coastguard Worker /*
2*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2014 Zubin Mithra <[email protected]>
3*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2014-2016 Dmitry V. Levin <[email protected]>
4*cf84ac9aSAndroid Build Coastguard Worker * Copyright (c) 2014-2018 The strace developers.
5*cf84ac9aSAndroid Build Coastguard Worker * All rights reserved.
6*cf84ac9aSAndroid Build Coastguard Worker *
7*cf84ac9aSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
8*cf84ac9aSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
9*cf84ac9aSAndroid Build Coastguard Worker * are met:
10*cf84ac9aSAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
11*cf84ac9aSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
12*cf84ac9aSAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
13*cf84ac9aSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
14*cf84ac9aSAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
15*cf84ac9aSAndroid Build Coastguard Worker * 3. The name of the author may not be used to endorse or promote products
16*cf84ac9aSAndroid Build Coastguard Worker * derived from this software without specific prior written permission.
17*cf84ac9aSAndroid Build Coastguard Worker *
18*cf84ac9aSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19*cf84ac9aSAndroid Build Coastguard Worker * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20*cf84ac9aSAndroid Build Coastguard Worker * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21*cf84ac9aSAndroid Build Coastguard Worker * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22*cf84ac9aSAndroid Build Coastguard Worker * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23*cf84ac9aSAndroid Build Coastguard Worker * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*cf84ac9aSAndroid Build Coastguard Worker * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*cf84ac9aSAndroid Build Coastguard Worker * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*cf84ac9aSAndroid Build Coastguard Worker * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27*cf84ac9aSAndroid Build Coastguard Worker * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*cf84ac9aSAndroid Build Coastguard Worker */
29*cf84ac9aSAndroid Build Coastguard Worker
30*cf84ac9aSAndroid Build Coastguard Worker #include "defs.h"
31*cf84ac9aSAndroid Build Coastguard Worker #include <netinet/in.h>
32*cf84ac9aSAndroid Build Coastguard Worker #include <sys/socket.h>
33*cf84ac9aSAndroid Build Coastguard Worker #include <arpa/inet.h>
34*cf84ac9aSAndroid Build Coastguard Worker #include "netlink.h"
35*cf84ac9aSAndroid Build Coastguard Worker #include <linux/sock_diag.h>
36*cf84ac9aSAndroid Build Coastguard Worker #include <linux/inet_diag.h>
37*cf84ac9aSAndroid Build Coastguard Worker #include <linux/unix_diag.h>
38*cf84ac9aSAndroid Build Coastguard Worker #include <linux/netlink_diag.h>
39*cf84ac9aSAndroid Build Coastguard Worker #include <linux/rtnetlink.h>
40*cf84ac9aSAndroid Build Coastguard Worker #if HAVE_LINUX_GENETLINK_H
41*cf84ac9aSAndroid Build Coastguard Worker #include <linux/genetlink.h>
42*cf84ac9aSAndroid Build Coastguard Worker #endif
43*cf84ac9aSAndroid Build Coastguard Worker
44*cf84ac9aSAndroid Build Coastguard Worker #include <sys/un.h>
45*cf84ac9aSAndroid Build Coastguard Worker #ifndef UNIX_PATH_MAX
46*cf84ac9aSAndroid Build Coastguard Worker # define UNIX_PATH_MAX sizeof(((struct sockaddr_un *) 0)->sun_path)
47*cf84ac9aSAndroid Build Coastguard Worker #endif
48*cf84ac9aSAndroid Build Coastguard Worker
49*cf84ac9aSAndroid Build Coastguard Worker #include "xstring.h"
50*cf84ac9aSAndroid Build Coastguard Worker
51*cf84ac9aSAndroid Build Coastguard Worker #define XLAT_MACROS_ONLY
52*cf84ac9aSAndroid Build Coastguard Worker # include "xlat/inet_protocols.h"
53*cf84ac9aSAndroid Build Coastguard Worker #undef XLAT_MACROS_ONLY
54*cf84ac9aSAndroid Build Coastguard Worker
55*cf84ac9aSAndroid Build Coastguard Worker typedef struct {
56*cf84ac9aSAndroid Build Coastguard Worker unsigned long inode;
57*cf84ac9aSAndroid Build Coastguard Worker char *details;
58*cf84ac9aSAndroid Build Coastguard Worker } cache_entry;
59*cf84ac9aSAndroid Build Coastguard Worker
60*cf84ac9aSAndroid Build Coastguard Worker #define CACHE_SIZE 1024U
61*cf84ac9aSAndroid Build Coastguard Worker static cache_entry cache[CACHE_SIZE];
62*cf84ac9aSAndroid Build Coastguard Worker #define CACHE_MASK (CACHE_SIZE - 1)
63*cf84ac9aSAndroid Build Coastguard Worker
64*cf84ac9aSAndroid Build Coastguard Worker static int
cache_inode_details(const unsigned long inode,char * const details)65*cf84ac9aSAndroid Build Coastguard Worker cache_inode_details(const unsigned long inode, char *const details)
66*cf84ac9aSAndroid Build Coastguard Worker {
67*cf84ac9aSAndroid Build Coastguard Worker cache_entry *e = &cache[inode & CACHE_MASK];
68*cf84ac9aSAndroid Build Coastguard Worker free(e->details);
69*cf84ac9aSAndroid Build Coastguard Worker e->inode = inode;
70*cf84ac9aSAndroid Build Coastguard Worker e->details = details;
71*cf84ac9aSAndroid Build Coastguard Worker
72*cf84ac9aSAndroid Build Coastguard Worker return 1;
73*cf84ac9aSAndroid Build Coastguard Worker }
74*cf84ac9aSAndroid Build Coastguard Worker
75*cf84ac9aSAndroid Build Coastguard Worker static const char *
get_sockaddr_by_inode_cached(const unsigned long inode)76*cf84ac9aSAndroid Build Coastguard Worker get_sockaddr_by_inode_cached(const unsigned long inode)
77*cf84ac9aSAndroid Build Coastguard Worker {
78*cf84ac9aSAndroid Build Coastguard Worker const cache_entry *const e = &cache[inode & CACHE_MASK];
79*cf84ac9aSAndroid Build Coastguard Worker return (e && inode == e->inode) ? e->details : NULL;
80*cf84ac9aSAndroid Build Coastguard Worker }
81*cf84ac9aSAndroid Build Coastguard Worker
82*cf84ac9aSAndroid Build Coastguard Worker static bool
print_sockaddr_by_inode_cached(const unsigned long inode)83*cf84ac9aSAndroid Build Coastguard Worker print_sockaddr_by_inode_cached(const unsigned long inode)
84*cf84ac9aSAndroid Build Coastguard Worker {
85*cf84ac9aSAndroid Build Coastguard Worker const char *const details = get_sockaddr_by_inode_cached(inode);
86*cf84ac9aSAndroid Build Coastguard Worker if (details) {
87*cf84ac9aSAndroid Build Coastguard Worker tprints(details);
88*cf84ac9aSAndroid Build Coastguard Worker return true;
89*cf84ac9aSAndroid Build Coastguard Worker }
90*cf84ac9aSAndroid Build Coastguard Worker return false;
91*cf84ac9aSAndroid Build Coastguard Worker }
92*cf84ac9aSAndroid Build Coastguard Worker
93*cf84ac9aSAndroid Build Coastguard Worker static bool
send_query(struct tcb * tcp,const int fd,void * req,size_t req_size)94*cf84ac9aSAndroid Build Coastguard Worker send_query(struct tcb *tcp, const int fd, void *req, size_t req_size)
95*cf84ac9aSAndroid Build Coastguard Worker {
96*cf84ac9aSAndroid Build Coastguard Worker struct sockaddr_nl nladdr = {
97*cf84ac9aSAndroid Build Coastguard Worker .nl_family = AF_NETLINK
98*cf84ac9aSAndroid Build Coastguard Worker };
99*cf84ac9aSAndroid Build Coastguard Worker struct iovec iov = {
100*cf84ac9aSAndroid Build Coastguard Worker .iov_base = req,
101*cf84ac9aSAndroid Build Coastguard Worker .iov_len = req_size
102*cf84ac9aSAndroid Build Coastguard Worker };
103*cf84ac9aSAndroid Build Coastguard Worker const struct msghdr msg = {
104*cf84ac9aSAndroid Build Coastguard Worker .msg_name = &nladdr,
105*cf84ac9aSAndroid Build Coastguard Worker .msg_namelen = sizeof(nladdr),
106*cf84ac9aSAndroid Build Coastguard Worker .msg_iov = &iov,
107*cf84ac9aSAndroid Build Coastguard Worker .msg_iovlen = 1
108*cf84ac9aSAndroid Build Coastguard Worker };
109*cf84ac9aSAndroid Build Coastguard Worker
110*cf84ac9aSAndroid Build Coastguard Worker for (;;) {
111*cf84ac9aSAndroid Build Coastguard Worker if (sendmsg(fd, &msg, 0) < 0) {
112*cf84ac9aSAndroid Build Coastguard Worker if (errno == EINTR)
113*cf84ac9aSAndroid Build Coastguard Worker continue;
114*cf84ac9aSAndroid Build Coastguard Worker return false;
115*cf84ac9aSAndroid Build Coastguard Worker }
116*cf84ac9aSAndroid Build Coastguard Worker return true;
117*cf84ac9aSAndroid Build Coastguard Worker }
118*cf84ac9aSAndroid Build Coastguard Worker }
119*cf84ac9aSAndroid Build Coastguard Worker
120*cf84ac9aSAndroid Build Coastguard Worker static bool
inet_send_query(struct tcb * tcp,const int fd,const int family,const int proto)121*cf84ac9aSAndroid Build Coastguard Worker inet_send_query(struct tcb *tcp, const int fd, const int family,
122*cf84ac9aSAndroid Build Coastguard Worker const int proto)
123*cf84ac9aSAndroid Build Coastguard Worker {
124*cf84ac9aSAndroid Build Coastguard Worker struct {
125*cf84ac9aSAndroid Build Coastguard Worker const struct nlmsghdr nlh;
126*cf84ac9aSAndroid Build Coastguard Worker const struct inet_diag_req_v2 idr;
127*cf84ac9aSAndroid Build Coastguard Worker } req = {
128*cf84ac9aSAndroid Build Coastguard Worker .nlh = {
129*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_len = sizeof(req),
130*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_type = SOCK_DIAG_BY_FAMILY,
131*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_flags = NLM_F_DUMP | NLM_F_REQUEST
132*cf84ac9aSAndroid Build Coastguard Worker },
133*cf84ac9aSAndroid Build Coastguard Worker .idr = {
134*cf84ac9aSAndroid Build Coastguard Worker .sdiag_family = family,
135*cf84ac9aSAndroid Build Coastguard Worker .sdiag_protocol = proto,
136*cf84ac9aSAndroid Build Coastguard Worker .idiag_states = -1
137*cf84ac9aSAndroid Build Coastguard Worker }
138*cf84ac9aSAndroid Build Coastguard Worker };
139*cf84ac9aSAndroid Build Coastguard Worker return send_query(tcp, fd, &req, sizeof(req));
140*cf84ac9aSAndroid Build Coastguard Worker }
141*cf84ac9aSAndroid Build Coastguard Worker
142*cf84ac9aSAndroid Build Coastguard Worker static int
inet_parse_response(const void * const data,const int data_len,const unsigned long inode,void * opaque_data)143*cf84ac9aSAndroid Build Coastguard Worker inet_parse_response(const void *const data, const int data_len,
144*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode, void *opaque_data)
145*cf84ac9aSAndroid Build Coastguard Worker {
146*cf84ac9aSAndroid Build Coastguard Worker const char *const proto_name = opaque_data;
147*cf84ac9aSAndroid Build Coastguard Worker const struct inet_diag_msg *const diag_msg = data;
148*cf84ac9aSAndroid Build Coastguard Worker static const char zero_addr[sizeof(struct in6_addr)];
149*cf84ac9aSAndroid Build Coastguard Worker socklen_t addr_size, text_size;
150*cf84ac9aSAndroid Build Coastguard Worker
151*cf84ac9aSAndroid Build Coastguard Worker if (data_len < (int) NLMSG_LENGTH(sizeof(*diag_msg)))
152*cf84ac9aSAndroid Build Coastguard Worker return -1;
153*cf84ac9aSAndroid Build Coastguard Worker if (diag_msg->idiag_inode != inode)
154*cf84ac9aSAndroid Build Coastguard Worker return 0;
155*cf84ac9aSAndroid Build Coastguard Worker
156*cf84ac9aSAndroid Build Coastguard Worker switch (diag_msg->idiag_family) {
157*cf84ac9aSAndroid Build Coastguard Worker case AF_INET:
158*cf84ac9aSAndroid Build Coastguard Worker addr_size = sizeof(struct in_addr);
159*cf84ac9aSAndroid Build Coastguard Worker text_size = INET_ADDRSTRLEN;
160*cf84ac9aSAndroid Build Coastguard Worker break;
161*cf84ac9aSAndroid Build Coastguard Worker case AF_INET6:
162*cf84ac9aSAndroid Build Coastguard Worker addr_size = sizeof(struct in6_addr);
163*cf84ac9aSAndroid Build Coastguard Worker text_size = INET6_ADDRSTRLEN;
164*cf84ac9aSAndroid Build Coastguard Worker break;
165*cf84ac9aSAndroid Build Coastguard Worker default:
166*cf84ac9aSAndroid Build Coastguard Worker return -1;
167*cf84ac9aSAndroid Build Coastguard Worker }
168*cf84ac9aSAndroid Build Coastguard Worker
169*cf84ac9aSAndroid Build Coastguard Worker char src_buf[text_size];
170*cf84ac9aSAndroid Build Coastguard Worker char *details;
171*cf84ac9aSAndroid Build Coastguard Worker
172*cf84ac9aSAndroid Build Coastguard Worker /* open/closing brackets for IPv6 addresses */
173*cf84ac9aSAndroid Build Coastguard Worker const char *ob = diag_msg->idiag_family == AF_INET6 ? "[" : "";
174*cf84ac9aSAndroid Build Coastguard Worker const char *cb = diag_msg->idiag_family == AF_INET6 ? "]" : "";
175*cf84ac9aSAndroid Build Coastguard Worker
176*cf84ac9aSAndroid Build Coastguard Worker if (!inet_ntop(diag_msg->idiag_family, diag_msg->id.idiag_src,
177*cf84ac9aSAndroid Build Coastguard Worker src_buf, text_size))
178*cf84ac9aSAndroid Build Coastguard Worker return -1;
179*cf84ac9aSAndroid Build Coastguard Worker
180*cf84ac9aSAndroid Build Coastguard Worker if (diag_msg->id.idiag_dport ||
181*cf84ac9aSAndroid Build Coastguard Worker memcmp(zero_addr, diag_msg->id.idiag_dst, addr_size)) {
182*cf84ac9aSAndroid Build Coastguard Worker char dst_buf[text_size];
183*cf84ac9aSAndroid Build Coastguard Worker
184*cf84ac9aSAndroid Build Coastguard Worker if (!inet_ntop(diag_msg->idiag_family, diag_msg->id.idiag_dst,
185*cf84ac9aSAndroid Build Coastguard Worker dst_buf, text_size))
186*cf84ac9aSAndroid Build Coastguard Worker return -1;
187*cf84ac9aSAndroid Build Coastguard Worker
188*cf84ac9aSAndroid Build Coastguard Worker if (asprintf(&details, "%s:[%s%s%s:%u->%s%s%s:%u]", proto_name,
189*cf84ac9aSAndroid Build Coastguard Worker ob, src_buf, cb, ntohs(diag_msg->id.idiag_sport),
190*cf84ac9aSAndroid Build Coastguard Worker ob, dst_buf, cb, ntohs(diag_msg->id.idiag_dport))
191*cf84ac9aSAndroid Build Coastguard Worker < 0)
192*cf84ac9aSAndroid Build Coastguard Worker return false;
193*cf84ac9aSAndroid Build Coastguard Worker } else {
194*cf84ac9aSAndroid Build Coastguard Worker if (asprintf(&details, "%s:[%s%s%s:%u]",
195*cf84ac9aSAndroid Build Coastguard Worker proto_name, ob, src_buf, cb,
196*cf84ac9aSAndroid Build Coastguard Worker ntohs(diag_msg->id.idiag_sport)) < 0)
197*cf84ac9aSAndroid Build Coastguard Worker return false;
198*cf84ac9aSAndroid Build Coastguard Worker }
199*cf84ac9aSAndroid Build Coastguard Worker
200*cf84ac9aSAndroid Build Coastguard Worker return cache_inode_details(inode, details);
201*cf84ac9aSAndroid Build Coastguard Worker }
202*cf84ac9aSAndroid Build Coastguard Worker
203*cf84ac9aSAndroid Build Coastguard Worker static bool
receive_responses(struct tcb * tcp,const int fd,const unsigned long inode,const unsigned long expected_msg_type,int (* parser)(const void *,int,unsigned long,void *),void * opaque_data)204*cf84ac9aSAndroid Build Coastguard Worker receive_responses(struct tcb *tcp, const int fd, const unsigned long inode,
205*cf84ac9aSAndroid Build Coastguard Worker const unsigned long expected_msg_type,
206*cf84ac9aSAndroid Build Coastguard Worker int (*parser)(const void *, int,
207*cf84ac9aSAndroid Build Coastguard Worker unsigned long, void *),
208*cf84ac9aSAndroid Build Coastguard Worker void *opaque_data)
209*cf84ac9aSAndroid Build Coastguard Worker {
210*cf84ac9aSAndroid Build Coastguard Worker static union {
211*cf84ac9aSAndroid Build Coastguard Worker struct nlmsghdr hdr;
212*cf84ac9aSAndroid Build Coastguard Worker long buf[8192 / sizeof(long)];
213*cf84ac9aSAndroid Build Coastguard Worker } hdr_buf;
214*cf84ac9aSAndroid Build Coastguard Worker
215*cf84ac9aSAndroid Build Coastguard Worker struct sockaddr_nl nladdr = {
216*cf84ac9aSAndroid Build Coastguard Worker .nl_family = AF_NETLINK
217*cf84ac9aSAndroid Build Coastguard Worker };
218*cf84ac9aSAndroid Build Coastguard Worker struct iovec iov = {
219*cf84ac9aSAndroid Build Coastguard Worker .iov_base = hdr_buf.buf,
220*cf84ac9aSAndroid Build Coastguard Worker .iov_len = sizeof(hdr_buf.buf)
221*cf84ac9aSAndroid Build Coastguard Worker };
222*cf84ac9aSAndroid Build Coastguard Worker int flags = 0;
223*cf84ac9aSAndroid Build Coastguard Worker
224*cf84ac9aSAndroid Build Coastguard Worker for (;;) {
225*cf84ac9aSAndroid Build Coastguard Worker struct msghdr msg = {
226*cf84ac9aSAndroid Build Coastguard Worker .msg_name = &nladdr,
227*cf84ac9aSAndroid Build Coastguard Worker .msg_namelen = sizeof(nladdr),
228*cf84ac9aSAndroid Build Coastguard Worker .msg_iov = &iov,
229*cf84ac9aSAndroid Build Coastguard Worker .msg_iovlen = 1
230*cf84ac9aSAndroid Build Coastguard Worker };
231*cf84ac9aSAndroid Build Coastguard Worker
232*cf84ac9aSAndroid Build Coastguard Worker ssize_t ret = recvmsg(fd, &msg, flags);
233*cf84ac9aSAndroid Build Coastguard Worker if (ret < 0) {
234*cf84ac9aSAndroid Build Coastguard Worker if (errno == EINTR)
235*cf84ac9aSAndroid Build Coastguard Worker continue;
236*cf84ac9aSAndroid Build Coastguard Worker return false;
237*cf84ac9aSAndroid Build Coastguard Worker }
238*cf84ac9aSAndroid Build Coastguard Worker
239*cf84ac9aSAndroid Build Coastguard Worker const struct nlmsghdr *h = &hdr_buf.hdr;
240*cf84ac9aSAndroid Build Coastguard Worker if (!is_nlmsg_ok(h, ret))
241*cf84ac9aSAndroid Build Coastguard Worker return false;
242*cf84ac9aSAndroid Build Coastguard Worker for (; is_nlmsg_ok(h, ret); h = NLMSG_NEXT(h, ret)) {
243*cf84ac9aSAndroid Build Coastguard Worker if (h->nlmsg_type != expected_msg_type)
244*cf84ac9aSAndroid Build Coastguard Worker return false;
245*cf84ac9aSAndroid Build Coastguard Worker const int rc = parser(NLMSG_DATA(h),
246*cf84ac9aSAndroid Build Coastguard Worker h->nlmsg_len, inode, opaque_data);
247*cf84ac9aSAndroid Build Coastguard Worker if (rc > 0)
248*cf84ac9aSAndroid Build Coastguard Worker return true;
249*cf84ac9aSAndroid Build Coastguard Worker if (rc < 0)
250*cf84ac9aSAndroid Build Coastguard Worker return false;
251*cf84ac9aSAndroid Build Coastguard Worker }
252*cf84ac9aSAndroid Build Coastguard Worker flags = MSG_DONTWAIT;
253*cf84ac9aSAndroid Build Coastguard Worker }
254*cf84ac9aSAndroid Build Coastguard Worker }
255*cf84ac9aSAndroid Build Coastguard Worker
256*cf84ac9aSAndroid Build Coastguard Worker static bool
unix_send_query(struct tcb * tcp,const int fd,const unsigned long inode)257*cf84ac9aSAndroid Build Coastguard Worker unix_send_query(struct tcb *tcp, const int fd, const unsigned long inode)
258*cf84ac9aSAndroid Build Coastguard Worker {
259*cf84ac9aSAndroid Build Coastguard Worker /*
260*cf84ac9aSAndroid Build Coastguard Worker * The kernel bug was fixed in mainline by commit v4.5-rc6~35^2~11
261*cf84ac9aSAndroid Build Coastguard Worker * and backported to stable/linux-4.4.y by commit v4.4.4~297.
262*cf84ac9aSAndroid Build Coastguard Worker */
263*cf84ac9aSAndroid Build Coastguard Worker const uint16_t dump_flag =
264*cf84ac9aSAndroid Build Coastguard Worker os_release < KERNEL_VERSION(4, 4, 4) ? NLM_F_DUMP : 0;
265*cf84ac9aSAndroid Build Coastguard Worker
266*cf84ac9aSAndroid Build Coastguard Worker struct {
267*cf84ac9aSAndroid Build Coastguard Worker const struct nlmsghdr nlh;
268*cf84ac9aSAndroid Build Coastguard Worker const struct unix_diag_req udr;
269*cf84ac9aSAndroid Build Coastguard Worker } req = {
270*cf84ac9aSAndroid Build Coastguard Worker .nlh = {
271*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_len = sizeof(req),
272*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_type = SOCK_DIAG_BY_FAMILY,
273*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_flags = NLM_F_REQUEST | dump_flag
274*cf84ac9aSAndroid Build Coastguard Worker },
275*cf84ac9aSAndroid Build Coastguard Worker .udr = {
276*cf84ac9aSAndroid Build Coastguard Worker .sdiag_family = AF_UNIX,
277*cf84ac9aSAndroid Build Coastguard Worker .udiag_ino = inode,
278*cf84ac9aSAndroid Build Coastguard Worker .udiag_states = -1,
279*cf84ac9aSAndroid Build Coastguard Worker .udiag_show = UDIAG_SHOW_NAME | UDIAG_SHOW_PEER,
280*cf84ac9aSAndroid Build Coastguard Worker .udiag_cookie = { ~0U, ~0U }
281*cf84ac9aSAndroid Build Coastguard Worker }
282*cf84ac9aSAndroid Build Coastguard Worker };
283*cf84ac9aSAndroid Build Coastguard Worker return send_query(tcp, fd, &req, sizeof(req));
284*cf84ac9aSAndroid Build Coastguard Worker }
285*cf84ac9aSAndroid Build Coastguard Worker
286*cf84ac9aSAndroid Build Coastguard Worker static int
unix_parse_response(const void * data,const int data_len,const unsigned long inode,void * opaque_data)287*cf84ac9aSAndroid Build Coastguard Worker unix_parse_response(const void *data, const int data_len,
288*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode, void *opaque_data)
289*cf84ac9aSAndroid Build Coastguard Worker {
290*cf84ac9aSAndroid Build Coastguard Worker const char *proto_name = opaque_data;
291*cf84ac9aSAndroid Build Coastguard Worker const struct unix_diag_msg *diag_msg = data;
292*cf84ac9aSAndroid Build Coastguard Worker struct rtattr *attr;
293*cf84ac9aSAndroid Build Coastguard Worker int rta_len = data_len - NLMSG_LENGTH(sizeof(*diag_msg));
294*cf84ac9aSAndroid Build Coastguard Worker uint32_t peer = 0;
295*cf84ac9aSAndroid Build Coastguard Worker size_t path_len = 0;
296*cf84ac9aSAndroid Build Coastguard Worker char path[UNIX_PATH_MAX + 1];
297*cf84ac9aSAndroid Build Coastguard Worker
298*cf84ac9aSAndroid Build Coastguard Worker if (rta_len < 0)
299*cf84ac9aSAndroid Build Coastguard Worker return -1;
300*cf84ac9aSAndroid Build Coastguard Worker if (diag_msg->udiag_ino != inode)
301*cf84ac9aSAndroid Build Coastguard Worker return 0;
302*cf84ac9aSAndroid Build Coastguard Worker if (diag_msg->udiag_family != AF_UNIX)
303*cf84ac9aSAndroid Build Coastguard Worker return -1;
304*cf84ac9aSAndroid Build Coastguard Worker
305*cf84ac9aSAndroid Build Coastguard Worker for (attr = (struct rtattr *) (diag_msg + 1);
306*cf84ac9aSAndroid Build Coastguard Worker RTA_OK(attr, rta_len);
307*cf84ac9aSAndroid Build Coastguard Worker attr = RTA_NEXT(attr, rta_len)) {
308*cf84ac9aSAndroid Build Coastguard Worker switch (attr->rta_type) {
309*cf84ac9aSAndroid Build Coastguard Worker case UNIX_DIAG_NAME:
310*cf84ac9aSAndroid Build Coastguard Worker if (!path_len) {
311*cf84ac9aSAndroid Build Coastguard Worker path_len = RTA_PAYLOAD(attr);
312*cf84ac9aSAndroid Build Coastguard Worker if (path_len > UNIX_PATH_MAX)
313*cf84ac9aSAndroid Build Coastguard Worker path_len = UNIX_PATH_MAX;
314*cf84ac9aSAndroid Build Coastguard Worker memcpy(path, RTA_DATA(attr), path_len);
315*cf84ac9aSAndroid Build Coastguard Worker path[path_len] = '\0';
316*cf84ac9aSAndroid Build Coastguard Worker }
317*cf84ac9aSAndroid Build Coastguard Worker break;
318*cf84ac9aSAndroid Build Coastguard Worker case UNIX_DIAG_PEER:
319*cf84ac9aSAndroid Build Coastguard Worker if (RTA_PAYLOAD(attr) >= 4)
320*cf84ac9aSAndroid Build Coastguard Worker peer = *(uint32_t *) RTA_DATA(attr);
321*cf84ac9aSAndroid Build Coastguard Worker break;
322*cf84ac9aSAndroid Build Coastguard Worker }
323*cf84ac9aSAndroid Build Coastguard Worker }
324*cf84ac9aSAndroid Build Coastguard Worker
325*cf84ac9aSAndroid Build Coastguard Worker /*
326*cf84ac9aSAndroid Build Coastguard Worker * print obtained information in the following format:
327*cf84ac9aSAndroid Build Coastguard Worker * "UNIX:[" SELF_INODE [ "->" PEER_INODE ][ "," SOCKET_FILE ] "]"
328*cf84ac9aSAndroid Build Coastguard Worker */
329*cf84ac9aSAndroid Build Coastguard Worker if (!peer && !path_len)
330*cf84ac9aSAndroid Build Coastguard Worker return -1;
331*cf84ac9aSAndroid Build Coastguard Worker
332*cf84ac9aSAndroid Build Coastguard Worker char peer_str[3 + sizeof(peer) * 3];
333*cf84ac9aSAndroid Build Coastguard Worker if (peer)
334*cf84ac9aSAndroid Build Coastguard Worker xsprintf(peer_str, "->%u", peer);
335*cf84ac9aSAndroid Build Coastguard Worker else
336*cf84ac9aSAndroid Build Coastguard Worker peer_str[0] = '\0';
337*cf84ac9aSAndroid Build Coastguard Worker
338*cf84ac9aSAndroid Build Coastguard Worker const char *path_str;
339*cf84ac9aSAndroid Build Coastguard Worker if (path_len) {
340*cf84ac9aSAndroid Build Coastguard Worker char *outstr = alloca(4 * path_len + 4);
341*cf84ac9aSAndroid Build Coastguard Worker
342*cf84ac9aSAndroid Build Coastguard Worker outstr[0] = ',';
343*cf84ac9aSAndroid Build Coastguard Worker if (path[0] == '\0') {
344*cf84ac9aSAndroid Build Coastguard Worker outstr[1] = '@';
345*cf84ac9aSAndroid Build Coastguard Worker string_quote(path + 1, outstr + 2,
346*cf84ac9aSAndroid Build Coastguard Worker path_len - 1, QUOTE_0_TERMINATED, NULL);
347*cf84ac9aSAndroid Build Coastguard Worker } else {
348*cf84ac9aSAndroid Build Coastguard Worker string_quote(path, outstr + 1,
349*cf84ac9aSAndroid Build Coastguard Worker path_len, QUOTE_0_TERMINATED, NULL);
350*cf84ac9aSAndroid Build Coastguard Worker }
351*cf84ac9aSAndroid Build Coastguard Worker path_str = outstr;
352*cf84ac9aSAndroid Build Coastguard Worker } else {
353*cf84ac9aSAndroid Build Coastguard Worker path_str = "";
354*cf84ac9aSAndroid Build Coastguard Worker }
355*cf84ac9aSAndroid Build Coastguard Worker
356*cf84ac9aSAndroid Build Coastguard Worker char *details;
357*cf84ac9aSAndroid Build Coastguard Worker if (asprintf(&details, "%s:[%lu%s%s]", proto_name, inode,
358*cf84ac9aSAndroid Build Coastguard Worker peer_str, path_str) < 0)
359*cf84ac9aSAndroid Build Coastguard Worker return -1;
360*cf84ac9aSAndroid Build Coastguard Worker
361*cf84ac9aSAndroid Build Coastguard Worker return cache_inode_details(inode, details);
362*cf84ac9aSAndroid Build Coastguard Worker }
363*cf84ac9aSAndroid Build Coastguard Worker
364*cf84ac9aSAndroid Build Coastguard Worker static bool
netlink_send_query(struct tcb * tcp,const int fd,const unsigned long inode)365*cf84ac9aSAndroid Build Coastguard Worker netlink_send_query(struct tcb *tcp, const int fd, const unsigned long inode)
366*cf84ac9aSAndroid Build Coastguard Worker {
367*cf84ac9aSAndroid Build Coastguard Worker struct {
368*cf84ac9aSAndroid Build Coastguard Worker const struct nlmsghdr nlh;
369*cf84ac9aSAndroid Build Coastguard Worker const struct netlink_diag_req ndr;
370*cf84ac9aSAndroid Build Coastguard Worker } req = {
371*cf84ac9aSAndroid Build Coastguard Worker .nlh = {
372*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_len = sizeof(req),
373*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_type = SOCK_DIAG_BY_FAMILY,
374*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_flags = NLM_F_DUMP | NLM_F_REQUEST
375*cf84ac9aSAndroid Build Coastguard Worker },
376*cf84ac9aSAndroid Build Coastguard Worker .ndr = {
377*cf84ac9aSAndroid Build Coastguard Worker .sdiag_family = AF_NETLINK,
378*cf84ac9aSAndroid Build Coastguard Worker .sdiag_protocol = NDIAG_PROTO_ALL
379*cf84ac9aSAndroid Build Coastguard Worker }
380*cf84ac9aSAndroid Build Coastguard Worker };
381*cf84ac9aSAndroid Build Coastguard Worker return send_query(tcp, fd, &req, sizeof(req));
382*cf84ac9aSAndroid Build Coastguard Worker }
383*cf84ac9aSAndroid Build Coastguard Worker
384*cf84ac9aSAndroid Build Coastguard Worker static int
netlink_parse_response(const void * data,const int data_len,const unsigned long inode,void * opaque_data)385*cf84ac9aSAndroid Build Coastguard Worker netlink_parse_response(const void *data, const int data_len,
386*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode, void *opaque_data)
387*cf84ac9aSAndroid Build Coastguard Worker {
388*cf84ac9aSAndroid Build Coastguard Worker const char *proto_name = opaque_data;
389*cf84ac9aSAndroid Build Coastguard Worker const struct netlink_diag_msg *const diag_msg = data;
390*cf84ac9aSAndroid Build Coastguard Worker const char *netlink_proto;
391*cf84ac9aSAndroid Build Coastguard Worker char *details;
392*cf84ac9aSAndroid Build Coastguard Worker
393*cf84ac9aSAndroid Build Coastguard Worker if (data_len < (int) NLMSG_LENGTH(sizeof(*diag_msg)))
394*cf84ac9aSAndroid Build Coastguard Worker return -1;
395*cf84ac9aSAndroid Build Coastguard Worker if (diag_msg->ndiag_ino != inode)
396*cf84ac9aSAndroid Build Coastguard Worker return 0;
397*cf84ac9aSAndroid Build Coastguard Worker
398*cf84ac9aSAndroid Build Coastguard Worker if (diag_msg->ndiag_family != AF_NETLINK)
399*cf84ac9aSAndroid Build Coastguard Worker return -1;
400*cf84ac9aSAndroid Build Coastguard Worker
401*cf84ac9aSAndroid Build Coastguard Worker netlink_proto = xlookup(netlink_protocols,
402*cf84ac9aSAndroid Build Coastguard Worker diag_msg->ndiag_protocol);
403*cf84ac9aSAndroid Build Coastguard Worker
404*cf84ac9aSAndroid Build Coastguard Worker if (netlink_proto) {
405*cf84ac9aSAndroid Build Coastguard Worker netlink_proto = STR_STRIP_PREFIX(netlink_proto, "NETLINK_");
406*cf84ac9aSAndroid Build Coastguard Worker if (asprintf(&details, "%s:[%s:%u]", proto_name,
407*cf84ac9aSAndroid Build Coastguard Worker netlink_proto, diag_msg->ndiag_portid) < 0)
408*cf84ac9aSAndroid Build Coastguard Worker return -1;
409*cf84ac9aSAndroid Build Coastguard Worker } else {
410*cf84ac9aSAndroid Build Coastguard Worker if (asprintf(&details, "%s:[%u]", proto_name,
411*cf84ac9aSAndroid Build Coastguard Worker (unsigned) diag_msg->ndiag_protocol) < 0)
412*cf84ac9aSAndroid Build Coastguard Worker return -1;
413*cf84ac9aSAndroid Build Coastguard Worker }
414*cf84ac9aSAndroid Build Coastguard Worker
415*cf84ac9aSAndroid Build Coastguard Worker return cache_inode_details(inode, details);
416*cf84ac9aSAndroid Build Coastguard Worker }
417*cf84ac9aSAndroid Build Coastguard Worker
418*cf84ac9aSAndroid Build Coastguard Worker static const char *
unix_get(struct tcb * tcp,const int fd,const int family,const int proto,const unsigned long inode,const char * name)419*cf84ac9aSAndroid Build Coastguard Worker unix_get(struct tcb *tcp, const int fd, const int family, const int proto,
420*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode, const char *name)
421*cf84ac9aSAndroid Build Coastguard Worker {
422*cf84ac9aSAndroid Build Coastguard Worker return unix_send_query(tcp, fd, inode)
423*cf84ac9aSAndroid Build Coastguard Worker && receive_responses(tcp, fd, inode, SOCK_DIAG_BY_FAMILY,
424*cf84ac9aSAndroid Build Coastguard Worker unix_parse_response, (void *) name)
425*cf84ac9aSAndroid Build Coastguard Worker ? get_sockaddr_by_inode_cached(inode) : NULL;
426*cf84ac9aSAndroid Build Coastguard Worker }
427*cf84ac9aSAndroid Build Coastguard Worker
428*cf84ac9aSAndroid Build Coastguard Worker static const char *
inet_get(struct tcb * tcp,const int fd,const int family,const int protocol,const unsigned long inode,const char * proto_name)429*cf84ac9aSAndroid Build Coastguard Worker inet_get(struct tcb *tcp, const int fd, const int family, const int protocol,
430*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode, const char *proto_name)
431*cf84ac9aSAndroid Build Coastguard Worker {
432*cf84ac9aSAndroid Build Coastguard Worker return inet_send_query(tcp, fd, family, protocol)
433*cf84ac9aSAndroid Build Coastguard Worker && receive_responses(tcp, fd, inode, SOCK_DIAG_BY_FAMILY,
434*cf84ac9aSAndroid Build Coastguard Worker inet_parse_response, (void *) proto_name)
435*cf84ac9aSAndroid Build Coastguard Worker ? get_sockaddr_by_inode_cached(inode) : NULL;
436*cf84ac9aSAndroid Build Coastguard Worker }
437*cf84ac9aSAndroid Build Coastguard Worker
438*cf84ac9aSAndroid Build Coastguard Worker static const char *
netlink_get(struct tcb * tcp,const int fd,const int family,const int protocol,const unsigned long inode,const char * proto_name)439*cf84ac9aSAndroid Build Coastguard Worker netlink_get(struct tcb *tcp, const int fd, const int family, const int protocol,
440*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode, const char *proto_name)
441*cf84ac9aSAndroid Build Coastguard Worker {
442*cf84ac9aSAndroid Build Coastguard Worker return netlink_send_query(tcp, fd, inode)
443*cf84ac9aSAndroid Build Coastguard Worker && receive_responses(tcp, fd, inode, SOCK_DIAG_BY_FAMILY,
444*cf84ac9aSAndroid Build Coastguard Worker netlink_parse_response,
445*cf84ac9aSAndroid Build Coastguard Worker (void *) proto_name)
446*cf84ac9aSAndroid Build Coastguard Worker ? get_sockaddr_by_inode_cached(inode) : NULL;
447*cf84ac9aSAndroid Build Coastguard Worker }
448*cf84ac9aSAndroid Build Coastguard Worker
449*cf84ac9aSAndroid Build Coastguard Worker static const struct {
450*cf84ac9aSAndroid Build Coastguard Worker const char *const name;
451*cf84ac9aSAndroid Build Coastguard Worker const char * (*const get)(struct tcb *, int fd, int family,
452*cf84ac9aSAndroid Build Coastguard Worker int protocol, unsigned long inode,
453*cf84ac9aSAndroid Build Coastguard Worker const char *proto_name);
454*cf84ac9aSAndroid Build Coastguard Worker int family;
455*cf84ac9aSAndroid Build Coastguard Worker int proto;
456*cf84ac9aSAndroid Build Coastguard Worker } protocols[] = {
457*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_UNIX] = { "UNIX", unix_get, AF_UNIX},
458*cf84ac9aSAndroid Build Coastguard Worker /*
459*cf84ac9aSAndroid Build Coastguard Worker * inet_diag handlers are currently implemented only for TCP,
460*cf84ac9aSAndroid Build Coastguard Worker * UDP(lite), SCTP, RAW, and DCCP, but we try to resolve it for all
461*cf84ac9aSAndroid Build Coastguard Worker * protocols anyway, just in case.
462*cf84ac9aSAndroid Build Coastguard Worker */
463*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_TCP] =
464*cf84ac9aSAndroid Build Coastguard Worker { "TCP", inet_get, AF_INET, IPPROTO_TCP },
465*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_UDP] =
466*cf84ac9aSAndroid Build Coastguard Worker { "UDP", inet_get, AF_INET, IPPROTO_UDP },
467*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_UDPLITE] =
468*cf84ac9aSAndroid Build Coastguard Worker { "UDPLITE", inet_get, AF_INET, IPPROTO_UDPLITE },
469*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_DCCP] =
470*cf84ac9aSAndroid Build Coastguard Worker { "DCCP", inet_get, AF_INET, IPPROTO_DCCP },
471*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_SCTP] =
472*cf84ac9aSAndroid Build Coastguard Worker { "SCTP", inet_get, AF_INET, IPPROTO_SCTP },
473*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_L2TP_IP] =
474*cf84ac9aSAndroid Build Coastguard Worker { "L2TP/IP", inet_get, AF_INET, IPPROTO_L2TP },
475*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_PING] =
476*cf84ac9aSAndroid Build Coastguard Worker { "PING", inet_get, AF_INET, IPPROTO_ICMP },
477*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_RAW] =
478*cf84ac9aSAndroid Build Coastguard Worker { "RAW", inet_get, AF_INET, IPPROTO_RAW },
479*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_TCPv6] =
480*cf84ac9aSAndroid Build Coastguard Worker { "TCPv6", inet_get, AF_INET6, IPPROTO_TCP },
481*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_UDPv6] =
482*cf84ac9aSAndroid Build Coastguard Worker { "UDPv6", inet_get, AF_INET6, IPPROTO_UDP },
483*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_UDPLITEv6] =
484*cf84ac9aSAndroid Build Coastguard Worker { "UDPLITEv6", inet_get, AF_INET6, IPPROTO_UDPLITE },
485*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_DCCPv6] =
486*cf84ac9aSAndroid Build Coastguard Worker { "DCCPv6", inet_get, AF_INET6, IPPROTO_DCCP },
487*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_SCTPv6] =
488*cf84ac9aSAndroid Build Coastguard Worker { "SCTPv6", inet_get, AF_INET6, IPPROTO_SCTP },
489*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_L2TP_IPv6] =
490*cf84ac9aSAndroid Build Coastguard Worker { "L2TP/IPv6", inet_get, AF_INET6, IPPROTO_L2TP },
491*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_PINGv6] =
492*cf84ac9aSAndroid Build Coastguard Worker { "PINGv6", inet_get, AF_INET6, IPPROTO_ICMP },
493*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_RAWv6] =
494*cf84ac9aSAndroid Build Coastguard Worker { "RAWv6", inet_get, AF_INET6, IPPROTO_RAW },
495*cf84ac9aSAndroid Build Coastguard Worker [SOCK_PROTO_NETLINK] = { "NETLINK", netlink_get, AF_NETLINK },
496*cf84ac9aSAndroid Build Coastguard Worker };
497*cf84ac9aSAndroid Build Coastguard Worker
498*cf84ac9aSAndroid Build Coastguard Worker enum sock_proto
get_proto_by_name(const char * const name)499*cf84ac9aSAndroid Build Coastguard Worker get_proto_by_name(const char *const name)
500*cf84ac9aSAndroid Build Coastguard Worker {
501*cf84ac9aSAndroid Build Coastguard Worker unsigned int i;
502*cf84ac9aSAndroid Build Coastguard Worker for (i = (unsigned int) SOCK_PROTO_UNKNOWN + 1;
503*cf84ac9aSAndroid Build Coastguard Worker i < ARRAY_SIZE(protocols); ++i) {
504*cf84ac9aSAndroid Build Coastguard Worker if (protocols[i].name && !strcmp(name, protocols[i].name))
505*cf84ac9aSAndroid Build Coastguard Worker return (enum sock_proto) i;
506*cf84ac9aSAndroid Build Coastguard Worker }
507*cf84ac9aSAndroid Build Coastguard Worker return SOCK_PROTO_UNKNOWN;
508*cf84ac9aSAndroid Build Coastguard Worker }
509*cf84ac9aSAndroid Build Coastguard Worker
510*cf84ac9aSAndroid Build Coastguard Worker int
get_family_by_proto(enum sock_proto proto)511*cf84ac9aSAndroid Build Coastguard Worker get_family_by_proto(enum sock_proto proto)
512*cf84ac9aSAndroid Build Coastguard Worker {
513*cf84ac9aSAndroid Build Coastguard Worker if ((size_t) proto < ARRAY_SIZE(protocols))
514*cf84ac9aSAndroid Build Coastguard Worker return protocols[proto].family;
515*cf84ac9aSAndroid Build Coastguard Worker
516*cf84ac9aSAndroid Build Coastguard Worker return AF_UNSPEC;
517*cf84ac9aSAndroid Build Coastguard Worker }
518*cf84ac9aSAndroid Build Coastguard Worker
519*cf84ac9aSAndroid Build Coastguard Worker static const char *
get_sockaddr_by_inode_uncached(struct tcb * tcp,const unsigned long inode,const enum sock_proto proto)520*cf84ac9aSAndroid Build Coastguard Worker get_sockaddr_by_inode_uncached(struct tcb *tcp, const unsigned long inode,
521*cf84ac9aSAndroid Build Coastguard Worker const enum sock_proto proto)
522*cf84ac9aSAndroid Build Coastguard Worker {
523*cf84ac9aSAndroid Build Coastguard Worker if ((unsigned int) proto >= ARRAY_SIZE(protocols) ||
524*cf84ac9aSAndroid Build Coastguard Worker (proto != SOCK_PROTO_UNKNOWN && !protocols[proto].get))
525*cf84ac9aSAndroid Build Coastguard Worker return NULL;
526*cf84ac9aSAndroid Build Coastguard Worker
527*cf84ac9aSAndroid Build Coastguard Worker const int fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_SOCK_DIAG);
528*cf84ac9aSAndroid Build Coastguard Worker if (fd < 0)
529*cf84ac9aSAndroid Build Coastguard Worker return NULL;
530*cf84ac9aSAndroid Build Coastguard Worker const char *details = NULL;
531*cf84ac9aSAndroid Build Coastguard Worker
532*cf84ac9aSAndroid Build Coastguard Worker if (proto != SOCK_PROTO_UNKNOWN) {
533*cf84ac9aSAndroid Build Coastguard Worker details = protocols[proto].get(tcp, fd, protocols[proto].family,
534*cf84ac9aSAndroid Build Coastguard Worker protocols[proto].proto, inode,
535*cf84ac9aSAndroid Build Coastguard Worker protocols[proto].name);
536*cf84ac9aSAndroid Build Coastguard Worker } else {
537*cf84ac9aSAndroid Build Coastguard Worker unsigned int i;
538*cf84ac9aSAndroid Build Coastguard Worker for (i = (unsigned int) SOCK_PROTO_UNKNOWN + 1;
539*cf84ac9aSAndroid Build Coastguard Worker i < ARRAY_SIZE(protocols); ++i) {
540*cf84ac9aSAndroid Build Coastguard Worker if (!protocols[i].get)
541*cf84ac9aSAndroid Build Coastguard Worker continue;
542*cf84ac9aSAndroid Build Coastguard Worker details = protocols[i].get(tcp, fd,
543*cf84ac9aSAndroid Build Coastguard Worker protocols[proto].family,
544*cf84ac9aSAndroid Build Coastguard Worker protocols[proto].proto,
545*cf84ac9aSAndroid Build Coastguard Worker inode,
546*cf84ac9aSAndroid Build Coastguard Worker protocols[proto].name);
547*cf84ac9aSAndroid Build Coastguard Worker if (details)
548*cf84ac9aSAndroid Build Coastguard Worker break;
549*cf84ac9aSAndroid Build Coastguard Worker }
550*cf84ac9aSAndroid Build Coastguard Worker }
551*cf84ac9aSAndroid Build Coastguard Worker
552*cf84ac9aSAndroid Build Coastguard Worker close(fd);
553*cf84ac9aSAndroid Build Coastguard Worker return details;
554*cf84ac9aSAndroid Build Coastguard Worker }
555*cf84ac9aSAndroid Build Coastguard Worker
556*cf84ac9aSAndroid Build Coastguard Worker static bool
print_sockaddr_by_inode_uncached(struct tcb * tcp,const unsigned long inode,const enum sock_proto proto)557*cf84ac9aSAndroid Build Coastguard Worker print_sockaddr_by_inode_uncached(struct tcb *tcp, const unsigned long inode,
558*cf84ac9aSAndroid Build Coastguard Worker const enum sock_proto proto)
559*cf84ac9aSAndroid Build Coastguard Worker {
560*cf84ac9aSAndroid Build Coastguard Worker const char *details = get_sockaddr_by_inode_uncached(tcp, inode, proto);
561*cf84ac9aSAndroid Build Coastguard Worker
562*cf84ac9aSAndroid Build Coastguard Worker if (details) {
563*cf84ac9aSAndroid Build Coastguard Worker tprints(details);
564*cf84ac9aSAndroid Build Coastguard Worker return true;
565*cf84ac9aSAndroid Build Coastguard Worker }
566*cf84ac9aSAndroid Build Coastguard Worker
567*cf84ac9aSAndroid Build Coastguard Worker if ((unsigned int) proto < ARRAY_SIZE(protocols) &&
568*cf84ac9aSAndroid Build Coastguard Worker protocols[proto].name) {
569*cf84ac9aSAndroid Build Coastguard Worker tprintf("%s:[%lu]", protocols[proto].name, inode);
570*cf84ac9aSAndroid Build Coastguard Worker return true;
571*cf84ac9aSAndroid Build Coastguard Worker }
572*cf84ac9aSAndroid Build Coastguard Worker
573*cf84ac9aSAndroid Build Coastguard Worker return false;
574*cf84ac9aSAndroid Build Coastguard Worker }
575*cf84ac9aSAndroid Build Coastguard Worker
576*cf84ac9aSAndroid Build Coastguard Worker /* Given an inode number of a socket, return its protocol details. */
577*cf84ac9aSAndroid Build Coastguard Worker const char *
get_sockaddr_by_inode(struct tcb * const tcp,const int fd,const unsigned long inode)578*cf84ac9aSAndroid Build Coastguard Worker get_sockaddr_by_inode(struct tcb *const tcp, const int fd,
579*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode)
580*cf84ac9aSAndroid Build Coastguard Worker {
581*cf84ac9aSAndroid Build Coastguard Worker const char *details = get_sockaddr_by_inode_cached(inode);
582*cf84ac9aSAndroid Build Coastguard Worker return details ? details :
583*cf84ac9aSAndroid Build Coastguard Worker get_sockaddr_by_inode_uncached(tcp, inode, getfdproto(tcp, fd));
584*cf84ac9aSAndroid Build Coastguard Worker }
585*cf84ac9aSAndroid Build Coastguard Worker
586*cf84ac9aSAndroid Build Coastguard Worker /* Given an inode number of a socket, print out its protocol details. */
587*cf84ac9aSAndroid Build Coastguard Worker bool
print_sockaddr_by_inode(struct tcb * const tcp,const int fd,const unsigned long inode)588*cf84ac9aSAndroid Build Coastguard Worker print_sockaddr_by_inode(struct tcb *const tcp, const int fd,
589*cf84ac9aSAndroid Build Coastguard Worker const unsigned long inode)
590*cf84ac9aSAndroid Build Coastguard Worker {
591*cf84ac9aSAndroid Build Coastguard Worker return print_sockaddr_by_inode_cached(inode) ? true :
592*cf84ac9aSAndroid Build Coastguard Worker print_sockaddr_by_inode_uncached(tcp, inode,
593*cf84ac9aSAndroid Build Coastguard Worker getfdproto(tcp, fd));
594*cf84ac9aSAndroid Build Coastguard Worker }
595*cf84ac9aSAndroid Build Coastguard Worker
596*cf84ac9aSAndroid Build Coastguard Worker #ifdef HAVE_LINUX_GENETLINK_H
597*cf84ac9aSAndroid Build Coastguard Worker /*
598*cf84ac9aSAndroid Build Coastguard Worker * Managing the cache for decoding communications of Netlink GENERIC protocol
599*cf84ac9aSAndroid Build Coastguard Worker *
600*cf84ac9aSAndroid Build Coastguard Worker * As name shown Netlink GENERIC protocol is generic protocol. The
601*cf84ac9aSAndroid Build Coastguard Worker * numbers of msg types used in the protocol are not defined
602*cf84ac9aSAndroid Build Coastguard Worker * statically. Kernel defines them on demand. So the xlat converted
603*cf84ac9aSAndroid Build Coastguard Worker * from header files doesn't help for decoding the protocol. Following
604*cf84ac9aSAndroid Build Coastguard Worker * codes are building xlat(dyxlat) at runtime.
605*cf84ac9aSAndroid Build Coastguard Worker */
606*cf84ac9aSAndroid Build Coastguard Worker static bool
genl_send_dump_families(struct tcb * tcp,const int fd)607*cf84ac9aSAndroid Build Coastguard Worker genl_send_dump_families(struct tcb *tcp, const int fd)
608*cf84ac9aSAndroid Build Coastguard Worker {
609*cf84ac9aSAndroid Build Coastguard Worker struct {
610*cf84ac9aSAndroid Build Coastguard Worker const struct nlmsghdr nlh;
611*cf84ac9aSAndroid Build Coastguard Worker struct genlmsghdr gnlh;
612*cf84ac9aSAndroid Build Coastguard Worker } req = {
613*cf84ac9aSAndroid Build Coastguard Worker .nlh = {
614*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_len = sizeof(req),
615*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_type = GENL_ID_CTRL,
616*cf84ac9aSAndroid Build Coastguard Worker .nlmsg_flags = NLM_F_DUMP | NLM_F_REQUEST,
617*cf84ac9aSAndroid Build Coastguard Worker },
618*cf84ac9aSAndroid Build Coastguard Worker .gnlh = {
619*cf84ac9aSAndroid Build Coastguard Worker .cmd = CTRL_CMD_GETFAMILY,
620*cf84ac9aSAndroid Build Coastguard Worker }
621*cf84ac9aSAndroid Build Coastguard Worker };
622*cf84ac9aSAndroid Build Coastguard Worker return send_query(tcp, fd, &req, sizeof(req));
623*cf84ac9aSAndroid Build Coastguard Worker }
624*cf84ac9aSAndroid Build Coastguard Worker
625*cf84ac9aSAndroid Build Coastguard Worker static int
genl_parse_families_response(const void * const data,const int data_len,const unsigned long inode,void * opaque_data)626*cf84ac9aSAndroid Build Coastguard Worker genl_parse_families_response(const void *const data,
627*cf84ac9aSAndroid Build Coastguard Worker const int data_len, const unsigned long inode,
628*cf84ac9aSAndroid Build Coastguard Worker void *opaque_data)
629*cf84ac9aSAndroid Build Coastguard Worker {
630*cf84ac9aSAndroid Build Coastguard Worker struct dyxlat *const dyxlat = opaque_data;
631*cf84ac9aSAndroid Build Coastguard Worker const struct genlmsghdr *const gnlh = data;
632*cf84ac9aSAndroid Build Coastguard Worker struct rtattr *attr;
633*cf84ac9aSAndroid Build Coastguard Worker int rta_len = data_len - NLMSG_LENGTH(sizeof(*gnlh));
634*cf84ac9aSAndroid Build Coastguard Worker
635*cf84ac9aSAndroid Build Coastguard Worker char *name = NULL;
636*cf84ac9aSAndroid Build Coastguard Worker unsigned int name_len = 0;
637*cf84ac9aSAndroid Build Coastguard Worker uint16_t *id = NULL;
638*cf84ac9aSAndroid Build Coastguard Worker
639*cf84ac9aSAndroid Build Coastguard Worker if (rta_len < 0)
640*cf84ac9aSAndroid Build Coastguard Worker return -1;
641*cf84ac9aSAndroid Build Coastguard Worker if (gnlh->cmd != CTRL_CMD_NEWFAMILY)
642*cf84ac9aSAndroid Build Coastguard Worker return -1;
643*cf84ac9aSAndroid Build Coastguard Worker if (gnlh->version != 2)
644*cf84ac9aSAndroid Build Coastguard Worker return -1;
645*cf84ac9aSAndroid Build Coastguard Worker
646*cf84ac9aSAndroid Build Coastguard Worker for (attr = (struct rtattr *) (gnlh + 1);
647*cf84ac9aSAndroid Build Coastguard Worker RTA_OK(attr, rta_len);
648*cf84ac9aSAndroid Build Coastguard Worker attr = RTA_NEXT(attr, rta_len)) {
649*cf84ac9aSAndroid Build Coastguard Worker switch (attr->rta_type) {
650*cf84ac9aSAndroid Build Coastguard Worker case CTRL_ATTR_FAMILY_NAME:
651*cf84ac9aSAndroid Build Coastguard Worker if (!name) {
652*cf84ac9aSAndroid Build Coastguard Worker name = RTA_DATA(attr);
653*cf84ac9aSAndroid Build Coastguard Worker name_len = RTA_PAYLOAD(attr);
654*cf84ac9aSAndroid Build Coastguard Worker }
655*cf84ac9aSAndroid Build Coastguard Worker break;
656*cf84ac9aSAndroid Build Coastguard Worker case CTRL_ATTR_FAMILY_ID:
657*cf84ac9aSAndroid Build Coastguard Worker if (!id && RTA_PAYLOAD(attr) == sizeof(*id))
658*cf84ac9aSAndroid Build Coastguard Worker id = RTA_DATA(attr);
659*cf84ac9aSAndroid Build Coastguard Worker break;
660*cf84ac9aSAndroid Build Coastguard Worker }
661*cf84ac9aSAndroid Build Coastguard Worker
662*cf84ac9aSAndroid Build Coastguard Worker if (name && id) {
663*cf84ac9aSAndroid Build Coastguard Worker dyxlat_add_pair(dyxlat, *id, name, name_len);
664*cf84ac9aSAndroid Build Coastguard Worker name = NULL;
665*cf84ac9aSAndroid Build Coastguard Worker id = NULL;
666*cf84ac9aSAndroid Build Coastguard Worker }
667*cf84ac9aSAndroid Build Coastguard Worker }
668*cf84ac9aSAndroid Build Coastguard Worker
669*cf84ac9aSAndroid Build Coastguard Worker return 0;
670*cf84ac9aSAndroid Build Coastguard Worker }
671*cf84ac9aSAndroid Build Coastguard Worker
672*cf84ac9aSAndroid Build Coastguard Worker const struct xlat *
genl_families_xlat(struct tcb * tcp)673*cf84ac9aSAndroid Build Coastguard Worker genl_families_xlat(struct tcb *tcp)
674*cf84ac9aSAndroid Build Coastguard Worker {
675*cf84ac9aSAndroid Build Coastguard Worker static struct dyxlat *dyxlat;
676*cf84ac9aSAndroid Build Coastguard Worker
677*cf84ac9aSAndroid Build Coastguard Worker if (!dyxlat) {
678*cf84ac9aSAndroid Build Coastguard Worker dyxlat = dyxlat_alloc(32);
679*cf84ac9aSAndroid Build Coastguard Worker
680*cf84ac9aSAndroid Build Coastguard Worker int fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC);
681*cf84ac9aSAndroid Build Coastguard Worker if (fd < 0)
682*cf84ac9aSAndroid Build Coastguard Worker goto out;
683*cf84ac9aSAndroid Build Coastguard Worker
684*cf84ac9aSAndroid Build Coastguard Worker if (genl_send_dump_families(tcp, fd))
685*cf84ac9aSAndroid Build Coastguard Worker receive_responses(tcp, fd, 0, GENL_ID_CTRL,
686*cf84ac9aSAndroid Build Coastguard Worker genl_parse_families_response, dyxlat);
687*cf84ac9aSAndroid Build Coastguard Worker close(fd);
688*cf84ac9aSAndroid Build Coastguard Worker }
689*cf84ac9aSAndroid Build Coastguard Worker
690*cf84ac9aSAndroid Build Coastguard Worker out:
691*cf84ac9aSAndroid Build Coastguard Worker return dyxlat_get(dyxlat);
692*cf84ac9aSAndroid Build Coastguard Worker }
693*cf84ac9aSAndroid Build Coastguard Worker
694*cf84ac9aSAndroid Build Coastguard Worker #else /* !HAVE_LINUX_GENETLINK_H */
695*cf84ac9aSAndroid Build Coastguard Worker
696*cf84ac9aSAndroid Build Coastguard Worker const struct xlat *
genl_families_xlat(struct tcb * tcp)697*cf84ac9aSAndroid Build Coastguard Worker genl_families_xlat(struct tcb *tcp)
698*cf84ac9aSAndroid Build Coastguard Worker {
699*cf84ac9aSAndroid Build Coastguard Worker return NULL;
700*cf84ac9aSAndroid Build Coastguard Worker }
701*cf84ac9aSAndroid Build Coastguard Worker #endif
702