ESPHome 2025.12.5
Loading...
Searching...
No Matches
lwip_sockets_impl.cpp
Go to the documentation of this file.
1#include "socket.h"
4
5#ifdef USE_SOCKET_IMPL_LWIP_SOCKETS
6
7#include <cstring>
9
10namespace esphome {
11namespace socket {
12
13std::string format_sockaddr(const struct sockaddr_storage &storage) {
14 if (storage.ss_family == AF_INET) {
15 const struct sockaddr_in *addr = reinterpret_cast<const struct sockaddr_in *>(&storage);
16 char buf[INET_ADDRSTRLEN];
17 const char *ret = lwip_inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf));
18 if (ret == nullptr)
19 return {};
20 return std::string{buf};
21 }
22#if LWIP_IPV6
23 else if (storage.ss_family == AF_INET6) {
24 const struct sockaddr_in6 *addr = reinterpret_cast<const struct sockaddr_in6 *>(&storage);
25 char buf[INET6_ADDRSTRLEN];
26 const char *ret = lwip_inet_ntop(AF_INET6, &addr->sin6_addr, buf, sizeof(buf));
27 if (ret == nullptr)
28 return {};
29 return std::string{buf};
30 }
31#endif
32 return {};
33}
34
35class LwIPSocketImpl : public Socket {
36 public:
37 LwIPSocketImpl(int fd, bool monitor_loop = false) : fd_(fd) {
38#ifdef USE_SOCKET_SELECT_SUPPORT
39 // Register new socket with the application for select() if monitoring requested
40 if (monitor_loop && fd_ >= 0) {
41 // Only set loop_monitored_ to true if registration succeeds
43 } else {
44 loop_monitored_ = false;
45 }
46#else
47 // Without select support, ignore monitor_loop parameter
48 (void) monitor_loop;
49#endif
50 }
51 ~LwIPSocketImpl() override {
52 if (!closed_) {
53 close(); // NOLINT(clang-analyzer-optin.cplusplus.VirtualCall)
54 }
55 }
56 int connect(const struct sockaddr *addr, socklen_t addrlen) override { return lwip_connect(fd_, addr, addrlen); }
57 std::unique_ptr<Socket> accept(struct sockaddr *addr, socklen_t *addrlen) override {
58 return accept_impl_(addr, addrlen, false);
59 }
60 std::unique_ptr<Socket> accept_loop_monitored(struct sockaddr *addr, socklen_t *addrlen) override {
61 return accept_impl_(addr, addrlen, true);
62 }
63
64 private:
65 std::unique_ptr<Socket> accept_impl_(struct sockaddr *addr, socklen_t *addrlen, bool loop_monitored) {
66 int fd = lwip_accept(fd_, addr, addrlen);
67 if (fd == -1)
68 return {};
69 return make_unique<LwIPSocketImpl>(fd, loop_monitored);
70 }
71
72 public:
73 int bind(const struct sockaddr *addr, socklen_t addrlen) override { return lwip_bind(fd_, addr, addrlen); }
74 int close() override {
75 if (!closed_) {
76#ifdef USE_SOCKET_SELECT_SUPPORT
77 // Unregister from select() before closing if monitored
78 if (loop_monitored_) {
80 }
81#endif
82 int ret = lwip_close(fd_);
83 closed_ = true;
84 return ret;
85 }
86 return 0;
87 }
88 int shutdown(int how) override { return lwip_shutdown(fd_, how); }
89
90 int getpeername(struct sockaddr *addr, socklen_t *addrlen) override { return lwip_getpeername(fd_, addr, addrlen); }
91 std::string getpeername() override {
92 struct sockaddr_storage storage;
93 socklen_t len = sizeof(storage);
94 int err = this->getpeername((struct sockaddr *) &storage, &len);
95 if (err != 0)
96 return {};
97 return format_sockaddr(storage);
98 }
99 int getsockname(struct sockaddr *addr, socklen_t *addrlen) override { return lwip_getsockname(fd_, addr, addrlen); }
100 std::string getsockname() override {
101 struct sockaddr_storage storage;
102 socklen_t len = sizeof(storage);
103 int err = this->getsockname((struct sockaddr *) &storage, &len);
104 if (err != 0)
105 return {};
106 return format_sockaddr(storage);
107 }
108 int getsockopt(int level, int optname, void *optval, socklen_t *optlen) override {
109 return lwip_getsockopt(fd_, level, optname, optval, optlen);
110 }
111 int setsockopt(int level, int optname, const void *optval, socklen_t optlen) override {
112 return lwip_setsockopt(fd_, level, optname, optval, optlen);
113 }
114 int listen(int backlog) override { return lwip_listen(fd_, backlog); }
115 ssize_t read(void *buf, size_t len) override { return lwip_read(fd_, buf, len); }
116 ssize_t recvfrom(void *buf, size_t len, sockaddr *addr, socklen_t *addr_len) override {
117 return lwip_recvfrom(fd_, buf, len, 0, addr, addr_len);
118 }
119 ssize_t readv(const struct iovec *iov, int iovcnt) override { return lwip_readv(fd_, iov, iovcnt); }
120 ssize_t write(const void *buf, size_t len) override { return lwip_write(fd_, buf, len); }
121 ssize_t send(void *buf, size_t len, int flags) { return lwip_send(fd_, buf, len, flags); }
122 ssize_t writev(const struct iovec *iov, int iovcnt) override { return lwip_writev(fd_, iov, iovcnt); }
123 ssize_t sendto(const void *buf, size_t len, int flags, const struct sockaddr *to, socklen_t tolen) override {
124 return lwip_sendto(fd_, buf, len, flags, to, tolen);
125 }
126 int setblocking(bool blocking) override {
127 int fl = lwip_fcntl(fd_, F_GETFL, 0);
128 if (blocking) {
129 fl &= ~O_NONBLOCK;
130 } else {
131 fl |= O_NONBLOCK;
132 }
133 lwip_fcntl(fd_, F_SETFL, fl);
134 return 0;
135 }
136
137 int get_fd() const override { return fd_; }
138
139 protected:
140 int fd_;
141 bool closed_ = false;
142};
143
144// Helper to create a socket with optional monitoring
145static std::unique_ptr<Socket> create_socket(int domain, int type, int protocol, bool loop_monitored = false) {
146 int ret = lwip_socket(domain, type, protocol);
147 if (ret == -1)
148 return nullptr;
149 return std::unique_ptr<Socket>{new LwIPSocketImpl(ret, loop_monitored)};
150}
151
152std::unique_ptr<Socket> socket(int domain, int type, int protocol) {
153 return create_socket(domain, type, protocol, false);
154}
155
156std::unique_ptr<Socket> socket_loop_monitored(int domain, int type, int protocol) {
157 return create_socket(domain, type, protocol, true);
158}
159
160} // namespace socket
161} // namespace esphome
162
163#endif // USE_SOCKET_IMPL_LWIP_SOCKETS
void unregister_socket_fd(int fd)
bool register_socket_fd(int fd)
Register/unregister a socket file descriptor to be monitored for read events.
bool loop_monitored_
Whether this socket is monitored by the event loop.
Definition socket.h:61
uint16_t type
uint16_t flags
uint16_t addr_len
uint32_t socklen_t
Definition headers.h:97
__int64 ssize_t
Definition httplib.h:178
std::string format_sockaddr(const struct sockaddr_storage &storage)
std::unique_ptr< Socket > socket(int domain, int type, int protocol)
Create a socket of the given domain, type and protocol.
std::unique_ptr< Socket > socket_loop_monitored(int domain, int type, int protocol)
Create a socket and monitor it for data in the main loop.
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:503
Application App
Global storage of Application pointer - only one Application can exist.
struct in6_addr sin6_addr
Definition headers.h:77
struct in_addr sin_addr
Definition headers.h:65
sa_family_t ss_family
Definition headers.h:92