// Copyright (C) 2003-2009 Dolphin Project. // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, version 2.0. // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License 2.0 for more details. // A copy of the GPL 2.0 should have been included with the program. // If not, see http://www.gnu.org/licenses/ // Official SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ #include "NetWindow.h" //-------------------------------- // GUI EVENTS //-------------------------------- void NetEvent::AppendText(const wxString text) { // I have the feeling SendEvent may be a bit better... #if 0 SendEvent(ADD_TEXT, std::string(text.mb_str())) #else wxMutexGuiEnter(); m_netptr->AppendText(text); wxMutexGuiLeave(); #endif } void NetEvent::SendEvent(int EventType, const std::string text, int integer) { wxCommandEvent event(wxEVT_HOST_COMMAND, wxID_ANY); event.SetId( EventType ); event.SetInt( integer ); event.SetString( wxString::FromAscii(text.c_str()) ); m_netptr->AddPendingEvent(event); } //-------------------------------- // SERVER SIDE THREAD //-------------------------------- ServerSide::ServerSide(NetPlay* netptr, sf::SocketTCP socket, int netmodel, std::string nick) : wxThread() { m_numplayers = 0; m_data_received = false; m_netmodel = netmodel; m_socket = socket; m_netptr = netptr; m_nick = nick; Event = new NetEvent(m_netptr); } char ServerSide::GetSocket(sf::SocketTCP Socket) { for (int i=0; i < m_numplayers; i++) { if(m_client[i].socket == Socket) return i; } return 0xE; } void *ServerSide::Entry() { // Add listening socket m_selector.Add(m_socket); while (1) { char nbSocketReady = m_selector.Wait(0.5); for (char i = 0; i < nbSocketReady; ++i) { m_CriticalSection.Enter(); sf::SocketTCP Socket = m_selector.GetSocketReady(i); if (Socket == m_socket) { // Incoming connection Event->AppendText(_("*Connection Request... ")); sf::SocketTCP Incoming; sf::IPAddress Address; m_socket.Accept(Incoming, &Address); unsigned char sent = 0x12; if ((m_netmodel == 0 && m_numplayers > 0) || m_numplayers == 3) { Incoming.Send((const char *)&sent, 1); // Tell it the server is full... Incoming.Close(); // Then close the connection Event->AppendText(_(" Server is Full !\n")); } else { Event->AppendText(_(" Connection accepted\n")); m_client[m_numplayers].socket = Incoming; SyncValues(m_numplayers, Address); } // Add it to the selector m_selector.Add(Incoming); Event->SendEvent(HOST_NEWPLAYER); m_numplayers++; } else { unsigned char recv; int socket_nb; size_t recv_size; sf::Socket::Status recv_status; if ((socket_nb = GetSocket(Socket)) == 0xE) PanicAlert("ERROR: How did you get there ?! Is that even possible ?!"); if ((recv_status = Socket.Receive((char *)&recv, 1, recv_size)) == sf::Socket::Done) { #ifdef NET_DEBUG char recv_str[32]; sprintf(recv_str, "received: 0x%02x | %c\n", recv, recv); Event->AppendText(wxString::FromAscii(recv_str)); #endif OnServerData(socket_nb, recv); } else { if (recv_status == sf::Socket::Disconnected) { Event->SendEvent(HOST_PLAYERLEFT); m_numplayers--; std::string player_left = m_client[socket_nb].nick; Event->AppendText( wxString::Format(wxT("*Player : %s left the game.\n\n"), player_left.c_str()) ); // We need to adjust the struct... for (int j = socket_nb; j < m_numplayers; j++) { m_client[j].socket = m_client[j+1].socket; m_client[j].nick = m_client[j+1].nick; m_client[j].ready = m_client[j+1].ready; } // Send disconnected message to all unsigned char send = 0x11; unsigned int str_size = (int)player_left.size(); for (int j=0; j < m_numplayers ; j++) { m_client[j].socket.Send((const char*)&send, 1); m_client[j].socket.Send((const char*)&str_size, 4); m_client[j].socket.Send(player_left.c_str(), (int)str_size + 1); } } else { // Hopefully this should never happen, the client is not // Even warned that he is being dropped... Event->SendEvent(HOST_ERROR, m_client[socket_nb].nick); } m_selector.Remove(Socket); Socket.Close(); } } m_CriticalSection.Leave(); } if(TestDestroy()) { // Delete the Thread and close clients sockets for (char i=0; i < m_numplayers ; i++) m_client[i].socket.Close(); break; } } return NULL; } void ServerSide::SyncValues(unsigned char socketnb, sf::IPAddress Address) { sf::SocketTCP Socket = m_client[socketnb].socket; std::string buffer_str = m_netptr->GetSelectedGame(); char *buffer = NULL; unsigned char init_number; u32 buffer_size = (u32)buffer_str.size(); size_t received; // First, Send the number of connected clients & netmodel Socket.Send((const char *)&m_numplayers, 1); Socket.Send((const char *)&m_netmodel, 4); // Send the Game String Socket.Send((const char *)&buffer_size, 4); Socket.Send(buffer_str.c_str(), buffer_size + 1); // Send the host Nickname buffer_size = (u32)m_nick.size(); Socket.Send((const char *)&buffer_size, 4); Socket.Send(m_nick.c_str(), buffer_size + 1); // Read returned nickname Socket.Receive((char *)&buffer_size, 4, received); buffer = new char[buffer_size + 1]; Socket.Receive(buffer, buffer_size + 1, received); m_client[socketnb].nick = std::string(buffer); m_client[socketnb].ready = false; // Check if the client has the game Socket.Receive((char *)&init_number, 1, received); // Send to all connected clients if (m_numplayers > 0) { unsigned char send = 0x10; buffer_size = (int)m_client[socketnb].nick.size(); for (int i=0; i < m_numplayers ; i++) { // Do not send to connecting player if (i == socketnb) continue; m_client[i].socket.Send((const char *)&send, 1); // Init new connection m_client[i].socket.Send((const char *)&init_number, 1); // Send Game found ? m_client[i].socket.Send((const char *)&buffer_size, 4); // Send client nickname m_client[i].socket.Send(m_client[socketnb].nick.c_str(), buffer_size + 1); } } Event->AppendText( wxString::Format(wxT("*Connection established to %s (%s)\n"), m_client[socketnb].nick.c_str(), Address.ToString().c_str()) ); if (init_number != 0x1F) // Not Found for (int i = 0; i < 4; i++) Event->AppendText(_("WARNING : Game Not Found on Client Side !\n")); delete[] buffer; } void ServerSide::Write(char socknb, const char *data, size_t size, long *ping) { wxCriticalSectionLocker lock(m_CriticalSection); if (ping != NULL) { // Ask for ping unsigned char value = 0x15; size_t recv_size; int four_bytes = 0x101A7FA6; Common::Timer timer; timer.Start(); for (int i=0; i < m_numplayers ; i++) { m_client[i].socket.Send((const char*)&value, 1); timer.Update(); m_client[i].socket.Send((const char*)&four_bytes, 4); m_client[i].socket.Receive((char*)&four_bytes, 4, recv_size); ping[i] = (long)timer.GetTimeDifference(); } return; } // Send the data safely, without intefering with another call m_client[socknb].socket.Send(data, size); } //-------------------------------- // CLIENT SIDE THREAD //-------------------------------- ClientSide::ClientSide(NetPlay* netptr, sf::SocketTCP socket, std::string addr, std::string nick) : wxThread() { m_numplayers = 0; m_data_received = false; m_netmodel = 0; m_socket = socket; m_netptr = netptr; m_nick = nick; m_addr = addr; Event = new NetEvent(m_netptr); } void *ClientSide::Entry() { Event->AppendText(_("*Connection Request... ")); // If we get here, the connection is already accepted, however the game may be full if (SyncValues()) { Event->AppendText(_("Connection successful !\n")); Event->AppendText( wxString::Format(wxT("*Connection established to %s (%s)\n*Game is : %s\n"), m_hostnick.c_str(), m_addr.c_str(), m_selectedgame.c_str() ) ); // Tell the server if the client has the game CheckGameFound(); } else { // Post ServerFull event to GUI m_socket.Close(); Event->SendEvent(HOST_FULL); return NULL; } m_netptr->ChangeSelectedGame(m_selectedgame); Event->SendEvent(HOST_NEWPLAYER); Event->SendEvent(GUI_UPDATE); m_selector.Add(m_socket); while (1) { unsigned char recv; size_t recv_size; sf::Socket::Status recv_status; // we use a selector because of the useful timeout if (m_selector.Wait(0.5) > 0) { m_CriticalSection.Enter(); if ((recv_status = m_socket.Receive((char *)&recv, 1, recv_size)) == sf::Socket::Done) { #ifdef NET_DEBUG char recv_str[32]; sprintf(recv_str, "received: 0x%02x | %c\n", recv, recv); Event->AppendText(wxString::FromAscii(recv_str)); #endif OnClientData(recv); } else { if (recv_status == sf::Socket::Disconnected) { Event->SendEvent(HOST_DISCONNECTED); } else { Event->SendEvent(HOST_ERROR); } m_selector.Remove(m_socket); m_socket.Close(); return NULL; } m_CriticalSection.Leave(); } if(TestDestroy()) { m_socket.Close(); break; } } return NULL; } bool ClientSide::SyncValues() { unsigned int buffer_size = (int)m_nick.size(); char *buffer = NULL; size_t recv_size; // First, Read the init number : nbplayers (0-2) or server full (0x12) m_socket.Receive((char *)&m_numplayers, 1, recv_size); if (m_numplayers == 0x12) return false; m_socket.Receive((char *)&m_netmodel, 4, recv_size); // Send client's nickname m_socket.Send((const char *)&buffer_size, 4); m_socket.Send(m_nick.c_str(), buffer_size + 1); // Read the Game String m_socket.Receive((char *)&buffer_size, 4, recv_size); buffer = new char[buffer_size + 1]; m_socket.Receive(buffer, buffer_size + 1, recv_size); m_selectedgame = std::string(buffer); // Read the host Nickname m_socket.Receive((char *)&buffer_size, 4, recv_size); buffer = new char[buffer_size + 1]; m_socket.Receive(buffer, buffer_size + 1, recv_size); m_hostnick = std::string(buffer); delete[] buffer; return true; } void ClientSide::CheckGameFound() { unsigned char send_value; // Check if the game selected by Host is in Client's Game List m_netptr->IsGameFound(&send_value, m_selectedgame); if (send_value == 0x1F) // Found { m_socket.Send((const char *)&send_value, 1); } else { m_socket.Send((const char *)&send_value, 1); for (int i = 0; i < 2; i++) Event->AppendText(_("WARNING : You do not have the Selected Game !\n")); } } void ClientSide::Write(const char *data, size_t size, long *ping) { wxCriticalSectionLocker lock(m_CriticalSection); if (ping != NULL) { // Ask for ping unsigned char value = 0x15; size_t recv_size; int four_bytes = 0x101A7FA6; Common::Timer timer; timer.Start(); m_socket.Send((const char*)&value, 1); m_socket.Send((const char*)&four_bytes, 4); m_socket.Receive((char*)&four_bytes, 4, recv_size); *ping = (long)timer.GetTimeElapsed(); return; } m_socket.Send(data, size); }