xref: /aosp_15_r20/external/musl/src/network/netlink.c (revision c9945492fdd68bbe62686c5b452b4dc1be3f8453)
1*c9945492SAndroid Build Coastguard Worker #include <errno.h>
2*c9945492SAndroid Build Coastguard Worker #include <string.h>
3*c9945492SAndroid Build Coastguard Worker #include <syscall.h>
4*c9945492SAndroid Build Coastguard Worker #include <sys/socket.h>
5*c9945492SAndroid Build Coastguard Worker #include "netlink.h"
6*c9945492SAndroid Build Coastguard Worker 
__netlink_enumerate(int fd,unsigned int seq,int type,int af,int (* cb)(void * ctx,struct nlmsghdr * h),void * ctx)7*c9945492SAndroid Build Coastguard Worker static int __netlink_enumerate(int fd, unsigned int seq, int type, int af,
8*c9945492SAndroid Build Coastguard Worker 	int (*cb)(void *ctx, struct nlmsghdr *h), void *ctx)
9*c9945492SAndroid Build Coastguard Worker {
10*c9945492SAndroid Build Coastguard Worker 	struct nlmsghdr *h;
11*c9945492SAndroid Build Coastguard Worker 	union {
12*c9945492SAndroid Build Coastguard Worker 		uint8_t buf[8192];
13*c9945492SAndroid Build Coastguard Worker 		struct {
14*c9945492SAndroid Build Coastguard Worker 			struct nlmsghdr nlh;
15*c9945492SAndroid Build Coastguard Worker 			struct rtgenmsg g;
16*c9945492SAndroid Build Coastguard Worker 		} req;
17*c9945492SAndroid Build Coastguard Worker 		struct nlmsghdr reply;
18*c9945492SAndroid Build Coastguard Worker 	} u;
19*c9945492SAndroid Build Coastguard Worker 	int r, ret;
20*c9945492SAndroid Build Coastguard Worker 
21*c9945492SAndroid Build Coastguard Worker 	memset(&u.req, 0, sizeof(u.req));
22*c9945492SAndroid Build Coastguard Worker 	u.req.nlh.nlmsg_len = sizeof(u.req);
23*c9945492SAndroid Build Coastguard Worker 	u.req.nlh.nlmsg_type = type;
24*c9945492SAndroid Build Coastguard Worker 	u.req.nlh.nlmsg_flags = NLM_F_DUMP | NLM_F_REQUEST;
25*c9945492SAndroid Build Coastguard Worker 	u.req.nlh.nlmsg_seq = seq;
26*c9945492SAndroid Build Coastguard Worker 	u.req.g.rtgen_family = af;
27*c9945492SAndroid Build Coastguard Worker 	r = send(fd, &u.req, sizeof(u.req), 0);
28*c9945492SAndroid Build Coastguard Worker 	if (r < 0) return r;
29*c9945492SAndroid Build Coastguard Worker 
30*c9945492SAndroid Build Coastguard Worker 	while (1) {
31*c9945492SAndroid Build Coastguard Worker 		r = recv(fd, u.buf, sizeof(u.buf), MSG_DONTWAIT);
32*c9945492SAndroid Build Coastguard Worker 		if (r <= 0) return -1;
33*c9945492SAndroid Build Coastguard Worker 		for (h = &u.reply; NLMSG_OK(h, (void*)&u.buf[r]); h = NLMSG_NEXT(h)) {
34*c9945492SAndroid Build Coastguard Worker 			if (h->nlmsg_type == NLMSG_DONE) return 0;
35*c9945492SAndroid Build Coastguard Worker 			if (h->nlmsg_type == NLMSG_ERROR) return -1;
36*c9945492SAndroid Build Coastguard Worker 			ret = cb(ctx, h);
37*c9945492SAndroid Build Coastguard Worker 			if (ret) return ret;
38*c9945492SAndroid Build Coastguard Worker 		}
39*c9945492SAndroid Build Coastguard Worker 	}
40*c9945492SAndroid Build Coastguard Worker }
41*c9945492SAndroid Build Coastguard Worker 
__rtnetlink_enumerate(int link_af,int addr_af,int (* cb)(void * ctx,struct nlmsghdr * h),void * ctx)42*c9945492SAndroid Build Coastguard Worker int __rtnetlink_enumerate(int link_af, int addr_af, int (*cb)(void *ctx, struct nlmsghdr *h), void *ctx)
43*c9945492SAndroid Build Coastguard Worker {
44*c9945492SAndroid Build Coastguard Worker 	int fd, r;
45*c9945492SAndroid Build Coastguard Worker 
46*c9945492SAndroid Build Coastguard Worker 	fd = socket(PF_NETLINK, SOCK_RAW|SOCK_CLOEXEC, NETLINK_ROUTE);
47*c9945492SAndroid Build Coastguard Worker 	if (fd < 0) return -1;
48*c9945492SAndroid Build Coastguard Worker 	r = __netlink_enumerate(fd, 1, RTM_GETLINK, link_af, cb, ctx);
49*c9945492SAndroid Build Coastguard Worker 	if (!r) r = __netlink_enumerate(fd, 2, RTM_GETADDR, addr_af, cb, ctx);
50*c9945492SAndroid Build Coastguard Worker 	__syscall(SYS_close,fd);
51*c9945492SAndroid Build Coastguard Worker 	return r;
52*c9945492SAndroid Build Coastguard Worker }
53