blob: a87dfc621401bf2cfeb8cf3b795b26bbddcd2020 [file] [log] [blame]
Austin Schuhe84c3ed2019-12-14 15:29:48 -08001#include "aos/network/sctp_client.h"
2
3#include <arpa/inet.h>
Adam Snaiderbe263512023-05-18 20:40:23 -07004#include <linux/sctp.h>
Austin Schuhe84c3ed2019-12-14 15:29:48 -08005#include <net/if.h>
Austin Schuhe84c3ed2019-12-14 15:29:48 -08006#include <sys/socket.h>
Tyler Chatowbf0609c2021-07-31 16:13:27 -07007
8#include <cstdlib>
9#include <cstring>
Austin Schuhe84c3ed2019-12-14 15:29:48 -080010#include <string_view>
11
Philipp Schrader790cb542023-07-05 21:06:52 -070012#include "glog/logging.h"
13
Austin Schuhe84c3ed2019-12-14 15:29:48 -080014#include "aos/network/sctp_lib.h"
15#include "aos/unique_malloc_ptr.h"
Austin Schuhe84c3ed2019-12-14 15:29:48 -080016
Austin Schuh3c4af572021-11-11 18:47:33 -080017DEFINE_int32(sinit_max_init_timeout, 0,
18 "Timeout in milliseconds for retrying the INIT packet when "
19 "connecting to the message bridge server");
20
Austin Schuhe84c3ed2019-12-14 15:29:48 -080021namespace aos {
22namespace message_bridge {
23
24SctpClient::SctpClient(std::string_view remote_host, int remote_port,
Austin Schuh0a0a8272021-12-08 13:19:32 -080025 int streams, std::string_view local_host,
26 int local_port) {
27 bool use_ipv6 = Ipv6Enabled();
28 sockaddr_local_ = ResolveSocket(local_host, local_port, use_ipv6);
29 sockaddr_remote_ = ResolveSocket(remote_host, remote_port, use_ipv6);
Austin Schuh507f7582021-07-31 20:39:55 -070030 sctp_.OpenSocket(sockaddr_local_);
Austin Schuhe84c3ed2019-12-14 15:29:48 -080031
32 {
33 struct sctp_initmsg initmsg;
34 memset(&initmsg, 0, sizeof(struct sctp_initmsg));
35 initmsg.sinit_num_ostreams = streams;
36 initmsg.sinit_max_instreams = streams;
Austin Schuh3c4af572021-11-11 18:47:33 -080037 // Max timeout in milliseconds for the INIT packet.
38 initmsg.sinit_max_init_timeo = FLAGS_sinit_max_init_timeout;
Austin Schuh507f7582021-07-31 20:39:55 -070039 PCHECK(setsockopt(fd(), IPPROTO_SCTP, SCTP_INITMSG, &initmsg,
Austin Schuhe84c3ed2019-12-14 15:29:48 -080040 sizeof(struct sctp_initmsg)) == 0);
41 }
42
43 {
Austin Schuh8e7034c2021-11-02 20:47:47 -070044 // Turn off the NAGLE algorithm so the timestamps heading back across the
45 // network arrive promptly.
46 int on = 1;
Austin Schuh507f7582021-07-31 20:39:55 -070047 PCHECK(setsockopt(fd(), IPPROTO_SCTP, SCTP_NODELAY, &on, sizeof(int)) == 0);
Austin Schuhe84c3ed2019-12-14 15:29:48 -080048 }
Austin Schuhe84c3ed2019-12-14 15:29:48 -080049}
50
51void SctpClient::LogSctpStatus(sctp_assoc_t assoc_id) {
52 message_bridge::LogSctpStatus(fd(), assoc_id);
53}
54
Sarah Newman1e1b0492023-01-12 14:57:31 -080055void SctpClient::SetPriorityScheduler([[maybe_unused]] sctp_assoc_t assoc_id) {
56// Kernel 4.9 does not have SCTP_SS_PRIO
57#ifdef SCTP_SS_PRIO
Jim Ostrowski5d5a44f2020-06-24 19:10:15 -070058 struct sctp_assoc_value scheduler;
59 memset(&scheduler, 0, sizeof(scheduler));
60 scheduler.assoc_id = assoc_id;
61 scheduler.assoc_value = SCTP_SS_PRIO;
62 if (setsockopt(fd(), IPPROTO_SCTP, SCTP_STREAM_SCHEDULER, &scheduler,
63 sizeof(scheduler)) != 0) {
Sarah Newman29da5ed2022-04-28 18:51:18 -070064 LOG_FIRST_N(WARNING, 1) << "Failed to set scheduler: " << strerror(errno)
65 << " [" << errno << "]";
Jim Ostrowski5d5a44f2020-06-24 19:10:15 -070066 }
Sarah Newman1e1b0492023-01-12 14:57:31 -080067#endif
Austin Schuhe84c3ed2019-12-14 15:29:48 -080068}
69
70} // namespace message_bridge
71} // namespace aos