mirror of
https://github.com/SFML/SFML.git
synced 2025-01-20 00:05:13 +08:00
2f524481c1
git-svn-id: https://sfml.svn.sourceforge.net/svnroot/sfml/trunk@1002 4e206d99-4929-0410-ac5d-dfc041789085
188 lines
5.8 KiB
D
188 lines
5.8 KiB
D
/*
|
|
* DSFML - SFML Library binding in D language.
|
|
* Copyright (C) 2008 Julien Dagorn (sirjulio13@gmail.com)
|
|
*
|
|
* This software is provided 'as-is', without any express or
|
|
* implied warranty. In no event will the authors be held
|
|
* liable for any damages arising from the use of this software.
|
|
*
|
|
* Permission is granted to anyone to use this software for any purpose,
|
|
* including commercial applications, and to alter it and redistribute
|
|
* it freely, subject to the following restrictions:
|
|
*
|
|
* 1. The origin of this software must not be misrepresented;
|
|
* you must not claim that you wrote the original software.
|
|
* If you use this software in a product, an acknowledgment
|
|
* in the product documentation would be appreciated but
|
|
* is not required.
|
|
*
|
|
* 2. Altered source versions must be plainly marked as such,
|
|
* and must not be misrepresented as being the original software.
|
|
*
|
|
* 3. This notice may not be removed or altered from any
|
|
* source distribution.
|
|
*/
|
|
|
|
module dsfml.network.selector;
|
|
|
|
import dsfml.network.sockettcp;
|
|
import dsfml.network.socketudp;
|
|
|
|
import dsfml.system.common;
|
|
|
|
/**
|
|
* Selector TCP allow reading from multiple sockets
|
|
* without blocking. It's a kind of multiplexer. Use SocketTCP or
|
|
* SocketUDP aliases.
|
|
*/
|
|
class Selector(T) : DSFMLObject
|
|
{
|
|
//Ensure type is correct
|
|
static if (!is(T : SocketTCP) && !is(T : SocketUDP))
|
|
static assert("Only SocketTCP and SocketUDP are valid for Selector.");
|
|
|
|
|
|
/**
|
|
* Default constructor
|
|
*/
|
|
this()
|
|
{
|
|
super(sfSelector_Create());
|
|
}
|
|
|
|
override void dispose()
|
|
{
|
|
sfSelector_Destroy(m_ptr);
|
|
}
|
|
|
|
/**
|
|
* Add a socket to watch
|
|
*
|
|
* Params:
|
|
* socket = A tcp or udp socket
|
|
*/
|
|
void add(T socket)
|
|
{
|
|
if (!(socket.getNativePointer in m_watchedSockets))
|
|
{
|
|
sfSelector_Add(m_ptr, socket.getNativePointer);
|
|
m_watchedSockets[socket.getNativePointer] = socket;
|
|
m_numSocketsWatched++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove a previously added socket
|
|
*
|
|
* Params:
|
|
* socket = A tcp or udp socket
|
|
*/
|
|
void remove(T socket)
|
|
{
|
|
if (socket.getNativePointer in m_watchedSockets)
|
|
{
|
|
sfSelector_Remove(m_ptr, socket.getNativePointer);
|
|
m_watchedSockets.remove(socket.getNativePointer);
|
|
m_numSocketsWatched--;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Clear all sockets being watched
|
|
*/
|
|
void clear()
|
|
{
|
|
sfSelector_Clear(m_ptr);
|
|
foreach(key; m_watchedSockets.keys)
|
|
m_watchedSockets.remove(key);
|
|
m_numSocketsWatched = 0;
|
|
}
|
|
|
|
/**
|
|
* Wait and collect sockets which are ready for reading.
|
|
* This functions will return either when at least one socket
|
|
* is ready, or when the given time is out
|
|
*
|
|
* Params:
|
|
* timeout = Maximum time to wait, in seconds (0 to disable timeout)
|
|
*
|
|
* Returns:
|
|
* Number of sockets ready
|
|
*/
|
|
uint wait(float timeout = 0.f)
|
|
{
|
|
return sfSelector_Wait(m_ptr, timeout);
|
|
}
|
|
|
|
/**
|
|
* After a call to Wait(), get the Index-th socket which is
|
|
* ready for reading. The total number of sockets ready
|
|
* is the integer returned by the previous call to Wait()
|
|
*
|
|
* Params:
|
|
* index = Index of the socket to get
|
|
*
|
|
* Returns:
|
|
* The Index-th socket
|
|
*/
|
|
T GetSocketsReady(uint index)
|
|
{
|
|
return m_watchedSockets[sfSelector_GetSocketReady(m_ptr, index)];
|
|
}
|
|
|
|
|
|
private:
|
|
size_t m_numSocketsWatched;
|
|
T[void*] m_watchedSockets;
|
|
|
|
// External ====================================================================
|
|
extern (C)
|
|
{
|
|
typedef void* function() pf_sfSelector_Create;
|
|
typedef void function(void*) pf_sfSelector_Destroy;
|
|
typedef void function(void*, void*) pf_sfSelector_Add;
|
|
typedef void function(void*, void*) pf_sfSelector_Remove;
|
|
typedef void function(void*) pf_sfSelector_Clear;
|
|
typedef uint function(void*, float) pf_sfSelector_Wait;
|
|
typedef void* function(void*, uint) pf_sfSelector_GetSocketReady;
|
|
|
|
static pf_sfSelector_Create sfSelector_Create;
|
|
static pf_sfSelector_Destroy sfSelector_Destroy;
|
|
static pf_sfSelector_Add sfSelector_Add;
|
|
static pf_sfSelector_Remove sfSelector_Remove;
|
|
static pf_sfSelector_Clear sfSelector_Clear;
|
|
static pf_sfSelector_Wait sfSelector_Wait;
|
|
static pf_sfSelector_GetSocketReady sfSelector_GetSocketReady;
|
|
}
|
|
|
|
static this()
|
|
{
|
|
DllLoader dll = DllLoader.load("csfml-network");
|
|
|
|
static if (is (T : SocketTCP))
|
|
{
|
|
char[] symbol = "sfSelectorTCP";
|
|
}
|
|
else static if (is (T : SocketUDP))
|
|
{
|
|
char[] symbol = "sfSelectorUDP";
|
|
}
|
|
|
|
sfSelector_Add = cast(pf_sfSelector_Add)dll.getSymbol(symbol ~ "_Add");
|
|
sfSelector_Clear = cast(pf_sfSelector_Clear)dll.getSymbol(symbol ~ "_Clear");
|
|
sfSelector_Create = cast(pf_sfSelector_Create)dll.getSymbol(symbol ~ "_Create");
|
|
sfSelector_Destroy = cast(pf_sfSelector_Destroy)dll.getSymbol(symbol ~ "_Destroy");
|
|
sfSelector_GetSocketReady = cast(pf_sfSelector_GetSocketReady)dll.getSymbol(symbol ~ "_GetSocketReady");
|
|
sfSelector_Wait = cast(pf_sfSelector_Wait)dll.getSymbol(symbol ~ "_Wait");
|
|
sfSelector_Remove = cast(pf_sfSelector_Remove)dll.getSymbol(symbol ~ "_Remove");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* alias of selector for TCP or UDP Socket.
|
|
*/
|
|
alias Selector!(SocketTCP) SelectorTCP;
|
|
/// ditto
|
|
alias Selector!(SocketUDP) SelectorUDP;
|
|
|