1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Copyright (C) 2022 Felix Fietkau <nbd@nbd.name>
6 #include <sys/socket.h>
8 #include <netinet/in.h>
9 #include <netinet/ip.h>
10 #include <netinet/ip6.h>
11 #include <netinet/udp.h>
18 static const char *pex_peer_id_str(const uint8_t *key
)
23 for (i
= 0; i
< 8; i
++)
24 sprintf(str
+ i
* 2, "%02x", key
[i
]);
29 static struct pex_hdr
*
30 pex_msg_init(struct network
*net
, uint8_t opcode
)
32 return __pex_msg_init(net
->config
.pubkey
, opcode
);
35 static struct pex_hdr
*
36 pex_msg_init_ext(struct network
*net
, uint8_t opcode
, bool ext
)
38 return __pex_msg_init_ext(net
->config
.pubkey
, net
->config
.auth_key
, opcode
, ext
);
41 static struct network_peer
*
42 pex_msg_peer(struct network
*net
, const uint8_t *id
)
44 struct network_peer
*peer
;
45 uint8_t key
[WG_KEY_LEN
] = {};
47 memcpy(key
, id
, PEX_ID_LEN
);
48 peer
= avl_find_ge_element(&net
->peers
.avl
, key
, peer
, node
.avl
);
49 if (!peer
|| memcmp(peer
->key
, key
, PEX_ID_LEN
) != 0) {
50 D_NET(net
, "can't find peer %s", pex_peer_id_str(id
));
58 pex_get_peer_addr(struct sockaddr_in6
*sin6
, struct network
*net
,
59 struct network_peer
*peer
)
61 *sin6
= (struct sockaddr_in6
){
62 .sin6_family
= AF_INET6
,
63 .sin6_addr
= peer
->local_addr
.in6
,
64 .sin6_port
= htons(peer
->pex_port
),
68 static void pex_msg_send(struct network
*net
, struct network_peer
*peer
)
70 struct sockaddr_in6 sin6
= {};
72 if (!peer
|| peer
== &net
->net_config
.local_host
->peer
||
76 pex_get_peer_addr(&sin6
, net
, peer
);
77 if (__pex_msg_send(net
->pex
.fd
.fd
, &sin6
, NULL
, 0) < 0)
78 D_PEER(net
, peer
, "pex_msg_send failed: %s", strerror(errno
));
81 static void pex_msg_send_ext(struct network
*net
, struct network_peer
*peer
,
82 struct sockaddr_in6
*addr
)
84 char addrbuf
[INET6_ADDRSTRLEN
];
87 return pex_msg_send(net
, peer
);
89 if (__pex_msg_send(-1, addr
, NULL
, 0) < 0)
90 D_NET(net
, "pex_msg_send_ext(%s) failed: %s",
91 inet_ntop(addr
->sin6_family
, (const void *)&addr
->sin6_addr
, addrbuf
,
97 pex_send_hello(struct network
*net
, struct network_peer
*peer
)
99 struct pex_hello
*data
;
101 pex_msg_init(net
, PEX_MSG_HELLO
);
102 data
= pex_msg_append(sizeof(*data
));
103 if (peer
->state
.endpoint
.sa
.sa_family
== AF_INET6
)
104 data
->flags
|= htons(PEER_EP_F_IPV6
);
105 if (network_get_local_addr(&data
->local_addr
, &peer
->state
.endpoint
))
108 pex_msg_send(net
, peer
);
112 pex_msg_add_peer_endpoint(struct network
*net
, struct network_peer
*peer
,
113 struct network_peer
*receiver
)
115 struct pex_peer_endpoint
*data
;
121 addr
= network_endpoint_addr(&peer
->state
.endpoint
, &len
);
122 port
= peer
->state
.endpoint
.in
.sin_port
;
124 flags
|= PEER_EP_F_IPV6
;
125 if (network_endpoint_addr_equal(&peer
->state
.endpoint
,
126 &receiver
->state
.endpoint
)) {
127 if (!peer
->state
.has_local_ep_addr
) {
128 D_PEER(net
, peer
, "can't send peer to %s, missing local address",
129 network_peer_name(receiver
));
133 addr
= &peer
->state
.local_ep_addr
;
134 port
= htons(peer
->port
);
135 flags
|= PEER_EP_F_LOCAL
;
138 data
= pex_msg_append(sizeof(*data
));
142 memcpy(data
->peer_id
, peer
->key
, sizeof(data
->peer_id
));
143 memcpy(data
->addr
, addr
, len
);
145 data
->flags
= htons(flags
);
146 D_PEER(net
, peer
, "send endpoint to %s", network_peer_name(receiver
));
152 network_pex_handle_endpoint_change(struct network
*net
, struct network_peer
*peer
)
154 struct network_peer
*cur
;
156 vlist_for_each_element(&net
->peers
, cur
, node
) {
157 if (cur
== peer
|| !cur
->state
.connected
)
160 pex_msg_init(net
, PEX_MSG_NOTIFY_PEERS
);
161 if (pex_msg_add_peer_endpoint(net
, peer
, cur
))
164 pex_msg_send(net
, cur
);
169 network_pex_host_request_update(struct network
*net
, struct network_pex_host
*host
)
182 char addrstr
[INET6_ADDRSTRLEN
];
183 union network_endpoint dest_ep
;
184 union network_addr local_addr
= {};
185 uint64_t version
= 0;
188 if (net
->net_data_len
)
189 version
= net
->net_data_version
;
191 D("request network data from host %s",
192 inet_ntop(host
->endpoint
.sa
.sa_family
,
193 (host
->endpoint
.sa
.sa_family
== AF_INET6
?
194 (const void *)&host
->endpoint
.in6
.sin6_addr
:
195 (const void *)&host
->endpoint
.in
.sin_addr
),
196 addrstr
, sizeof(addrstr
)));
198 if (!pex_msg_update_request_init(net
->config
.pubkey
, net
->config
.key
,
199 net
->config
.auth_key
, &host
->endpoint
,
203 __pex_msg_send(-1, &host
->endpoint
, NULL
, 0);
205 if (!net
->net_config
.local_host
)
208 pex_msg_init_ext(net
, PEX_MSG_ENDPOINT_NOTIFY
, true);
210 memcpy(&dest_ep
, &host
->endpoint
, sizeof(dest_ep
));
212 /* work around issue with local address lookup for local broadcast */
213 if (host
->endpoint
.sa
.sa_family
== AF_INET
) {
214 uint8_t *data
= (uint8_t *)&dest_ep
.in
.sin_addr
;
219 network_get_local_addr(&local_addr
, &dest_ep
);
221 memset(&dest_ep
, 0, sizeof(dest_ep
));
222 dest_ep
.sa
.sa_family
= host
->endpoint
.sa
.sa_family
;
223 if (host
->endpoint
.sa
.sa_family
== AF_INET
) {
224 packet
.ipv4
.ip
= (struct ip
){
229 .ip_src
= local_addr
.in
,
230 .ip_dst
= host
->endpoint
.in
.sin_addr
,
232 dest_ep
.in
.sin_addr
= host
->endpoint
.in
.sin_addr
;
233 udp
= &packet
.ipv4
.udp
;
234 len
= sizeof(packet
.ipv4
);
236 packet
.ipv6
.ip
= (struct ip6_hdr
){
237 .ip6_flow
= htonl(6 << 28),
239 .ip6_nxt
= IPPROTO_UDP
,
240 .ip6_src
= local_addr
.in6
,
241 .ip6_dst
= host
->endpoint
.in6
.sin6_addr
,
243 dest_ep
.in6
.sin6_addr
= host
->endpoint
.in6
.sin6_addr
;
244 udp
= &packet
.ipv6
.udp
;
245 len
= sizeof(packet
.ipv6
);
248 udp
->uh_sport
= htons(net
->net_config
.local_host
->peer
.port
);
249 udp
->uh_dport
= host
->endpoint
.in6
.sin6_port
;
251 if (__pex_msg_send(-1, &dest_ep
, &packet
, len
) < 0)
252 D_NET(net
, "pex_msg_send_raw failed: %s", strerror(errno
));
256 network_pex_request_update_cb(struct uloop_timeout
*t
)
258 struct network
*net
= container_of(t
, struct network
, pex
.request_update_timer
);
259 struct network_pex
*pex
= &net
->pex
;
260 struct network_pex_host
*host
;
262 uloop_timeout_set(t
, 5000);
265 if (list_empty(&pex
->hosts
))
268 host
= list_first_entry(&pex
->hosts
, struct network_pex_host
, list
);
269 if (host
->timeout
&& host
->timeout
< unet_gettime()) {
270 list_del(&host
->list
);
275 list_move_tail(&host
->list
, &pex
->hosts
);
276 network_pex_host_request_update(net
, host
);
279 void network_pex_init(struct network
*net
)
281 struct network_pex
*pex
= &net
->pex
;
283 memset(pex
, 0, sizeof(*pex
));
285 INIT_LIST_HEAD(&pex
->hosts
);
286 pex
->request_update_timer
.cb
= network_pex_request_update_cb
;
290 network_pex_query_hosts(struct network
*net
)
292 struct network_host
*host
;
297 pex_msg_init(net
, PEX_MSG_QUERY
);
299 avl_for_each_element(&net
->hosts
, host
, node
) {
300 struct network_peer
*peer
= &host
->peer
;
303 if (host
== net
->net_config
.local_host
||
304 peer
->state
.connected
||
308 id
= pex_msg_append(PEX_ID_LEN
);
312 memcpy(id
, peer
->key
, PEX_ID_LEN
);
319 rv
%= net
->hosts
.count
;
320 for (i
= 0; i
< 2; i
++) {
321 avl_for_each_element(&net
->hosts
, host
, node
) {
322 struct network_peer
*peer
= &host
->peer
;
329 if (host
== net
->net_config
.local_host
)
332 if (!peer
->state
.connected
)
335 D_PEER(net
, peer
, "send query for %d hosts", hosts
);
336 pex_msg_send(net
, peer
);
344 network_pex_send_ping(struct network
*net
, struct network_peer
*peer
)
346 pex_msg_init(net
, PEX_MSG_PING
);
347 pex_msg_send(net
, peer
);
351 network_pex_send_update_request(struct network
*net
, struct network_peer
*peer
,
352 struct sockaddr_in6
*addr
)
354 union network_endpoint ep
= {};
355 uint64_t version
= 0;
358 memcpy(&ep
.in6
, addr
, sizeof(ep
.in6
));
360 pex_get_peer_addr(&ep
.in6
, net
, peer
);
362 if (net
->net_data_len
)
363 version
= net
->net_data_version
;
365 if (!pex_msg_update_request_init(net
->config
.pubkey
, net
->config
.key
,
366 net
->config
.auth_key
, &ep
,
370 pex_msg_send_ext(net
, peer
, addr
);
373 void network_pex_event(struct network
*net
, struct network_peer
*peer
,
376 if (!network_pex_active(&net
->pex
))
380 D_PEER(net
, peer
, "PEX event type=%d", ev
);
382 D_NET(net
, "PEX event type=%d", ev
);
385 case PEX_EV_HANDSHAKE
:
386 pex_send_hello(net
, peer
);
387 if (net
->config
.type
== NETWORK_TYPE_DYNAMIC
)
388 network_pex_send_update_request(net
, peer
, NULL
);
390 case PEX_EV_ENDPOINT_CHANGE
:
391 network_pex_handle_endpoint_change(net
, peer
);
394 network_pex_query_hosts(net
);
397 network_pex_send_ping(net
, peer
);
403 network_pex_recv_hello(struct network
*net
, struct network_peer
*peer
,
404 const struct pex_hello
*data
, size_t len
)
406 char addrstr
[INET6_ADDRSTRLEN
];
410 if (len
< sizeof(*data
))
413 if (peer
->state
.has_local_ep_addr
&&
414 !memcmp(&peer
->state
.local_ep_addr
, data
->local_addr
, sizeof(data
->local_addr
)))
417 flags
= ntohs(data
->flags
);
418 af
= (flags
& PEER_EP_F_IPV6
) ? AF_INET6
: AF_INET
;
419 D_PEER(net
, peer
, "set local endpoint address to %s",
420 inet_ntop(af
, data
->local_addr
, addrstr
, sizeof(addrstr
)));
421 peer
->state
.has_local_ep_addr
= true;
422 memcpy(&peer
->state
.local_ep_addr
, data
->local_addr
, sizeof(data
->local_addr
));
426 network_pex_recv_peers(struct network
*net
, struct network_peer
*peer
,
427 const struct pex_peer_endpoint
*data
, size_t len
)
429 struct network_peer
*local
= &net
->net_config
.local_host
->peer
;
430 struct network_peer
*cur
;
432 for (; len
>= sizeof(*data
); len
-= sizeof(*data
), data
++) {
433 union network_endpoint
*ep
;
438 cur
= pex_msg_peer(net
, data
->peer_id
);
442 if (cur
== peer
|| cur
== local
)
445 D_PEER(net
, peer
, "received peer address for %s",
446 network_peer_name(cur
));
447 flags
= ntohs(data
->flags
);
448 ep
= &cur
->state
.next_endpoint
;
449 ep
->sa
.sa_family
= (flags
& PEER_EP_F_IPV6
) ? AF_INET6
: AF_INET
;
450 addr
= network_endpoint_addr(ep
, &len
);
451 memcpy(addr
, data
->addr
, len
);
452 ep
->in
.sin_port
= data
->port
;
457 network_pex_recv_query(struct network
*net
, struct network_peer
*peer
,
458 const uint8_t *data
, size_t len
)
460 struct network_peer
*cur
;
463 pex_msg_init(net
, PEX_MSG_NOTIFY_PEERS
);
464 for (; len
>= 8; data
+= 8, len
-= 8) {
465 cur
= pex_msg_peer(net
, data
);
466 if (!cur
|| !cur
->state
.connected
)
469 if (!pex_msg_add_peer_endpoint(net
, cur
, peer
))
476 D_PEER(net
, peer
, "send query response with %d hosts", resp
);
477 pex_msg_send(net
, peer
);
481 network_pex_recv_ping(struct network
*net
, struct network_peer
*peer
)
483 time_t now
= time(NULL
);
485 if (peer
->state
.last_request
== now
)
488 peer
->state
.last_request
= now
;
489 pex_msg_init(net
, PEX_MSG_PONG
);
490 pex_msg_send(net
, peer
);
494 network_pex_recv_update_request(struct network
*net
, struct network_peer
*peer
,
495 const uint8_t *data
, size_t len
,
496 struct sockaddr_in6
*addr
)
498 struct pex_update_request
*req
= (struct pex_update_request
*)data
;
499 struct pex_msg_update_send_ctx ctx
= {};
500 uint64_t req_version
= be64_to_cpu(req
->cur_version
);
504 if (len
< sizeof(struct pex_update_request
))
507 if (net
->config
.type
!= NETWORK_TYPE_DYNAMIC
)
511 query_count
= &peer
->state
.num_net_queries
;
513 query_count
= &net
->num_net_queries
;
515 if (++*query_count
> 10)
518 D("receive update request, local version=%"PRIu64
", remote version=%"PRIu64
, net
->net_data_version
, req_version
);
520 if (req_version
>= net
->net_data_version
) {
521 struct pex_update_response_no_data
*res
;
523 pex_msg_init_ext(net
, PEX_MSG_UPDATE_RESPONSE_NO_DATA
, !!addr
);
524 res
= pex_msg_append(sizeof(*res
));
525 res
->req_id
= req
->req_id
;
526 res
->cur_version
= cpu_to_be64(net
->net_data_version
);
527 pex_msg_send_ext(net
, peer
, addr
);
530 if (req_version
> net
->net_data_version
)
531 network_pex_send_update_request(net
, peer
, addr
);
533 if (!peer
|| !net
->net_data_len
)
536 if (req_version
>= net
->net_data_version
)
539 pex_msg_update_response_init(&ctx
, net
->config
.pubkey
, net
->config
.auth_key
,
540 peer
->key
, !!addr
, (void *)data
,
541 net
->net_data
, net
->net_data_len
);
543 pex_msg_send_ext(net
, peer
, addr
);
544 done
= !pex_msg_update_response_continue(&ctx
);
549 network_pex_recv_update_response(struct network
*net
, const uint8_t *data
, size_t len
,
550 struct sockaddr_in6
*addr
, enum pex_opcode op
)
552 struct network_peer
*peer
;
554 int net_data_len
= 0;
555 uint64_t version
= 0;
556 bool no_prev_data
= !net
->net_data_len
;
558 if (net
->config
.type
!= NETWORK_TYPE_DYNAMIC
)
561 net_data
= pex_msg_update_response_recv(data
, len
, op
, &net_data_len
, &version
);
565 if (version
<= net
->net_data_version
) {
570 D_NET(net
, "received updated network data, len=%d", net_data_len
);
573 net
->net_data
= net_data
;
574 net
->net_data_len
= net_data_len
;
575 net
->net_data_version
= version
;
576 if (network_save_dynamic(net
) < 0)
579 uloop_timeout_set(&net
->reload_timer
, no_prev_data
? 1 : UNETD_DATA_UPDATE_DELAY
);
580 vlist_for_each_element(&net
->peers
, peer
, node
) {
581 if (!peer
->state
.connected
)
583 network_pex_send_update_request(net
, peer
, NULL
);
588 network_pex_recv(struct network
*net
, struct network_peer
*peer
, struct pex_hdr
*hdr
)
590 const void *data
= hdr
+ 1;
592 if (hdr
->version
!= 0)
595 D_PEER(net
, peer
, "PEX rx op=%d", hdr
->opcode
);
596 switch (hdr
->opcode
) {
598 network_pex_recv_hello(net
, peer
, data
, hdr
->len
);
600 case PEX_MSG_NOTIFY_PEERS
:
601 network_pex_recv_peers(net
, peer
, data
, hdr
->len
);
604 network_pex_recv_query(net
, peer
, data
, hdr
->len
);
607 network_pex_recv_ping(net
, peer
);
611 case PEX_MSG_UPDATE_REQUEST
:
612 network_pex_recv_update_request(net
, peer
, data
, hdr
->len
,
615 case PEX_MSG_UPDATE_RESPONSE
:
616 case PEX_MSG_UPDATE_RESPONSE_DATA
:
617 case PEX_MSG_UPDATE_RESPONSE_NO_DATA
:
618 network_pex_recv_update_response(net
, data
, hdr
->len
,
621 case PEX_MSG_ENDPOINT_NOTIFY
:
627 network_pex_fd_cb(struct uloop_fd
*fd
, unsigned int events
)
629 struct network
*net
= container_of(fd
, struct network
, pex
.fd
);
630 struct network_peer
*local
= &net
->net_config
.local_host
->peer
;
631 struct network_peer
*peer
;
632 struct sockaddr_in6 sin6
;
633 static char buf
[PEX_BUF_SIZE
];
634 struct pex_hdr
*hdr
= (struct pex_hdr
*)buf
;
638 socklen_t slen
= sizeof(sin6
);
640 len
= recvfrom(fd
->fd
, buf
, sizeof(buf
), 0, (struct sockaddr
*)&sin6
, &slen
);
648 D_NET(net
, "recvfrom failed: %s", strerror(errno
));
649 network_pex_close(net
);
656 if (len
< sizeof(*hdr
))
659 hdr
->len
= ntohs(hdr
->len
);
660 if (len
- sizeof(hdr
) < hdr
->len
)
663 peer
= pex_msg_peer(net
, hdr
->id
);
667 if (memcmp(&sin6
.sin6_addr
, &peer
->local_addr
.in6
, sizeof(sin6
.sin6_addr
)) != 0)
673 network_pex_recv(net
, peer
, hdr
);
677 void network_pex_create_host(struct network
*net
, union network_endpoint
*ep
,
678 unsigned int timeout
)
680 struct network_pex
*pex
= &net
->pex
;
681 struct network_pex_host
*host
;
682 bool new_host
= false;
684 list_for_each_entry(host
, &pex
->hosts
, list
) {
685 if (memcmp(&host
->endpoint
, ep
, sizeof(host
->endpoint
)) != 0)
688 list_move_tail(&host
->list
, &pex
->hosts
);
692 host
= calloc(1, sizeof(*host
));
694 memcpy(&host
->endpoint
, ep
, sizeof(host
->endpoint
));
695 list_add_tail(&host
->list
, &pex
->hosts
);
698 if (timeout
&& (new_host
|| host
->timeout
))
699 host
->timeout
= timeout
+ unet_gettime();
700 network_pex_host_request_update(net
, host
);
704 network_pex_open_auth_connect(struct network
*net
)
706 struct network_pex
*pex
= &net
->pex
;
707 struct network_peer
*peer
;
708 struct blob_attr
*cur
;
711 if (net
->config
.type
!= NETWORK_TYPE_DYNAMIC
)
714 uloop_timeout_set(&pex
->request_update_timer
, 5000);
716 vlist_for_each_element(&net
->peers
, peer
, node
) {
717 union network_endpoint ep
= {};
722 if (network_get_endpoint(&ep
, peer
->endpoint
,
723 UNETD_GLOBAL_PEX_PORT
, 0) < 0)
726 ep
.in
.sin_port
= htons(UNETD_GLOBAL_PEX_PORT
);
727 network_pex_create_host(net
, &ep
, 0);
730 if (!net
->config
.auth_connect
)
733 blobmsg_for_each_attr(cur
, net
->config
.auth_connect
, rem
) {
734 union network_endpoint ep
= {};
736 if (network_get_endpoint(&ep
, blobmsg_get_string(cur
),
737 UNETD_GLOBAL_PEX_PORT
, 0) < 0)
740 network_pex_create_host(net
, &ep
, 0);
745 int network_pex_open(struct network
*net
)
747 struct network_host
*local_host
= net
->net_config
.local_host
;
748 struct network_peer
*local
;
749 struct network_pex
*pex
= &net
->pex
;
750 struct sockaddr_in6 sin6
= {};
754 network_pex_open_auth_connect(net
);
756 if (!local_host
|| !local_host
->peer
.pex_port
)
759 local
= &local_host
->peer
;
760 fd
= socket(PF_INET6
, SOCK_DGRAM
, IPPROTO_UDP
);
764 fcntl(fd
, F_SETFL
, fcntl(fd
, F_GETFL
) | O_NONBLOCK
);
765 fcntl(fd
, F_SETFD
, fcntl(fd
, F_GETFD
) | FD_CLOEXEC
);
767 sin6
.sin6_family
= AF_INET6
;
768 memcpy(&sin6
.sin6_addr
, &local
->local_addr
.in6
,
769 sizeof(local
->local_addr
.in6
));
770 sin6
.sin6_port
= htons(local_host
->peer
.pex_port
);
772 if (bind(fd
, (struct sockaddr
*)&sin6
, sizeof(sin6
)) < 0) {
777 setsockopt(fd
, SOL_SOCKET
, SO_REUSEADDR
, &yes
, sizeof(yes
));
778 setsockopt(fd
, SOL_SOCKET
, SO_REUSEPORT
, &yes
, sizeof(yes
));
780 setsockopt(fd
, SOL_SOCKET
, SO_BINDTODEVICE
,
781 network_name(net
), strlen(network_name(net
)));
785 pex
->fd
.cb
= network_pex_fd_cb
;
786 uloop_fd_add(&pex
->fd
, ULOOP_READ
);
795 void network_pex_close(struct network
*net
)
797 struct network_pex
*pex
= &net
->pex
;
798 struct network_pex_host
*host
, *tmp
;
800 uloop_timeout_cancel(&pex
->request_update_timer
);
801 list_for_each_entry_safe(host
, tmp
, &pex
->hosts
, list
) {
805 list_del(&host
->list
);
812 uloop_fd_delete(&pex
->fd
);
814 network_pex_init(net
);
817 void network_pex_free(struct network
*net
)
819 struct network_pex
*pex
= &net
->pex
;
820 struct network_pex_host
*host
, *tmp
;
822 list_for_each_entry_safe(host
, tmp
, &pex
->hosts
, list
) {
823 list_del(&host
->list
);
828 static struct network
*
829 global_pex_find_network(const uint8_t *id
)
833 avl_for_each_element(&networks
, net
, node
) {
834 if (!memcmp(id
, net
->config
.auth_key
, PEX_ID_LEN
))
842 global_pex_recv(struct pex_hdr
*hdr
, struct sockaddr_in6
*addr
)
844 struct pex_ext_hdr
*ehdr
= (void *)(hdr
+ 1);
845 struct network_peer
*peer
;
847 void *data
= (void *)(ehdr
+ 1);
848 char buf
[INET6_ADDRSTRLEN
];
851 if (hdr
->version
!= 0)
854 net
= global_pex_find_network(ehdr
->auth_id
);
855 if (!net
|| net
->config
.type
!= NETWORK_TYPE_DYNAMIC
)
858 *(uint64_t *)hdr
->id
^= pex_network_hash(net
->config
.auth_key
, ehdr
->nonce
);
860 D("PEX global rx op=%d", hdr
->opcode
);
861 switch (hdr
->opcode
) {
863 case PEX_MSG_NOTIFY_PEERS
:
868 case PEX_MSG_UPDATE_REQUEST
:
869 peer
= pex_msg_peer(net
, hdr
->id
);
870 network_pex_recv_update_request(net
, peer
, data
, hdr
->len
,
873 case PEX_MSG_UPDATE_RESPONSE
:
874 case PEX_MSG_UPDATE_RESPONSE_DATA
:
875 case PEX_MSG_UPDATE_RESPONSE_NO_DATA
:
876 network_pex_recv_update_response(net
, data
, hdr
->len
, addr
, hdr
->opcode
);
878 case PEX_MSG_ENDPOINT_NOTIFY
:
879 peer
= pex_msg_peer(net
, hdr
->id
);
883 if (IN6_IS_ADDR_V4MAPPED(&addr
->sin6_addr
)) {
884 struct sockaddr_in
*sin
= (struct sockaddr_in
*)addr
;
885 struct in_addr in
= *(struct in_addr
*)&addr
->sin6_addr
.s6_addr
[12];
886 int port
= addr
->sin6_port
;
888 memset(addr
, 0, sizeof(*addr
));
889 sin
->sin_port
= port
;
890 sin
->sin_family
= AF_INET
;
894 D_PEER(net
, peer
, "receive endpoint notification from %s",
895 inet_ntop(addr
->sin6_family
, network_endpoint_addr((void *)addr
, &addr_len
),
898 memcpy(&peer
->state
.next_endpoint
, addr
, sizeof(*addr
));
903 int global_pex_open(void)
905 struct sockaddr_in6 sin6
= {};
907 sin6
.sin6_family
= AF_INET6
;
908 sin6
.sin6_port
= htons(global_pex_port
);
910 return pex_open(&sin6
, sizeof(sin6
), global_pex_recv
, true);