mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-07 04:05:49 +00:00
2a71f17fe3
--HG-- rename : modules/libjar/zipwriter/src/StreamFunctions.cpp => modules/libjar/zipwriter/StreamFunctions.cpp rename : modules/libjar/zipwriter/src/StreamFunctions.h => modules/libjar/zipwriter/StreamFunctions.h rename : modules/libjar/zipwriter/src/ZipWriterModule.cpp => modules/libjar/zipwriter/ZipWriterModule.cpp rename : modules/libjar/zipwriter/src/nsDeflateConverter.cpp => modules/libjar/zipwriter/nsDeflateConverter.cpp rename : modules/libjar/zipwriter/src/nsDeflateConverter.h => modules/libjar/zipwriter/nsDeflateConverter.h rename : modules/libjar/zipwriter/public/nsIZipWriter.idl => modules/libjar/zipwriter/nsIZipWriter.idl rename : modules/libjar/zipwriter/src/nsZipDataStream.cpp => modules/libjar/zipwriter/nsZipDataStream.cpp rename : modules/libjar/zipwriter/src/nsZipDataStream.h => modules/libjar/zipwriter/nsZipDataStream.h rename : modules/libjar/zipwriter/src/nsZipHeader.cpp => modules/libjar/zipwriter/nsZipHeader.cpp rename : modules/libjar/zipwriter/src/nsZipHeader.h => modules/libjar/zipwriter/nsZipHeader.h rename : modules/libjar/zipwriter/src/nsZipWriter.cpp => modules/libjar/zipwriter/nsZipWriter.cpp rename : modules/libjar/zipwriter/src/nsZipWriter.h => modules/libjar/zipwriter/nsZipWriter.h
199 lines
6.2 KiB
C++
199 lines
6.2 KiB
C++
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#include "StreamFunctions.h"
|
|
#include "nsDeflateConverter.h"
|
|
#include "nsIStringStream.h"
|
|
#include "nsIInputStreamPump.h"
|
|
#include "nsComponentManagerUtils.h"
|
|
#include "nsMemory.h"
|
|
#include "nsAutoPtr.h"
|
|
#include "plstr.h"
|
|
|
|
#define ZLIB_TYPE "deflate"
|
|
#define GZIP_TYPE "gzip"
|
|
#define X_GZIP_TYPE "x-gzip"
|
|
|
|
/**
|
|
* nsDeflateConverter is a stream converter applies the deflate compression
|
|
* method to the data.
|
|
*/
|
|
NS_IMPL_ISUPPORTS(nsDeflateConverter, nsIStreamConverter,
|
|
nsIStreamListener,
|
|
nsIRequestObserver)
|
|
|
|
nsresult nsDeflateConverter::Init()
|
|
{
|
|
int zerr;
|
|
|
|
mOffset = 0;
|
|
|
|
mZstream.zalloc = Z_NULL;
|
|
mZstream.zfree = Z_NULL;
|
|
mZstream.opaque = Z_NULL;
|
|
|
|
int32_t window = MAX_WBITS;
|
|
switch (mWrapMode) {
|
|
case WRAP_NONE:
|
|
window = -window;
|
|
break;
|
|
case WRAP_GZIP:
|
|
window += 16;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
zerr = deflateInit2(&mZstream, mLevel, Z_DEFLATED, window, 8,
|
|
Z_DEFAULT_STRATEGY);
|
|
if (zerr != Z_OK) return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
mZstream.next_out = mWriteBuffer;
|
|
mZstream.avail_out = sizeof(mWriteBuffer);
|
|
|
|
// mark the input buffer as empty.
|
|
mZstream.avail_in = 0;
|
|
mZstream.next_in = Z_NULL;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
/* nsIInputStream convert (in nsIInputStream aFromStream, in string aFromType
|
|
* in string aToType, in nsISupports aCtxt); */
|
|
NS_IMETHODIMP nsDeflateConverter::Convert(nsIInputStream *aFromStream,
|
|
const char *aFromType,
|
|
const char *aToType,
|
|
nsISupports *aCtxt,
|
|
nsIInputStream **_retval)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
/* void asyncConvertData (in string aFromType, in string aToType,
|
|
* in nsIStreamListener aListener,
|
|
* in nsISupports aCtxt); */
|
|
NS_IMETHODIMP nsDeflateConverter::AsyncConvertData(const char *aFromType,
|
|
const char *aToType,
|
|
nsIStreamListener *aListener,
|
|
nsISupports *aCtxt)
|
|
{
|
|
if (mListener)
|
|
return NS_ERROR_ALREADY_INITIALIZED;
|
|
|
|
NS_ENSURE_ARG_POINTER(aListener);
|
|
|
|
if (!PL_strncasecmp(aToType, ZLIB_TYPE, sizeof(ZLIB_TYPE)-1))
|
|
mWrapMode = WRAP_ZLIB;
|
|
else if (!PL_strcasecmp(aToType, GZIP_TYPE) ||
|
|
!PL_strcasecmp(aToType, X_GZIP_TYPE))
|
|
mWrapMode = WRAP_GZIP;
|
|
else
|
|
mWrapMode = WRAP_NONE;
|
|
|
|
nsresult rv = Init();
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mListener = aListener;
|
|
mContext = aCtxt;
|
|
return rv;
|
|
}
|
|
|
|
/* void onDataAvailable (in nsIRequest aRequest, in nsISupports aContext,
|
|
* in nsIInputStream aInputStream,
|
|
* in unsigned long long aOffset,
|
|
* in unsigned long aCount); */
|
|
NS_IMETHODIMP nsDeflateConverter::OnDataAvailable(nsIRequest *aRequest,
|
|
nsISupports *aContext,
|
|
nsIInputStream *aInputStream,
|
|
uint64_t aOffset,
|
|
uint32_t aCount)
|
|
{
|
|
if (!mListener)
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
nsAutoArrayPtr<char> buffer(new char[aCount]);
|
|
NS_ENSURE_TRUE(buffer, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
nsresult rv = ZW_ReadData(aInputStream, buffer.get(), aCount);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// make sure we aren't reading too much
|
|
mZstream.avail_in = aCount;
|
|
mZstream.next_in = (unsigned char*)buffer.get();
|
|
|
|
int zerr = Z_OK;
|
|
// deflate loop
|
|
while (mZstream.avail_in > 0 && zerr == Z_OK) {
|
|
zerr = deflate(&mZstream, Z_NO_FLUSH);
|
|
|
|
while (mZstream.avail_out == 0) {
|
|
// buffer is full, push the data out to the listener
|
|
rv = PushAvailableData(aRequest, aContext);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
zerr = deflate(&mZstream, Z_NO_FLUSH);
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
/* void onStartRequest (in nsIRequest aRequest, in nsISupports aContext); */
|
|
NS_IMETHODIMP nsDeflateConverter::OnStartRequest(nsIRequest *aRequest,
|
|
nsISupports *aContext)
|
|
{
|
|
if (!mListener)
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
return mListener->OnStartRequest(aRequest, mContext);
|
|
}
|
|
|
|
/* void onStopRequest (in nsIRequest aRequest, in nsISupports aContext,
|
|
* in nsresult aStatusCode); */
|
|
NS_IMETHODIMP nsDeflateConverter::OnStopRequest(nsIRequest *aRequest,
|
|
nsISupports *aContext,
|
|
nsresult aStatusCode)
|
|
{
|
|
if (!mListener)
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
nsresult rv;
|
|
|
|
int zerr;
|
|
do {
|
|
zerr = deflate(&mZstream, Z_FINISH);
|
|
rv = PushAvailableData(aRequest, aContext);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
} while (zerr == Z_OK);
|
|
|
|
deflateEnd(&mZstream);
|
|
|
|
return mListener->OnStopRequest(aRequest, mContext, aStatusCode);
|
|
}
|
|
|
|
nsresult nsDeflateConverter::PushAvailableData(nsIRequest *aRequest,
|
|
nsISupports *aContext)
|
|
{
|
|
uint32_t bytesToWrite = sizeof(mWriteBuffer) - mZstream.avail_out;
|
|
// We don't need to do anything if there isn't any data
|
|
if (bytesToWrite == 0)
|
|
return NS_OK;
|
|
|
|
nsresult rv;
|
|
nsCOMPtr<nsIStringInputStream> stream =
|
|
do_CreateInstance("@mozilla.org/io/string-input-stream;1", &rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
stream->ShareData((char*)mWriteBuffer, bytesToWrite);
|
|
rv = mListener->OnDataAvailable(aRequest, mContext, stream, mOffset,
|
|
bytesToWrite);
|
|
|
|
// now set the state for 'deflate'
|
|
mZstream.next_out = mWriteBuffer;
|
|
mZstream.avail_out = sizeof(mWriteBuffer);
|
|
|
|
mOffset += bytesToWrite;
|
|
return rv;
|
|
}
|