//
// (c) Yuriy Gorvitovskiy
// for Openh323, www.Openh323.org
//
// Windows CE Port
//
// streambuf implementation
//

#include <iostream.h>
#include <stdlib.h>

#ifndef BUFSIZ
#define BUFSIZ 512
#endif  /* BUFSIZ */

/*******************************************************************************/

streambuf::streambuf()
{
    _fAlloc = 0;
    _fUnbuf = 0;
    x_lastc = EOF;
    _base = NULL;
    _ebuf = NULL;
    _pbase = NULL;
    _pptr = NULL;
    _epptr = NULL;
    _eback = NULL;
    _gptr = NULL;
    _egptr = NULL;
}

/*******************************************************************************/
streambuf::streambuf( char* pBuf, int cbBuf )
{
    _fAlloc = 0;
    _fUnbuf = 0;
    x_lastc = EOF;
    _base = pBuf;
    _ebuf = pBuf + (unsigned)cbBuf;
    _pbase = NULL;
    _pptr = NULL;
    _epptr = NULL;
    _eback = NULL;
    _gptr = NULL;
    _egptr = NULL;

    if( pBuf == NULL || cbBuf == 0 )
	{
        _fUnbuf = 1;
        _base = NULL;
        _ebuf = NULL;
    }
}

/*******************************************************************************/
streambuf::~streambuf()
{
    sync();     // make sure buffer empty before possibly destroying it
    if( (_fAlloc) && (_base) )
        delete _base;
}

/*******************************************************************************/
streambuf * streambuf::setbuf(char * p, int len)
{
    if (!_base)
    {
        if ((!p) || (!len))
            _fUnbuf = 1;        // mark as unbuffered
        else
        {
            _base = p;
            _ebuf = p + (unsigned)len;
            _fUnbuf = 0;
        }
        return (this);
    }
    return((streambuf *)NULL);
}


/*******************************************************************************/
int streambuf::xsputn( const char* pBuf, int cbBuf )
{
    int cbOut;

    for (cbOut = 0; cbBuf--; cbOut++)
	{
        if ((_fUnbuf) || (_pptr >= _epptr))
        {
            if (overflow((unsigned char)*pBuf)==EOF)    // 0-extend 0xFF !=EOF
                break;
        }
        else
		{
            *(_pptr++) = *pBuf;
        }
        pBuf++;
	}
    return cbOut;
}

/*******************************************************************************/
int streambuf::xsgetn( char * pBuf, int cbBuf)
{
    int count;
    int cbIn = 0;
    if (_fUnbuf)
    {
		if (x_lastc==EOF)
			x_lastc=underflow();

		while (cbBuf--)
        {
			if (x_lastc==EOF)
				break;

			*(pBuf++) = (char)x_lastc;
			cbIn++;
			x_lastc=underflow();
        }
    }
    else
    {
	    while (cbBuf)
        {
			if (underflow()==EOF)       // make sure something to read
				break;

			count = __min(egptr() - gptr(),cbBuf);
			if (count>0)
            {
				memcpy(pBuf,gptr(),count);
				pBuf  += count;
				_gptr += count;
				cbIn  += count;
				cbBuf -= count;
            }
        }
    }
    return cbIn;
}

/*******************************************************************************/

int streambuf::sync()
{
    if ((gptr() <_egptr) || (_pptr > _pbase))
	    return EOF;

    return 0;
}

/*******************************************************************************/

int streambuf::allocate()
{
    if ((_fUnbuf) || (_base))
        return 0;
    
	if (doallocate()==EOF) 
		return EOF;

    return(1);
}

/*******************************************************************************/

int streambuf::doallocate()
{
    char * tptr;
    if (!( tptr = new char[BUFSIZ]))
        return(EOF);

    setb(tptr, tptr + BUFSIZ, 1);
    return(1);
}

/*******************************************************************************/
void streambuf::setb(char * b, char * eb, int a )
{
    if ((_fAlloc) && (_base))
        delete _base;

    _base = b;
    _fAlloc = a;
    _ebuf = eb;
}

/*******************************************************************************/
streampos streambuf::seekoff(streamoff,ios::seek_dir,int)
{
	return EOF;
}

/*******************************************************************************/
streampos streambuf::seekpos(streampos pos,int mode)
{
	return seekoff(streamoff(pos), ios::beg, mode);
}

/*******************************************************************************/
int streambuf::pbackfail(int c)
{
    if (eback()<gptr()) return sputbackc((char)c);

    if (seekoff( -1, ios::cur, ios::in)==EOF)  // always EOF for streambufs
        return EOF;

    if (!unbuffered() && egptr())
    {
        memmove((gptr()+1),gptr(),(egptr()-(gptr()+1)));
        *gptr()=(char)c;
    }
    return(c);
}

/*******************************************************************************/
int streambuf::sgetc()
{
    if (_fUnbuf)  // no buffer
    {
        if (x_lastc==EOF)
            x_lastc = underflow();
        return x_lastc;
    }
    else
        return underflow();

}
int streambuf::snextc()
{
    if (_fUnbuf)
	{
        if (x_lastc==EOF)
            underflow();                // skip 1st character
        return x_lastc = underflow();   // return next character, or EOF
	}
    else
	{
        if ((!egptr()) || (gptr()>=egptr()))
            underflow();                // make sure buffer

        if ((++_gptr) < egptr())
            return (int)(unsigned char) *gptr();
        return underflow();             // returns next character, or EOF
   }

}
int streambuf::sbumpc()
{
    int c;
    if (_fUnbuf) // no buffer
    {
        if (x_lastc==EOF)
        {
            c = underflow();
        }
        else
        {
            c = x_lastc;
            x_lastc = EOF;
        }
    }
    else
    {
        if( gptr() < egptr() )
		{
            c = (int)(unsigned char)*(gptr());
		}
        else
		{
            c = underflow();
		}
        _gptr++;
	}
    return c;
}
void streambuf::stossc()
{
	if (_fUnbuf)
    {
        if (x_lastc==EOF)
            underflow();        // throw away current character
        else
            x_lastc=EOF;        // discard current cached character
    }
    else
    {
        if (gptr() >= egptr())
            underflow();
        if (gptr() < egptr())
            _gptr++;
    }
}

syntax highlighted by Code2HTML, v. 0.9.1