rpcsx/rpcs3/Emu/NP/ip_address.cpp
2025-02-12 15:00:50 +01:00

250 lines
6.4 KiB
C++

#include "stdafx.h"
#include "ip_address.h"
#include "Utilities/StrFmt.h"
#include "Emu/IdManager.h"
#include "util/endian.hpp"
#include "util/types.hpp"
#include "Emu/NP/rpcn_config.h"
#include <algorithm>
#ifndef _WIN32
#include <unistd.h>
#include <fcntl.h>
#endif
LOG_CHANNEL(IPv6_log, "IPv6_layer");
namespace np
{
bool ip_address_translator::is_ipv6(u32 addr_be)
{
const u32 addr = std::bit_cast<u32, be_t<u32>>(addr_be);
return (addr >= translation_range_begin && addr <= translation_range_end);
}
u32 ip_address_translator::register_ipv6(const std::array<u8, 16>& ipv6_addr)
{
if (ipv6_addr[10] == 0xFF && ipv6_addr[11] == 0xFF &&
std::all_of(ipv6_addr.begin(), ipv6_addr.begin() + 10, [](u8 val)
{
return val == 0;
}))
{
// IPv4 over IPv6
u32 ip_addr{};
std::memcpy(&ip_addr, &ipv6_addr[12], sizeof(u32));
return ip_addr;
}
std::lock_guard lock(mutex);
const auto it = ipv6_to_ipv4.find(ipv6_addr);
if (it == ipv6_to_ipv4.end())
{
ensure(cur_addr != translation_range_end);
const u32 translation_ip = cur_addr++;
const u32 translation_ip_be = std::bit_cast<u32, be_t<u32>>(translation_ip);
ipv6_to_ipv4.emplace(ipv6_addr, translation_ip_be);
ipv4_to_ipv6.push_back(ipv6_addr);
return translation_ip_be;
}
return it->second;
}
sockaddr_in6 ip_address_translator::get_ipv6_sockaddr(u32 addr_be, u16 port_be)
{
const u32 addr_native = std::bit_cast<u32, be_t<u32>>(addr_be);
const auto ipv6_addr = ::at32(ipv4_to_ipv6, addr_native - translation_range_begin);
sockaddr_in6 sockaddr_ipv6{.sin6_family = AF_INET6, .sin6_port = port_be};
std::memcpy(&sockaddr_ipv6.sin6_addr, ipv6_addr.data(), ipv6_addr.size());
return sockaddr_ipv6;
}
u32 register_ip(const flatbuffers::Vector<std::uint8_t>* vec)
{
if (vec->size() == 4)
{
const u32 ip = static_cast<u32>(vec->Get(0)) << 24 | static_cast<u32>(vec->Get(1)) << 16 |
static_cast<u32>(vec->Get(2)) << 8 | static_cast<u32>(vec->Get(3));
u32 result_ip = std::bit_cast<u32, be_t<u32>>(ip);
return result_ip;
}
else if (vec->size() == 16)
{
std::array<u8, 16> ipv6_addr{};
std::memcpy(ipv6_addr.data(), vec->Data(), 16);
auto& translator = g_fxo->get<np::ip_address_translator>();
return translator.register_ipv6(ipv6_addr);
}
else
{
fmt::throw_exception("Received ip address with size = %d", vec->size());
}
}
bool is_ipv6_supported(std::optional<IPV6_SUPPORT> force_state)
{
static atomic_t<IPV6_SUPPORT> ipv6_status = IPV6_SUPPORT::IPV6_UNKNOWN;
if (force_state)
ipv6_status = *force_state;
if (ipv6_status != IPV6_SUPPORT::IPV6_UNKNOWN)
return ipv6_status == IPV6_SUPPORT::IPV6_SUPPORTED;
static shared_mutex mtx;
std::lock_guard lock(mtx);
if (ipv6_status != IPV6_SUPPORT::IPV6_UNKNOWN)
return ipv6_status == IPV6_SUPPORT::IPV6_SUPPORTED;
// IPv6 feature is only used by RPCN
if (!g_cfg_rpcn.get_ipv6_support())
{
IPv6_log.notice("is_ipv6_supported(): disabled through config");
ipv6_status = IPV6_SUPPORT::IPV6_UNSUPPORTED;
return false;
}
// We try to connect to ipv6.google.com:8080
addrinfo* addr_info{};
socket_type socket_ipv6{};
auto cleanup = [&]()
{
if (socket_ipv6)
close_socket(socket_ipv6);
if (addr_info)
freeaddrinfo(addr_info);
};
auto error_and_disable = [&](const char* message) -> bool
{
IPv6_log.error("is_ipv6_supported(): %s", message);
ipv6_status = IPV6_SUPPORT::IPV6_UNSUPPORTED;
cleanup();
return false;
};
addrinfo hints{.ai_family = AF_INET6};
if (getaddrinfo("ipv6.google.com", nullptr, &hints, &addr_info))
return error_and_disable("Failed to resolve ipv6.google.com!");
addrinfo* found = addr_info;
while (found != nullptr)
{
if (found->ai_family == AF_INET6)
break;
found = found->ai_next;
}
if (found == nullptr)
return error_and_disable("Failed to find IPv6 for ipv6.google.com");
socket_type socket_or_err = ::socket(AF_INET6, SOCK_STREAM, 0);
#ifdef _WIN32
if (socket_or_err == INVALID_SOCKET)
#else
if (socket_or_err == -1)
#endif
return error_and_disable("Failed to create IPv6 socket!");
socket_ipv6 = socket_or_err;
sockaddr_in6 ipv6_addr = *reinterpret_cast<const sockaddr_in6*>(found->ai_addr);
ipv6_addr.sin6_port = std::bit_cast<u16, be_t<u16>>(443);
if (::connect(socket_ipv6, reinterpret_cast<const sockaddr*>(&ipv6_addr), sizeof(ipv6_addr)) != 0)
return error_and_disable("Failed to connect to ipv6.google.com");
cleanup();
IPv6_log.success("Successfully tested IPv6 support!");
ipv6_status = IPV6_SUPPORT::IPV6_SUPPORTED;
return true;
}
s32 sendto_possibly_ipv6(socket_type native_socket, const char* data, u32 size, const sockaddr_in* addr, int native_flags)
{
if (is_ipv6_supported())
{
sockaddr_in6 addr_ipv6{};
if (np::ip_address_translator::is_ipv6(addr->sin_addr.s_addr))
{
auto& translator = g_fxo->get<np::ip_address_translator>();
addr_ipv6 = translator.get_ipv6_sockaddr(addr->sin_addr.s_addr, addr->sin_port);
}
else
{
addr_ipv6 = sockaddr_to_sockaddr6(*addr);
}
return ::sendto(native_socket, data, size, native_flags, reinterpret_cast<const sockaddr*>(&addr_ipv6), sizeof(sockaddr_in6));
}
return ::sendto(native_socket, data, size, native_flags, reinterpret_cast<const sockaddr*>(addr), sizeof(sockaddr_in));
}
sockaddr_in6 sockaddr_to_sockaddr6(const sockaddr_in& addr)
{
sockaddr_in6 addr_ipv6{.sin6_family = AF_INET6, .sin6_port = addr.sin_port};
std::array<u8, 16> ipv6{};
// IPv4 over IPv6
ipv6[10] = 0xFF;
ipv6[11] = 0xFF;
std::memcpy(ipv6.data() + 12, &addr.sin_addr.s_addr, 4);
std::memcpy(&addr_ipv6.sin6_addr, ipv6.data(), ipv6.size());
return addr_ipv6;
}
sockaddr_in sockaddr6_to_sockaddr(const sockaddr_in6& addr)
{
sockaddr_in addr_ipv4{.sin_family = AF_INET, .sin_port = addr.sin6_port};
std::array<u8, 16> ipv6{};
std::memcpy(ipv6.data(), &addr.sin6_addr, 16);
auto& translator = g_fxo->get<np::ip_address_translator>();
addr_ipv4.sin_addr.s_addr = translator.register_ipv6(ipv6);
return addr_ipv4;
}
void close_socket(socket_type socket)
{
if (socket)
{
#ifdef _WIN32
::closesocket(socket);
#else
::close(socket);
#endif
}
}
void set_socket_non_blocking(socket_type socket)
{
if (socket)
{
#ifdef _WIN32
u_long _true = 1;
::ioctlsocket(socket, FIONBIO, &_true);
#else
::fcntl(socket, F_SETFL, ::fcntl(socket, F_GETFL, 0) | O_NONBLOCK);
#endif
}
}
} // namespace np