You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
181 lines
4.5 KiB
181 lines
4.5 KiB
#include "dvlnet/udp_p2p.h" |
|
|
|
#include <SDL.h> |
|
|
|
#ifdef USE_SDL1 |
|
#include "sdl2_to_1_2_backports.h" |
|
#endif |
|
|
|
namespace dvl { |
|
namespace net { |
|
|
|
const udp_p2p::endpoint udp_p2p::none; |
|
|
|
int udp_p2p::create(std::string addrstr, std::string passwd) |
|
{ |
|
sock = asio::ip::udp::socket(io_context); // to be removed later |
|
setup_password(passwd); |
|
auto ipaddr = asio::ip::make_address(addrstr); |
|
if (ipaddr.is_v4()) |
|
sock.open(asio::ip::udp::v4()); |
|
else if (ipaddr.is_v6()) |
|
sock.open(asio::ip::udp::v6()); |
|
sock.non_blocking(true); |
|
unsigned short port = default_port; |
|
/* |
|
while (port <= default_port+try_ports) { |
|
try { |
|
sock.bind(asio::ip::udp::endpoint(asio::ip::address_v6(), port)); |
|
} catch (std::exception &e) { |
|
SDL_Log("bind: %s, %s", asio::ip::address_v6().to_string(), |
|
e.what()); |
|
} |
|
++port; |
|
} |
|
*/ |
|
try { |
|
sock.bind(endpoint(ipaddr, port)); |
|
} catch (std::exception &e) { |
|
SDL_SetError(e.what()); |
|
return -1; |
|
} |
|
plr_self = 0; |
|
return plr_self; |
|
} |
|
|
|
int udp_p2p::join(std::string addrstr, std::string passwd) |
|
{ |
|
sock = asio::ip::udp::socket(io_context); // to be removed later |
|
setup_password(passwd); |
|
auto ipaddr = asio::ip::make_address(addrstr); |
|
if (ipaddr.is_v4()) |
|
sock.open(asio::ip::udp::v4()); |
|
else if (ipaddr.is_v6()) |
|
sock.open(asio::ip::udp::v6()); |
|
sock.non_blocking(true); |
|
endpoint themaster(ipaddr, default_port); |
|
sock.connect(themaster); |
|
master = themaster; |
|
{ // hack: try to join for 5 seconds |
|
randombytes_buf(reinterpret_cast<unsigned char *>(&cookie_self), |
|
sizeof(cookie_t)); |
|
auto pkt = pktfty->make_packet<PT_JOIN_REQUEST>(PLR_BROADCAST, |
|
PLR_MASTER, cookie_self, |
|
game_init_info); |
|
send(*pkt); |
|
for (auto i = 0; i < 5; ++i) { |
|
recv(); |
|
if (plr_self != PLR_BROADCAST) |
|
break; // join successful |
|
SDL_Delay(1000); |
|
} |
|
} |
|
return (plr_self == PLR_BROADCAST ? MAX_PLRS : plr_self); |
|
} |
|
|
|
void udp_p2p::poll() |
|
{ |
|
recv(); |
|
} |
|
|
|
void udp_p2p::send(packet &pkt) |
|
{ |
|
send_internal(pkt, none); |
|
} |
|
|
|
void udp_p2p::recv() |
|
{ |
|
try { |
|
while (1) { // read until kernel buffer is empty? |
|
try { |
|
endpoint sender; |
|
buffer_t pkt_buf(packet_factory::max_packet_size); |
|
size_t pkt_len; |
|
pkt_len = sock.receive_from(asio::buffer(pkt_buf), sender); |
|
pkt_buf.resize(pkt_len); |
|
auto pkt = pktfty->make_packet(pkt_buf); |
|
recv_decrypted(*pkt, sender); |
|
} catch (packet_exception &e) { |
|
SDL_Log(e.what()); |
|
// drop packet |
|
} |
|
} |
|
} catch (std::exception &e) { |
|
SDL_Log(e.what()); |
|
return; |
|
} |
|
} |
|
|
|
void udp_p2p::send_internal(packet &pkt, endpoint sender) |
|
{ |
|
for (auto &dest : dests_for_addr(pkt.dest(), sender)) { |
|
sock.send_to(asio::buffer(pkt.data()), dest); |
|
} |
|
} |
|
|
|
std::set<udp_p2p::endpoint> udp_p2p::dests_for_addr(plr_t dest, endpoint sender) |
|
{ |
|
auto ret = std::set<endpoint>(); |
|
if (dest == plr_self) |
|
return ret; |
|
|
|
if (dest < MAX_PLRS) { |
|
if (connected_table[dest]) |
|
ret.insert(nexthop_table[dest]); |
|
} else if (dest == PLR_BROADCAST) { |
|
for (auto i = 0; i < MAX_PLRS; ++i) |
|
if (i != plr_self && connected_table[i]) |
|
ret.insert(nexthop_table[i]); |
|
ret.insert(connection_requests_pending.begin(), |
|
connection_requests_pending.end()); |
|
} else if (dest == PLR_MASTER) { |
|
if (master != none) |
|
ret.insert(master); |
|
} |
|
ret.erase(sender); |
|
return ret; |
|
} |
|
|
|
void udp_p2p::handle_join_request(packet &pkt, endpoint sender) |
|
{ |
|
plr_t i; |
|
for (i = 0; i < MAX_PLRS; ++i) { |
|
if (i != plr_self && nexthop_table[i] == none) { |
|
nexthop_table[i] = sender; |
|
break; |
|
} |
|
} |
|
auto reply = pktfty->make_packet<PT_JOIN_ACCEPT>(plr_self, PLR_BROADCAST, |
|
pkt.cookie(), i, |
|
game_init_info); |
|
send(*reply); |
|
} |
|
|
|
void udp_p2p::recv_decrypted(packet &pkt, endpoint sender) |
|
{ |
|
// 1. route |
|
send_internal(pkt, sender); |
|
// 2. handle local |
|
if (pkt.src() == PLR_BROADCAST && pkt.dest() == PLR_MASTER) { |
|
connection_requests_pending.insert(sender); |
|
if (master == none) { |
|
handle_join_request(pkt, sender); |
|
} |
|
} |
|
// normal packets |
|
if (pkt.src() >= MAX_PLRS) |
|
return; //drop packet |
|
if (connected_table[pkt.src()]) { //WRONG?!? |
|
if (sender != nexthop_table[pkt.src()]) |
|
return; //rpfilter fail: drop packet |
|
} else { |
|
nexthop_table[pkt.src()] = sender; // new connection: accept |
|
} |
|
connected_table[pkt.src()] = true; |
|
if (pkt.dest() != plr_self && pkt.dest() != PLR_BROADCAST) |
|
return; //packet not for us, drop |
|
recv_local(pkt); |
|
} |
|
|
|
} // namespace net |
|
} // namespace dvl
|
|
|