xref: /aosp_15_r20/external/strace/socketutils.c (revision cf84ac9a129d8ea9952db616b4e9b904c4bdde56)
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