/*************************************************************************
* UrBackup - Client/Server backup system
* Copyright (C) 2011 Martin Raiber
*
* 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 "MemoryPipe.h"
#include "Server.h"
#ifndef _WIN32
#include
#endif
CMemoryPipe::CMemoryPipe(void)
{
mutex=Server->createMutex();
cond=Server->createCondition();
}
CMemoryPipe::~CMemoryPipe(void)
{
Server->destroy(mutex);
Server->destroy(cond);
}
size_t CMemoryPipe::Read(char *buffer, size_t bsize, int timeoutms)
{
IScopedLock lock(mutex);
if( timeoutms>0 )
{
if( queue.size()==0 )
{
cond->wait( &lock, timeoutms );
}
if( queue.size()==0 )
return 0;
}
else if( timeoutms==0 )
{
if( queue.size()==0 )
return 0;
}
else
{
while( queue.size()==0 )
{
cond->wait(&lock);
}
}
std::string *cstr=&queue[0];
size_t psize=cstr->size();
if( psize<=bsize )
{
memcpy( buffer, cstr->c_str(), psize );
queue.erase( queue.begin() );
return psize;
}
else
{
memcpy( buffer, cstr->c_str(), bsize );
cstr->erase(0, bsize );
return bsize;
}
}
bool CMemoryPipe::Write(const char *buffer, size_t bsize, int timeoutms)
{
IScopedLock lock(mutex);
queue.push_back("");
std::deque::iterator iter=queue.end();
--iter;
std::string *nstr=&(*iter);
nstr->resize( bsize );
memcpy( (char*)nstr->c_str(), buffer, bsize );
cond->notify_one();
return true;
}
size_t CMemoryPipe::Read(std::string *str, int timeoutms )
{
IScopedLock lock(mutex);
if( timeoutms>0 )
{
if( queue.size()==0 )
{
cond->wait( &lock, timeoutms );
}
if( queue.size()==0 )
return 0;
}
else if( timeoutms==0 )
{
if( queue.size()==0 )
return 0;
}
else
{
while( queue.size()==0 )
{
cond->wait(&lock);
}
}
std::string *fs=&queue[0];
size_t fsize=fs->size();
str->resize( fsize );
memcpy( (char*) str->c_str(), fs->c_str(), fsize );
queue.erase( queue.begin() );
return fsize;
}
bool CMemoryPipe::Write(const std::string &str, int timeoutms)
{
IScopedLock lock(mutex);
queue.push_back( str );
cond->notify_one();
return true;
}
bool CMemoryPipe::isWritable(int timeoutms)
{
return true;
}
bool CMemoryPipe::isReadable(int timeoutms)
{
IScopedLock lock(mutex);
if( queue.size()>0 )
return true;
if(timeoutms>0)
cond->wait( &lock, timeoutms );
else if(timeoutms<0)
cond->wait(&lock);
if( queue.size()>0 )
return true;
else
return false;
}
bool CMemoryPipe::hasError(void)
{
return false;
}
size_t CMemoryPipe::getNumElements(void)
{
IScopedLock lock(mutex);
return queue.size();
}
void CMemoryPipe::shutdown(void)
{
}