mirror of
https://github.com/esphome/esphome.git
synced 2026-01-20 18:09:10 -07:00
176 lines
5.7 KiB
C++
176 lines
5.7 KiB
C++
#include "socket.h"
|
|
#if defined(USE_SOCKET_IMPL_LWIP_TCP) || defined(USE_SOCKET_IMPL_LWIP_SOCKETS) || defined(USE_SOCKET_IMPL_BSD_SOCKETS)
|
|
#include <cerrno>
|
|
#include <cstring>
|
|
#include <string>
|
|
#include "esphome/core/log.h"
|
|
#include "esphome/core/application.h"
|
|
|
|
namespace esphome::socket {
|
|
|
|
Socket::~Socket() {}
|
|
|
|
// Platform-specific inet_ntop wrappers
|
|
#if defined(USE_SOCKET_IMPL_LWIP_TCP)
|
|
// LWIP raw TCP (ESP8266) uses inet_ntoa_r which takes struct by value
|
|
static inline const char *esphome_inet_ntop4(const void *addr, char *buf, size_t size) {
|
|
inet_ntoa_r(*reinterpret_cast<const struct in_addr *>(addr), buf, size);
|
|
return buf;
|
|
}
|
|
#if USE_NETWORK_IPV6
|
|
static inline const char *esphome_inet_ntop6(const void *addr, char *buf, size_t size) {
|
|
inet6_ntoa_r(*reinterpret_cast<const ip6_addr_t *>(addr), buf, size);
|
|
return buf;
|
|
}
|
|
#endif
|
|
#elif defined(USE_SOCKET_IMPL_LWIP_SOCKETS)
|
|
// LWIP sockets (LibreTiny, ESP32 Arduino)
|
|
static inline const char *esphome_inet_ntop4(const void *addr, char *buf, size_t size) {
|
|
return lwip_inet_ntop(AF_INET, addr, buf, size);
|
|
}
|
|
#if USE_NETWORK_IPV6
|
|
static inline const char *esphome_inet_ntop6(const void *addr, char *buf, size_t size) {
|
|
return lwip_inet_ntop(AF_INET6, addr, buf, size);
|
|
}
|
|
#endif
|
|
#else
|
|
// BSD sockets (host, ESP32-IDF)
|
|
static inline const char *esphome_inet_ntop4(const void *addr, char *buf, size_t size) {
|
|
return inet_ntop(AF_INET, addr, buf, size);
|
|
}
|
|
#if USE_NETWORK_IPV6
|
|
static inline const char *esphome_inet_ntop6(const void *addr, char *buf, size_t size) {
|
|
return inet_ntop(AF_INET6, addr, buf, size);
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
// Format sockaddr into caller-provided buffer, returns length written (excluding null)
|
|
static size_t format_sockaddr_to(const struct sockaddr_storage &storage, std::span<char, SOCKADDR_STR_LEN> buf) {
|
|
if (storage.ss_family == AF_INET) {
|
|
const auto *addr = reinterpret_cast<const struct sockaddr_in *>(&storage);
|
|
if (esphome_inet_ntop4(&addr->sin_addr, buf.data(), buf.size()) != nullptr)
|
|
return strlen(buf.data());
|
|
}
|
|
#if USE_NETWORK_IPV6
|
|
else if (storage.ss_family == AF_INET6) {
|
|
const auto *addr = reinterpret_cast<const struct sockaddr_in6 *>(&storage);
|
|
#ifndef USE_SOCKET_IMPL_LWIP_TCP
|
|
// Format IPv4-mapped IPv6 addresses as regular IPv4 (not supported on ESP8266 raw TCP)
|
|
if (addr->sin6_addr.un.u32_addr[0] == 0 && addr->sin6_addr.un.u32_addr[1] == 0 &&
|
|
addr->sin6_addr.un.u32_addr[2] == htonl(0xFFFF) &&
|
|
esphome_inet_ntop4(&addr->sin6_addr.un.u32_addr[3], buf.data(), buf.size()) != nullptr) {
|
|
return strlen(buf.data());
|
|
}
|
|
#endif
|
|
if (esphome_inet_ntop6(&addr->sin6_addr, buf.data(), buf.size()) != nullptr)
|
|
return strlen(buf.data());
|
|
}
|
|
#endif
|
|
buf[0] = '\0';
|
|
return 0;
|
|
}
|
|
|
|
size_t Socket::getpeername_to(std::span<char, SOCKADDR_STR_LEN> buf) {
|
|
struct sockaddr_storage storage;
|
|
socklen_t len = sizeof(storage);
|
|
if (this->getpeername(reinterpret_cast<struct sockaddr *>(&storage), &len) != 0) {
|
|
buf[0] = '\0';
|
|
return 0;
|
|
}
|
|
return format_sockaddr_to(storage, buf);
|
|
}
|
|
|
|
size_t Socket::getsockname_to(std::span<char, SOCKADDR_STR_LEN> buf) {
|
|
struct sockaddr_storage storage;
|
|
socklen_t len = sizeof(storage);
|
|
if (this->getsockname(reinterpret_cast<struct sockaddr *>(&storage), &len) != 0) {
|
|
buf[0] = '\0';
|
|
return 0;
|
|
}
|
|
return format_sockaddr_to(storage, buf);
|
|
}
|
|
|
|
std::unique_ptr<Socket> socket_ip(int type, int protocol) {
|
|
#if USE_NETWORK_IPV6
|
|
return socket(AF_INET6, type, protocol);
|
|
#else
|
|
return socket(AF_INET, type, protocol);
|
|
#endif /* USE_NETWORK_IPV6 */
|
|
}
|
|
|
|
std::unique_ptr<Socket> socket_ip_loop_monitored(int type, int protocol) {
|
|
#if USE_NETWORK_IPV6
|
|
return socket_loop_monitored(AF_INET6, type, protocol);
|
|
#else
|
|
return socket_loop_monitored(AF_INET, type, protocol);
|
|
#endif /* USE_NETWORK_IPV6 */
|
|
}
|
|
|
|
socklen_t set_sockaddr(struct sockaddr *addr, socklen_t addrlen, const char *ip_address, uint16_t port) {
|
|
#if USE_NETWORK_IPV6
|
|
if (strchr(ip_address, ':') != nullptr) {
|
|
if (addrlen < sizeof(sockaddr_in6)) {
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
auto *server = reinterpret_cast<sockaddr_in6 *>(addr);
|
|
memset(server, 0, sizeof(sockaddr_in6));
|
|
server->sin6_family = AF_INET6;
|
|
server->sin6_port = htons(port);
|
|
|
|
#ifdef USE_SOCKET_IMPL_BSD_SOCKETS
|
|
// Use standard inet_pton for BSD sockets
|
|
if (inet_pton(AF_INET6, ip_address, &server->sin6_addr) != 1) {
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
#else
|
|
// Use LWIP-specific functions
|
|
ip6_addr_t ip6;
|
|
inet6_aton(ip_address, &ip6);
|
|
memcpy(server->sin6_addr.un.u32_addr, ip6.addr, sizeof(ip6.addr));
|
|
#endif
|
|
return sizeof(sockaddr_in6);
|
|
}
|
|
#endif /* USE_NETWORK_IPV6 */
|
|
if (addrlen < sizeof(sockaddr_in)) {
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
auto *server = reinterpret_cast<sockaddr_in *>(addr);
|
|
memset(server, 0, sizeof(sockaddr_in));
|
|
server->sin_family = AF_INET;
|
|
server->sin_addr.s_addr = inet_addr(ip_address);
|
|
server->sin_port = htons(port);
|
|
return sizeof(sockaddr_in);
|
|
}
|
|
|
|
socklen_t set_sockaddr_any(struct sockaddr *addr, socklen_t addrlen, uint16_t port) {
|
|
#if USE_NETWORK_IPV6
|
|
if (addrlen < sizeof(sockaddr_in6)) {
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
auto *server = reinterpret_cast<sockaddr_in6 *>(addr);
|
|
memset(server, 0, sizeof(sockaddr_in6));
|
|
server->sin6_family = AF_INET6;
|
|
server->sin6_port = htons(port);
|
|
server->sin6_addr = IN6ADDR_ANY_INIT;
|
|
return sizeof(sockaddr_in6);
|
|
#else
|
|
if (addrlen < sizeof(sockaddr_in)) {
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
auto *server = reinterpret_cast<sockaddr_in *>(addr);
|
|
memset(server, 0, sizeof(sockaddr_in));
|
|
server->sin_family = AF_INET;
|
|
server->sin_addr.s_addr = ESPHOME_INADDR_ANY;
|
|
server->sin_port = htons(port);
|
|
return sizeof(sockaddr_in);
|
|
#endif /* USE_NETWORK_IPV6 */
|
|
}
|
|
} // namespace esphome::socket
|
|
#endif
|