mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-01 06:35:42 +00:00
313 lines
13 KiB
C++
313 lines
13 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public License
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
* for the specific language governing rights and limitations under the
|
|
* NPL.
|
|
*
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
* Reserved.
|
|
*/
|
|
|
|
#ifndef nsIBuffer_h___
|
|
#define nsIBuffer_h___
|
|
|
|
/**
|
|
* nsIBuffer is something that we use to implement pipes (buffered
|
|
* input/output stream pairs). It might be useful to you for other
|
|
* purposes, but if not, oh well.
|
|
*
|
|
* One of the important things to understand about pipes is how
|
|
* they work with respect to EOF and result values. The following
|
|
* table describes:
|
|
*
|
|
* | empty & not EOF | full | reader closed | writer closed |
|
|
* -------------------------------------------------------------------------------------------------------------
|
|
* buffer Read | readCount == 0 | readCount == N | N/A | readCount == N, return NS_OK -or- |
|
|
* operations | return WOULD_BLOCK | return NS_OK | | readCount == 0, return EOF |
|
|
* -------------------------------------------------------------------------------------------------------------
|
|
* buffer Write | writeCount == N | writeCount == 0 | N/A | assertion! |
|
|
* operations | return NS_OK | return WOULD_BLOCK | | |
|
|
* -------------------------------------------------------------------------------------------------------------
|
|
* input stream | readCount == 0 | readCount == N | assertion! | readCount == N, return NS_OK -or- |
|
|
* Read ops | return WOULD_BLOCK | return NS_OK | | readCount == 0, return EOF |
|
|
* -------------------------------------------------------------------------------------------------------------
|
|
* output stream | writeCount == N | writeCount == 0 | return | assertion! |
|
|
* Write ops | return NS_OK | return WOULD_BLOCK | STREAM_CLOSED | |
|
|
* -------------------------------------------------------------------------------------------------------------
|
|
*/
|
|
|
|
#include "nsISupports.h"
|
|
#include "nscore.h"
|
|
|
|
class nsIInputStream;
|
|
class nsIAllocator;
|
|
class nsIBufferInputStream;
|
|
class nsIBufferOutputStream;
|
|
class nsIBufferObserver;
|
|
|
|
#define NS_IBUFFER_IID \
|
|
{ /* 1eebb300-fb8b-11d2-9324-00104ba0fd40 */ \
|
|
0x1eebb300, \
|
|
0xfb8b, \
|
|
0x11d2, \
|
|
{0x93, 0x24, 0x00, 0x10, 0x4b, 0xa0, 0xfd, 0x40} \
|
|
}
|
|
|
|
#define NS_BUFFER_CID \
|
|
{ /* 5dbe4de0-fbab-11d2-9324-00104ba0fd40 */ \
|
|
0x5dbe4de0, \
|
|
0xfbab, \
|
|
0x11d2, \
|
|
{0x93, 0x24, 0x00, 0x10, 0x4b, 0xa0, 0xfd, 0x40} \
|
|
}
|
|
|
|
#define NS_BUFFER_PROGID "component://netscape/buffer"
|
|
#define NS_BUFFER_CLASSNAME "Buffer"
|
|
|
|
/**
|
|
* The signature for the reader function passed to WriteSegment. This
|
|
* specifies where the data should come from that gets written into the buffer.
|
|
* Implementers should return the following:
|
|
* @return NS_OK and readCount - if successfully read something
|
|
* @return NS_BASE_STREAM_EOF - if no more to read
|
|
* @return NS_BASE_STREAM_WOULD_BLOCK - if there is currently no data (in
|
|
* a non-blocking mode)
|
|
* @return <other-error> - on failure
|
|
*/
|
|
typedef NS_CALLBACK(nsReadSegmentFun)(void* closure,
|
|
char* toRawSegment,
|
|
PRUint32 fromOffset,
|
|
PRUint32 count,
|
|
PRUint32 *readCount);
|
|
|
|
/**
|
|
* The signature of the writer function passed to ReadSegments. This
|
|
* specifies where the data should go that gets read from the buffer.
|
|
* Implementers should return the following:
|
|
* @return NS_OK and writeCount - if successfully wrote something
|
|
* @return NS_BASE_STREAM_CLOSED - if no more can be written
|
|
* @return NS_BASE_STREAM_WOULD_BLOCK - if there is currently space to write (in
|
|
* a non-blocking mode)
|
|
* @return <other-error> - on failure
|
|
*/
|
|
typedef NS_CALLBACK(nsWriteSegmentFun)(void* closure,
|
|
const char* fromRawSegment,
|
|
PRUint32 toOffset,
|
|
PRUint32 count,
|
|
PRUint32 *writeCount);
|
|
|
|
class nsIBuffer : public nsISupports {
|
|
public:
|
|
NS_DEFINE_STATIC_IID_ACCESSOR(NS_IBUFFER_IID);
|
|
|
|
/**
|
|
* The segment overhead is the amount of space chopped out of each
|
|
* segment for implementation purposes. The remainder of the segment
|
|
* is available for data, e.g.:
|
|
* segmentDataSize = growBySize - SEGMENT_OVERHEAD;
|
|
*/
|
|
enum { SEGMENT_OVERHEAD = 8 };
|
|
|
|
/**
|
|
* Initializes a buffer. The segment size (including overhead) will
|
|
* start from and increment by the growBySize, until reaching maxSize.
|
|
* The size of the data that can fit in a segment will be the growBySize
|
|
* minus SEGMENT_OVERHEAD bytes.
|
|
*/
|
|
NS_IMETHOD Init(PRUint32 growBySize, PRUint32 maxSize,
|
|
nsIBufferObserver* observer, nsIAllocator* allocator) = 0;
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
// Methods for Readers
|
|
|
|
/**
|
|
* Reads from the read cursor into a char buffer up to a specified length.
|
|
*/
|
|
NS_IMETHOD Read(char* toBuf, PRUint32 bufLen, PRUint32 *readCount) = 0;
|
|
|
|
/**
|
|
* This read method allows you to pass a callback function that gets called
|
|
* repeatedly for each buffer segment until the entire amount is read.
|
|
* This avoids the need to copy data to/from and intermediate buffer.
|
|
*/
|
|
NS_IMETHOD ReadSegments(nsWriteSegmentFun writer, void* closure, PRUint32 count,
|
|
PRUint32 *readCount) = 0;
|
|
|
|
/**
|
|
* Returns the raw char buffer segment and its length available for reading.
|
|
* @param segmentLogicalOffset - The offset from the current read cursor for
|
|
* the segment to be returned. If this is beyond the available written area,
|
|
* NULL is returned for the resultSegment.
|
|
* @param resultSegment - The resulting read segment.
|
|
* @param resultSegmentLength - The resulting read segment length.
|
|
*
|
|
* @return NS_OK - if a read segment is successfully returned, or if
|
|
* the requested offset is at or beyond the write cursor (in which case
|
|
* the resultSegment will be nsnull and the resultSegmentLen will be 0)
|
|
* @return NS_BASE_STREAM_WOULD_BLOCK - if the buffer size becomes 0
|
|
* @return any error set by SetCondition if the requested offset is at
|
|
* or beyond the write cursor (in which case the resultSegment will be
|
|
* nsnull and the resultSegmentLen will be 0). Note that NS_OK will be
|
|
* returned if SetCondition has not been called.
|
|
* @return any error returned by OnEmpty
|
|
*/
|
|
NS_IMETHOD GetReadSegment(PRUint32 segmentLogicalOffset,
|
|
const char* *resultSegment,
|
|
PRUint32 *resultSegmentLen) = 0;
|
|
|
|
/**
|
|
* Returns the amount of data currently in the buffer available for reading.
|
|
*/
|
|
NS_IMETHOD GetReadableAmount(PRUint32 *amount) = 0;
|
|
|
|
/**
|
|
* Searches for a string in the buffer. Since the buffer has a notion
|
|
* of EOF, it is possible that the string may at some time be in the
|
|
* buffer, but is is not currently found up to some offset. Consequently,
|
|
* both the found and not found cases return an offset:
|
|
* if found, return offset where it was found
|
|
* if not found, return offset of the first byte not searched
|
|
* In the case the buffer is at EOF and the string is not found, the first
|
|
* byte not searched will correspond to the length of the buffer.
|
|
*/
|
|
NS_IMETHOD Search(const char* forString, PRBool ignoreCase,
|
|
PRBool *found, PRUint32 *offsetSearchedTo) = 0;
|
|
|
|
/**
|
|
* Sets that the reader has closed their end of the stream.
|
|
*/
|
|
NS_IMETHOD ReaderClosed(void) = 0;
|
|
|
|
/**
|
|
* Tests whether EOF marker is set. Note that this does not necessarily mean that
|
|
* all the data in the buffer has yet been consumed.
|
|
*/
|
|
NS_IMETHOD GetCondition(nsresult *result) = 0;
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
// Methods for Writers
|
|
|
|
/**
|
|
* Writes from a char buffer up to a specified length.
|
|
* @param writeCount - The amount that could be written. If the buffer becomes full,
|
|
* this could be less then the specified bufLen.
|
|
*/
|
|
NS_IMETHOD Write(const char* fromBuf, PRUint32 bufLen, PRUint32 *writeCount) = 0;
|
|
|
|
/**
|
|
* Writes from an input stream up to a specified count of bytes.
|
|
* @param writeCount - The amount that could be written. If the buffer becomes full,
|
|
* this could be less then the specified count.
|
|
*/
|
|
NS_IMETHOD WriteFrom(nsIInputStream* fromStream, PRUint32 count, PRUint32 *writeCount) = 0;
|
|
|
|
/**
|
|
* This write method allows you to pass a callback function that gets called
|
|
* repeatedly for each buffer segment until the entire amount is written.
|
|
* This avoids the need to copy data to/from and intermediate buffer.
|
|
*/
|
|
NS_IMETHOD WriteSegments(nsReadSegmentFun reader, void* closure, PRUint32 count,
|
|
PRUint32 *writeCount) = 0;
|
|
|
|
/**
|
|
* Returns the raw char buffer segment and its length available for writing.
|
|
* @param resultSegment - The resulting write segment.
|
|
* @param resultSegmentLength - The resulting write segment length.
|
|
*
|
|
* @return NS_OK - if there is a segment available to write to
|
|
* @return NS_BASE_STREAM_CLOSED - if ReaderClosed has been called
|
|
* @return NS_BASE_STREAM_WOULD_BLOCK - if the max buffer size is exceeded
|
|
* @return NS_ERROR_OUT_OF_MEMORY - if a new segment could not be allocated
|
|
* @return any error returned by OnFull
|
|
*/
|
|
NS_IMETHOD GetWriteSegment(char* *resultSegment,
|
|
PRUint32 *resultSegmentLen) = 0;
|
|
|
|
/**
|
|
* Returns the amount of space currently in the buffer available for writing.
|
|
*/
|
|
NS_IMETHOD GetWritableAmount(PRUint32 *amount) = 0;
|
|
|
|
/**
|
|
* Returns whether the reader has closed their end of the stream.
|
|
*/
|
|
NS_IMETHOD GetReaderClosed(PRBool *result) = 0;
|
|
|
|
/**
|
|
* Sets an EOF marker (typcially done by the writer) so that a reader can be informed
|
|
* when all the data in the buffer is consumed. After the EOF marker has been
|
|
* set, all subsequent calls to the above write methods will return NS_BASE_STREAM_EOF.
|
|
*/
|
|
NS_IMETHOD SetCondition(nsresult condition) = 0;
|
|
};
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define NS_IBUFFEROBSERVER_IID \
|
|
{ /* 0c18bef0-22a8-11d3-9349-00104ba0fd40 */ \
|
|
0x0c18bef0, \
|
|
0x22a8, \
|
|
0x11d3, \
|
|
{0x93, 0x49, 0x00, 0x10, 0x4b, 0xa0, 0xfd, 0x40} \
|
|
}
|
|
|
|
/**
|
|
* A buffer observer is used to detect when the buffer becomes completely full
|
|
* or completely empty.
|
|
*/
|
|
class nsIBufferObserver : public nsISupports {
|
|
public:
|
|
NS_DEFINE_STATIC_IID_ACCESSOR(NS_IBUFFEROBSERVER_IID);
|
|
|
|
NS_IMETHOD OnFull(nsIBuffer* buffer) = 0;
|
|
|
|
NS_IMETHOD OnWrite(nsIBuffer*, PRUint32 amount) = 0;
|
|
|
|
NS_IMETHOD OnEmpty(nsIBuffer* buffer) = 0;
|
|
};
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/**
|
|
* Creates a new buffer.
|
|
* @param observer - may be null
|
|
*/
|
|
extern NS_COM nsresult
|
|
NS_NewBuffer(nsIBuffer* *result,
|
|
PRUint32 growBySize, PRUint32 maxSize,
|
|
nsIBufferObserver* observer);
|
|
|
|
/**
|
|
* Creates a new buffer, allocating segments from virtual memory pages.
|
|
*/
|
|
extern NS_COM nsresult
|
|
NS_NewPageBuffer(nsIBuffer* *result,
|
|
PRUint32 growBySize, PRUint32 maxSize,
|
|
nsIBufferObserver* observer);
|
|
|
|
extern NS_COM nsresult
|
|
NS_NewBufferInputStream(nsIBufferInputStream* *result,
|
|
nsIBuffer* buffer, PRBool blocking = PR_FALSE);
|
|
|
|
extern NS_COM nsresult
|
|
NS_NewBufferOutputStream(nsIBufferOutputStream* *result,
|
|
nsIBuffer* buffer, PRBool blocking = PR_FALSE);
|
|
|
|
extern NS_COM nsresult
|
|
NS_NewPipe(nsIBufferInputStream* *inStrResult,
|
|
nsIBufferOutputStream* *outStrResult,
|
|
PRUint32 growBySize, PRUint32 maxSize,
|
|
PRBool blocking, nsIBufferObserver* observer);
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#endif // nsIBuffer_h___
|