Bug 1004313: Handle partial flushes correctly in nsBufferedOutputStream::WriteSegments. r=mcmanus

This commit is contained in:
Jim Blandy 2014-05-02 11:32:23 -07:00
parent bbf5d32307
commit cbe08bbca4
5 changed files with 187 additions and 5 deletions

View File

@ -643,7 +643,7 @@ nsBufferedOutputStream::Flush()
// |<-------------->|<---|----->|
// b a c s
uint32_t rem = mFillPoint - amt;
memcpy(mBuffer, mBuffer + amt, rem);
memmove(mBuffer, mBuffer + amt, rem);
mFillPoint = mCursor = rem;
return NS_ERROR_FAILURE; // didn't flush all
}
@ -698,7 +698,7 @@ nsBufferedOutputStream::WriteSegments(nsReadSegmentFun reader, void * closure, u
if (left == 0) {
rv = Flush();
if (NS_FAILED(rv))
return rv;
return (*_retval > 0) ? NS_OK : rv;
continue;
}

View File

@ -109,7 +109,7 @@ public:
}
protected:
NS_IMETHOD Fill() { return NS_OK; } // no-op for input streams
NS_IMETHOD Fill() { return NS_OK; } // no-op for output streams
nsCOMPtr<nsISafeOutputStream> mSafeStream; // QI'd from mStream
};

View File

@ -0,0 +1,179 @@
function run_test() { run_next_test(); }
var CC = Components.Constructor;
var Pipe = CC('@mozilla.org/pipe;1', Ci.nsIPipe, 'init');
var BufferedOutputStream = CC('@mozilla.org/network/buffered-output-stream;1',
Ci.nsIBufferedOutputStream, 'init');
var ScriptableInputStream = CC('@mozilla.org/scriptableinputstream;1',
Ci.nsIScriptableInputStream, 'init');
// Verify that pipes behave as we expect. Subsequent tests assume
// pipes behave as demonstrated here.
add_test(function checkWouldBlockPipe() {
// Create a pipe with a one-byte buffer
var pipe = new Pipe(true, true, 1, 1);
// Writing two bytes should transfer only one byte, and
// return a partial count, not would-block.
do_check_eq(pipe.outputStream.write('xy', 2), 1);
do_check_eq(pipe.inputStream.available(), 1);
do_check_throws_nsIException(() => pipe.outputStream.write('y', 1),
'NS_BASE_STREAM_WOULD_BLOCK');
// Check that nothing was written to the pipe.
do_check_eq(pipe.inputStream.available(), 1);
run_next_test();
});
// A writeFrom to a buffered stream should return
// NS_BASE_STREAM_WOULD_BLOCK if no data was written.
add_test(function writeFromBlocksImmediately() {
// Create a full pipe for our output stream. This will 'would-block' when
// written to.
var outPipe = new Pipe(true, true, 1, 1);
do_check_eq(outPipe.outputStream.write('x', 1), 1);
// Create a buffered stream, and fill its buffer, so the next write will
// try to flush.
var buffered = new BufferedOutputStream(outPipe.outputStream, 10);
do_check_eq(buffered.write('0123456789', 10), 10);
// Create a pipe with some data to be our input stream for the writeFrom
// call.
var inPipe = new Pipe(true, true, 1, 1);
do_check_eq(inPipe.outputStream.write('y', 1), 1);
do_check_eq(inPipe.inputStream.available(), 1);
do_check_throws_nsIException(() => buffered.writeFrom(inPipe.inputStream, 1),
'NS_BASE_STREAM_WOULD_BLOCK');
// No data should have been consumed from the pipe.
do_check_eq(inPipe.inputStream.available(), 1);
run_next_test();
});
// A writeFrom to a buffered stream should return a partial count if any
// data is written, when the last Flush call can only flush a portion of
// the data.
add_test(function writeFromReturnsPartialCountOnPartialFlush() {
// Create a pipe for our output stream. This will accept five bytes, and
// then 'would-block'.
var outPipe = new Pipe(true, true, 5, 1);
// Create a reference to the pipe's readable end that can be used
// from JavaScript.
var outPipeReadable = new ScriptableInputStream(outPipe.inputStream);
// Create a buffered stream whose buffer is too large to be flushed
// entirely to the output pipe.
var buffered = new BufferedOutputStream(outPipe.outputStream, 7);
// Create a pipe to be our input stream for the writeFrom call.
var inPipe = new Pipe(true, true, 15, 1);
// Write some data to our input pipe, for the rest of the test to consume.
do_check_eq(inPipe.outputStream.write('0123456789abcde', 15), 15);
do_check_eq(inPipe.inputStream.available(), 15);
// Write from the input pipe to the buffered stream. The buffered stream
// will fill its seven-byte buffer; and then the flush will only succeed
// in writing five bytes to the output pipe. The writeFrom call should
// return the number of bytes it consumed from inputStream.
do_check_eq(buffered.writeFrom(inPipe.inputStream, 11), 7);
do_check_eq(outPipe.inputStream.available(), 5);
do_check_eq(inPipe.inputStream.available(), 8);
// The partially-successful Flush should have created five bytes of
// available space in the buffered stream's buffer, so we should be able
// to write five bytes to it without blocking.
do_check_eq(buffered.writeFrom(inPipe.inputStream, 5), 5);
do_check_eq(outPipe.inputStream.available(), 5);
do_check_eq(inPipe.inputStream.available(), 3);
// Attempting to write any more data should would-block.
do_check_throws_nsIException(() => buffered.writeFrom(inPipe.inputStream, 1),
'NS_BASE_STREAM_WOULD_BLOCK');
// No data should have been consumed from the pipe.
do_check_eq(inPipe.inputStream.available(), 3);
// Push the rest of the data through, checking that it all came through.
do_check_eq(outPipeReadable.available(), 5);
do_check_eq(outPipeReadable.read(5), '01234');
// Flush returns NS_ERROR_FAILURE if it can't transfer the full amount.
do_check_throws_nsIException(() => buffered.flush(), 'NS_ERROR_FAILURE');
do_check_eq(outPipeReadable.available(), 5);
do_check_eq(outPipeReadable.read(5), '56789');
buffered.flush();
do_check_eq(outPipeReadable.available(), 2);
do_check_eq(outPipeReadable.read(2), 'ab');
do_check_eq(buffered.writeFrom(inPipe.inputStream, 3), 3);
buffered.flush();
do_check_eq(outPipeReadable.available(), 3);
do_check_eq(outPipeReadable.read(3), 'cde');
run_next_test();
});
// A writeFrom to a buffered stream should return a partial count if any
// data is written, when the last Flush call blocks.
add_test(function writeFromReturnsPartialCountOnBlock() {
// Create a pipe for our output stream. This will accept five bytes, and
// then 'would-block'.
var outPipe = new Pipe(true, true, 5, 1);
// Create a reference to the pipe's readable end that can be used
// from JavaScript.
var outPipeReadable = new ScriptableInputStream(outPipe.inputStream);
// Create a buffered stream whose buffer is too large to be flushed
// entirely to the output pipe.
var buffered = new BufferedOutputStream(outPipe.outputStream, 7);
// Create a pipe to be our input stream for the writeFrom call.
var inPipe = new Pipe(true, true, 15, 1);
// Write some data to our input pipe, for the rest of the test to consume.
do_check_eq(inPipe.outputStream.write('0123456789abcde', 15), 15);
do_check_eq(inPipe.inputStream.available(), 15);
// Write enough from the input pipe to the buffered stream to fill the
// output pipe's buffer, and then flush it. Nothing should block or fail,
// but the output pipe should now be full.
do_check_eq(buffered.writeFrom(inPipe.inputStream, 5), 5);
buffered.flush();
do_check_eq(outPipe.inputStream.available(), 5);
do_check_eq(inPipe.inputStream.available(), 10);
// Now try to write more from the input pipe than the buffered stream's
// buffer can hold. It will attempt to flush, but the output pipe will
// would-block without accepting any data. writeFrom should return the
// correct partial count.
do_check_eq(buffered.writeFrom(inPipe.inputStream, 10), 7);
do_check_eq(outPipe.inputStream.available(), 5);
do_check_eq(inPipe.inputStream.available(), 3);
// Attempting to write any more data should would-block.
do_check_throws_nsIException(() => buffered.writeFrom(inPipe.inputStream, 3),
'NS_BASE_STREAM_WOULD_BLOCK');
// No data should have been consumed from the pipe.
do_check_eq(inPipe.inputStream.available(), 3);
// Push the rest of the data through, checking that it all came through.
do_check_eq(outPipeReadable.available(), 5);
do_check_eq(outPipeReadable.read(5), '01234');
// Flush returns NS_ERROR_FAILURE if it can't transfer the full amount.
do_check_throws_nsIException(() => buffered.flush(), 'NS_ERROR_FAILURE');
do_check_eq(outPipeReadable.available(), 5);
do_check_eq(outPipeReadable.read(5), '56789');
do_check_eq(buffered.writeFrom(inPipe.inputStream, 3), 3);
buffered.flush();
do_check_eq(outPipeReadable.available(), 5);
do_check_eq(outPipeReadable.read(5), 'abcde');
run_next_test();
});

View File

@ -18,6 +18,7 @@ support-files =
test_link.desktop
test_link.url
[test_nsIBufferedOutputStream_writeFrom_block.js]
[test_cache2-01-basic.js]
[test_cache2-01a-basic-readonly.js]
[test_cache2-01b-basic-datasize.js]

View File

@ -92,7 +92,8 @@ interface nsIOutputStream : nsISupports
* @return number of bytes written (may be less than aCount)
*
* @throws NS_BASE_STREAM_WOULD_BLOCK if writing to the output stream would
* block the calling thread (non-blocking mode only)
* block the calling thread (non-blocking mode only). This failure
* means no bytes were transferred.
* @throws <other-error> on failure
*
* NOTE: This method is defined by this interface in order to allow the
@ -118,7 +119,8 @@ interface nsIOutputStream : nsISupports
* @return number of bytes written (may be less than aCount)
*
* @throws NS_BASE_STREAM_WOULD_BLOCK if writing to the output stream would
* block the calling thread (non-blocking mode only)
* block the calling thread (non-blocking mode only). This failure
* means no bytes were transferred.
* @throws NS_ERROR_NOT_IMPLEMENTED if the stream has no underlying buffer
* @throws <other-error> on failure
*