blob: fa1828dec6afddbfe204a605eb1cae973e015910 [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,
Adam Snaider96a0f4b2023-05-18 20:41:19 -070025 int streams, std::string_view local_host, int local_port,
Adam Snaider9bb33442023-06-26 16:31:37 -070026 SctpAuthMethod requested_authentication)
27 : sctp_(requested_authentication) {
Austin Schuh0a0a8272021-12-08 13:19:32 -080028 bool use_ipv6 = Ipv6Enabled();
29 sockaddr_local_ = ResolveSocket(local_host, local_port, use_ipv6);
30 sockaddr_remote_ = ResolveSocket(remote_host, remote_port, use_ipv6);
Austin Schuh507f7582021-07-31 20:39:55 -070031 sctp_.OpenSocket(sockaddr_local_);
Austin Schuhe84c3ed2019-12-14 15:29:48 -080032
33 {
34 struct sctp_initmsg initmsg;
35 memset(&initmsg, 0, sizeof(struct sctp_initmsg));
36 initmsg.sinit_num_ostreams = streams;
37 initmsg.sinit_max_instreams = streams;
Austin Schuh3c4af572021-11-11 18:47:33 -080038 // Max timeout in milliseconds for the INIT packet.
39 initmsg.sinit_max_init_timeo = FLAGS_sinit_max_init_timeout;
Austin Schuh507f7582021-07-31 20:39:55 -070040 PCHECK(setsockopt(fd(), IPPROTO_SCTP, SCTP_INITMSG, &initmsg,
Austin Schuhe84c3ed2019-12-14 15:29:48 -080041 sizeof(struct sctp_initmsg)) == 0);
42 }
43
44 {
Austin Schuh8e7034c2021-11-02 20:47:47 -070045 // Turn off the NAGLE algorithm so the timestamps heading back across the
46 // network arrive promptly.
47 int on = 1;
Austin Schuh507f7582021-07-31 20:39:55 -070048 PCHECK(setsockopt(fd(), IPPROTO_SCTP, SCTP_NODELAY, &on, sizeof(int)) == 0);
Austin Schuhe84c3ed2019-12-14 15:29:48 -080049 }
Austin Schuhe84c3ed2019-12-14 15:29:48 -080050}
51
52void SctpClient::LogSctpStatus(sctp_assoc_t assoc_id) {
53 message_bridge::LogSctpStatus(fd(), assoc_id);
54}
55
Sarah Newman1e1b0492023-01-12 14:57:31 -080056void SctpClient::SetPriorityScheduler([[maybe_unused]] sctp_assoc_t assoc_id) {
57// Kernel 4.9 does not have SCTP_SS_PRIO
58#ifdef SCTP_SS_PRIO
Jim Ostrowski5d5a44f2020-06-24 19:10:15 -070059 struct sctp_assoc_value scheduler;
60 memset(&scheduler, 0, sizeof(scheduler));
61 scheduler.assoc_id = assoc_id;
62 scheduler.assoc_value = SCTP_SS_PRIO;
63 if (setsockopt(fd(), IPPROTO_SCTP, SCTP_STREAM_SCHEDULER, &scheduler,
64 sizeof(scheduler)) != 0) {
Sarah Newman29da5ed2022-04-28 18:51:18 -070065 LOG_FIRST_N(WARNING, 1) << "Failed to set scheduler: " << strerror(errno)
66 << " [" << errno << "]";
Jim Ostrowski5d5a44f2020-06-24 19:10:15 -070067 }
Sarah Newman1e1b0492023-01-12 14:57:31 -080068#endif
Austin Schuhe84c3ed2019-12-14 15:29:48 -080069}
70
71} // namespace message_bridge
72} // namespace aos