Line data Source code
1 : /* SPDX-License-Identifier: BSD-3-Clause
2 : * Copyright (C) 2021 Intel Corporation. All rights reserved.
3 : * Copyright (C) 2024 Samsung Electronics Co., Ltd.
4 : * All rights reserved.
5 : */
6 :
7 : #include "spdk/stdinc.h"
8 : #include "spdk/net.h"
9 : #include "spdk/log.h"
10 :
11 : int
12 0 : spdk_net_get_interface_name(const char *ip, char *ifc, size_t len)
13 : {
14 : struct ifaddrs *addrs, *iap;
15 : struct sockaddr_in *sa;
16 : char buf[32];
17 0 : int rc = -ENODEV;
18 :
19 0 : getifaddrs(&addrs);
20 0 : for (iap = addrs; iap != NULL; iap = iap->ifa_next) {
21 0 : if (!(iap->ifa_addr && (iap->ifa_flags & IFF_UP) && iap->ifa_addr->sa_family == AF_INET)) {
22 0 : continue;
23 : }
24 0 : sa = (struct sockaddr_in *)(iap->ifa_addr);
25 0 : inet_ntop(iap->ifa_addr->sa_family, &sa->sin_addr, buf, sizeof(buf));
26 0 : if (strcmp(ip, buf) != 0) {
27 0 : continue;
28 : }
29 0 : if (strnlen(iap->ifa_name, len) == len) {
30 0 : rc = -ENOMEM;
31 0 : goto ret;
32 : }
33 0 : snprintf(ifc, len, "%s", iap->ifa_name);
34 0 : rc = 0;
35 0 : break;
36 0 : }
37 : ret:
38 0 : freeifaddrs(addrs);
39 0 : return rc;
40 : }
41 :
42 : int
43 0 : spdk_net_get_address_string(struct sockaddr *sa, char *addr, size_t len)
44 : {
45 0 : const char *result = NULL;
46 :
47 0 : if (sa == NULL || addr == NULL) {
48 0 : return -1;
49 : }
50 :
51 0 : switch (sa->sa_family) {
52 : case AF_INET:
53 0 : result = inet_ntop(AF_INET, &(((struct sockaddr_in *)sa)->sin_addr),
54 0 : addr, len);
55 0 : break;
56 : case AF_INET6:
57 0 : result = inet_ntop(AF_INET6, &(((struct sockaddr_in6 *)sa)->sin6_addr),
58 0 : addr, len);
59 0 : break;
60 : default:
61 0 : break;
62 : }
63 :
64 0 : if (result != NULL) {
65 0 : return 0;
66 : } else {
67 0 : return -errno;
68 : }
69 0 : }
70 :
71 : bool
72 0 : spdk_net_is_loopback(int fd)
73 : {
74 : struct ifaddrs *addrs, *tmp;
75 0 : struct sockaddr_storage sa = {};
76 : socklen_t salen;
77 0 : struct ifreq ifr = {};
78 : char ip_addr[256], ip_addr_tmp[256];
79 : int rc;
80 0 : bool is_loopback = false;
81 :
82 0 : salen = sizeof(sa);
83 0 : rc = getsockname(fd, (struct sockaddr *)&sa, &salen);
84 0 : if (rc != 0) {
85 0 : return is_loopback;
86 : }
87 :
88 0 : memset(ip_addr, 0, sizeof(ip_addr));
89 0 : rc = spdk_net_get_address_string((struct sockaddr *)&sa, ip_addr, sizeof(ip_addr));
90 0 : if (rc != 0) {
91 0 : return is_loopback;
92 : }
93 :
94 0 : getifaddrs(&addrs);
95 0 : for (tmp = addrs; tmp != NULL; tmp = tmp->ifa_next) {
96 0 : if (tmp->ifa_addr && (tmp->ifa_flags & IFF_UP) &&
97 0 : (tmp->ifa_addr->sa_family == sa.ss_family)) {
98 0 : memset(ip_addr_tmp, 0, sizeof(ip_addr_tmp));
99 0 : rc = spdk_net_get_address_string(tmp->ifa_addr, ip_addr_tmp, sizeof(ip_addr_tmp));
100 0 : if (rc != 0) {
101 0 : continue;
102 : }
103 :
104 0 : if (strncmp(ip_addr, ip_addr_tmp, sizeof(ip_addr)) == 0) {
105 0 : memcpy(ifr.ifr_name, tmp->ifa_name, sizeof(ifr.ifr_name));
106 0 : ioctl(fd, SIOCGIFFLAGS, &ifr);
107 0 : if (ifr.ifr_flags & IFF_LOOPBACK) {
108 0 : is_loopback = true;
109 0 : }
110 0 : goto end;
111 : }
112 0 : }
113 0 : }
114 :
115 : end:
116 0 : freeifaddrs(addrs);
117 0 : return is_loopback;
118 0 : }
119 :
120 : int
121 0 : spdk_net_getaddr(int fd, char *laddr, int llen, uint16_t *lport,
122 : char *paddr, int plen, uint16_t *pport)
123 : {
124 : struct sockaddr_storage sa;
125 : int val;
126 : socklen_t len;
127 : int rc;
128 :
129 0 : memset(&sa, 0, sizeof(sa));
130 0 : len = sizeof(sa);
131 0 : rc = getsockname(fd, (struct sockaddr *)&sa, &len);
132 0 : if (rc != 0) {
133 0 : SPDK_ERRLOG("getsockname() failed (errno=%d)\n", errno);
134 0 : return -1;
135 : }
136 :
137 0 : switch (sa.ss_family) {
138 : case AF_UNIX:
139 : /* Acceptable connection types that don't have IPs */
140 0 : return 0;
141 : case AF_INET:
142 : case AF_INET6:
143 : /* Code below will get IP addresses */
144 0 : break;
145 : default:
146 : /* Unsupported socket family */
147 0 : return -1;
148 : }
149 :
150 0 : if (laddr) {
151 0 : rc = spdk_net_get_address_string((struct sockaddr *)&sa, laddr, llen);
152 0 : if (rc != 0) {
153 0 : SPDK_ERRLOG("spdk_net_get_address_string() failed (errno=%d)\n", rc);
154 0 : return -1;
155 : }
156 0 : }
157 :
158 0 : if (lport) {
159 0 : if (sa.ss_family == AF_INET) {
160 0 : *lport = ntohs(((struct sockaddr_in *)&sa)->sin_port);
161 0 : } else if (sa.ss_family == AF_INET6) {
162 0 : *lport = ntohs(((struct sockaddr_in6 *)&sa)->sin6_port);
163 0 : }
164 0 : }
165 :
166 0 : len = sizeof(val);
167 0 : rc = getsockopt(fd, SOL_SOCKET, SO_ACCEPTCONN, &val, &len);
168 0 : if (rc == 0 && val == 1) {
169 : /* It is an error to getaddr for a peer address on a listen socket. */
170 0 : if (paddr != NULL || pport != NULL) {
171 0 : SPDK_ERRLOG("paddr, pport not valid on listen sockets\n");
172 0 : return -1;
173 : }
174 0 : return 0;
175 : }
176 :
177 0 : memset(&sa, 0, sizeof(sa));
178 0 : len = sizeof(sa);
179 0 : rc = getpeername(fd, (struct sockaddr *)&sa, &len);
180 0 : if (rc != 0) {
181 0 : SPDK_ERRLOG("getpeername() failed (errno=%d)\n", errno);
182 0 : return -1;
183 : }
184 :
185 0 : if (paddr) {
186 0 : rc = spdk_net_get_address_string((struct sockaddr *)&sa, paddr, plen);
187 0 : if (rc != 0) {
188 0 : SPDK_ERRLOG("spdk_net_get_address_string() failed (errno=%d)\n", rc);
189 0 : return -1;
190 : }
191 0 : }
192 :
193 0 : if (pport) {
194 0 : if (sa.ss_family == AF_INET) {
195 0 : *pport = ntohs(((struct sockaddr_in *)&sa)->sin_port);
196 0 : } else if (sa.ss_family == AF_INET6) {
197 0 : *pport = ntohs(((struct sockaddr_in6 *)&sa)->sin6_port);
198 0 : }
199 0 : }
200 :
201 0 : return 0;
202 0 : }
|