/**
* \file
*
* \author Georg Hopp
*
* \copyright
* Copyright © 2014 Georg Hopp
*
* 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, either version 3 of the License, or
* (at your option) any later version.
*
* 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 for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include
#include
#include "trbase.h"
#include "trevent.h"
#include "tr/interface/comm_manager.h"
#include "tr/comm_end_point.h"
#include "tr/comm_manager.h"
TR_CREATE_INTERFACE(TR_CommManager, 7);
void
TR_commManagerAddEndpoint(void * _this, TR_CommEndPoint endpoint)
{
TR_CommManager this = _this;
if (this->endpoints[endpoint->transport->handle]) {
// this should never happen, but if so we assume this is a leftover
// that still has to be deleted.
TR_delete(this->endpoints[endpoint->transport->handle]);
}
this->endpoints[endpoint->transport->handle] = endpoint;
TR_CALL(_this, TR_CommManager, addEndpoint, endpoint);
}
int
TR_commManagerSelect(void * _this, TR_Event event)
{
int timeout; // milliseconds
int * timeoutptr = event->data;
TR_EventDispatcher dispatcher = (TR_EventDispatcher)event->subject;
if (NULL == timeoutptr) {
timeout = TR_eventDispatcherGetDataWaitTime(dispatcher);
} else {
timeout = *timeoutptr;
TR_MEM_FREE(timeoutptr);
}
TR_CALL(_this, TR_CommManager, select, event, timeout);
return 1;
}
int
TR_commManagerEnableWrite(void * _this, TR_Event event)
{
TR_CALL(_this, TR_CommManager, enableWrite, event);
return 1;
}
int
TR_commManagerDisableWrite(void * _this, TR_Event event)
{
TR_EventHandler this = _this;
TR_CommEndPoint endpoint = (TR_CommEndPoint)event->subject;
TR_CALL(_this, TR_CommManager, disableWrite, event);
if (TR_socketFinRd(endpoint->transport)) {
TR_eventHandlerIssueEvent(
this,
event->subject,
TR_CEP_EVENT_SHUT_READ,
NULL);
}
return 1;
}
int
TR_commManagerClose(void * _this, TR_Event event)
{
TR_CommManager this = _this;
TR_CommEndPoint endpoint = (TR_CommEndPoint)event->subject;
TR_socketShutdown(endpoint->transport);
TR_CALL(_this, TR_CommManager, close, event);
TR_delete(this->endpoints[endpoint->transport->handle]);
return 0;
}
int
TR_commManagerShutdownRead(void * _this, TR_Event event)
{
TR_CALL(_this, TR_CommManager, shutdownRead, event);
if (TR_socketFinRd(((TR_CommEndPoint)event->subject)->transport)) {
// close
TR_eventHandlerIssueEvent(
(TR_EventHandler)_this,
event->subject,
TR_CEP_EVENT_CLOSE,
NULL);
} else if (TR_cepHasPendingData((TR_CommEndPoint)event->subject)) {
// handle pending data... close is issued from disableWrite
TR_eventHandlerIssueEvent(
(TR_EventHandler)_this,
event->subject,
TR_CEP_EVENT_CLOSE,
NULL);
} else {
TR_cepSetClose((TR_CommEndPoint)event->subject);
}
return 0;
}
int
TR_commManagerShutdownWrite(void * _this, TR_Event event)
{
TR_CALL(_this, TR_CommManager, shutdownWrite, event);
if (TR_socketFinRd(((TR_CommEndPoint)event->subject)->transport)) {
TR_eventHandlerIssueEvent(
(TR_EventHandler)_this,
event->subject,
TR_CEP_EVENT_CLOSE,
NULL);
}
return 0;
}
// vim: set ts=4 sw=4: