/* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is Mozilla. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 2002 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Darin Fisher * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsIOService.h" #include "nsAsyncStreamCopier.h" #include "nsIEventTarget.h" #include "nsStreamUtils.h" #include "nsNetSegmentUtils.h" #include "nsNetUtil.h" #include "nsAutoLock.h" #include "prlog.h" #if defined(PR_LOGGING) // // NSPR_LOG_MODULES=nsStreamCopier:5 // static PRLogModuleInfo *gStreamCopierLog = nsnull; #endif #define LOG(args) PR_LOG(gStreamCopierLog, PR_LOG_DEBUG, args) //----------------------------------------------------------------------------- nsAsyncStreamCopier::nsAsyncStreamCopier() : mLock(nsnull) , mMode(NS_ASYNCCOPY_VIA_READSEGMENTS) , mChunkSize(nsIOService::gDefaultSegmentSize) , mStatus(NS_OK) , mIsPending(PR_FALSE) { #if defined(PR_LOGGING) if (!gStreamCopierLog) gStreamCopierLog = PR_NewLogModule("nsStreamCopier"); #endif LOG(("Creating nsAsyncStreamCopier @%x\n", this)); } nsAsyncStreamCopier::~nsAsyncStreamCopier() { LOG(("Destroying nsAsyncStreamCopier @%x\n", this)); if (mLock) PR_DestroyLock(mLock); } PRBool nsAsyncStreamCopier::IsComplete(nsresult *status) { nsAutoLock lock(mLock); if (status) *status = mStatus; return !mIsPending; } void nsAsyncStreamCopier::Complete(nsresult status) { LOG(("nsAsyncStreamCopier::Complete [this=%x status=%x]\n", this, status)); nsCOMPtr observer; nsCOMPtr ctx; { nsAutoLock lock(mLock); mCopierCtx = nsnull; if (mIsPending) { mIsPending = PR_FALSE; mStatus = status; // setup OnStopRequest callback and release references... observer = mObserver; ctx = mObserverContext; mObserver = nsnull; mObserverContext = nsnull; } } if (observer) { LOG((" calling OnStopRequest [status=%x]\n", status)); observer->OnStopRequest(this, ctx, status); } } void nsAsyncStreamCopier::OnAsyncCopyComplete(void *closure, nsresult status) { nsAsyncStreamCopier *self = (nsAsyncStreamCopier *) closure; self->Complete(status); NS_RELEASE(self); // addref'd in AsyncCopy } //----------------------------------------------------------------------------- // nsISupports NS_IMPL_THREADSAFE_ISUPPORTS2(nsAsyncStreamCopier, nsIRequest, nsIAsyncStreamCopier) //----------------------------------------------------------------------------- // nsIRequest NS_IMETHODIMP nsAsyncStreamCopier::GetName(nsACString &name) { name.Truncate(); return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::IsPending(PRBool *result) { *result = !IsComplete(); return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::GetStatus(nsresult *status) { IsComplete(status); return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::Cancel(nsresult status) { nsCOMPtr copierCtx; { nsAutoLock lock(mLock); if (!mIsPending) return NS_OK; copierCtx.swap(mCopierCtx); } if (NS_SUCCEEDED(status)) { NS_WARNING("cancel with non-failure status code"); status = NS_BASE_STREAM_CLOSED; } if (copierCtx) NS_CancelAsyncCopy(copierCtx, status); return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::Suspend() { NS_NOTREACHED("nsAsyncStreamCopier::Suspend"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsAsyncStreamCopier::Resume() { NS_NOTREACHED("nsAsyncStreamCopier::Resume"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsAsyncStreamCopier::GetLoadFlags(nsLoadFlags *aLoadFlags) { *aLoadFlags = LOAD_NORMAL; return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::SetLoadFlags(nsLoadFlags aLoadFlags) { return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::GetLoadGroup(nsILoadGroup **aLoadGroup) { *aLoadGroup = nsnull; return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::SetLoadGroup(nsILoadGroup *aLoadGroup) { return NS_OK; } //----------------------------------------------------------------------------- // nsIAsyncStreamCopier NS_IMETHODIMP nsAsyncStreamCopier::Init(nsIInputStream *source, nsIOutputStream *sink, nsIEventTarget *target, PRBool sourceBuffered, PRBool sinkBuffered, PRUint32 chunkSize, PRBool closeSource, PRBool closeSink) { NS_ASSERTION(sourceBuffered || sinkBuffered, "at least one stream must be buffered"); NS_ASSERTION(!mLock, "already initialized"); mLock = PR_NewLock(); if (!mLock) return NS_ERROR_OUT_OF_MEMORY; if (chunkSize == 0) chunkSize = nsIOService::gDefaultSegmentSize; mChunkSize = chunkSize; mSource = source; mSink = sink; mCloseSource = closeSource; mCloseSink = closeSink; mMode = sourceBuffered ? NS_ASYNCCOPY_VIA_READSEGMENTS : NS_ASYNCCOPY_VIA_WRITESEGMENTS; if (target) mTarget = target; else { nsresult rv; mTarget = do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID, &rv); if (NS_FAILED(rv)) return rv; } return NS_OK; } NS_IMETHODIMP nsAsyncStreamCopier::AsyncCopy(nsIRequestObserver *observer, nsISupports *ctx) { LOG(("nsAsyncStreamCopier::AsyncCopy [this=%x observer=%x]\n", this, observer)); NS_ASSERTION(mSource && mSink, "not initialized"); nsresult rv; if (observer) { // build proxy for observer events rv = NS_NewRequestObserverProxy(getter_AddRefs(mObserver), observer); if (NS_FAILED(rv)) return rv; } // from this point forward, AsyncCopy is going to return NS_OK. any errors // will be reported via OnStopRequest. mIsPending = PR_TRUE; mObserverContext = ctx; if (mObserver) { rv = mObserver->OnStartRequest(this, mObserverContext); if (NS_FAILED(rv)) Cancel(rv); } // we want to receive progress notifications; release happens in // OnAsyncCopyComplete. NS_ADDREF_THIS(); rv = NS_AsyncCopy(mSource, mSink, mTarget, mMode, mChunkSize, OnAsyncCopyComplete, this, mCloseSource, mCloseSink, getter_AddRefs(mCopierCtx)); if (NS_FAILED(rv)) { NS_RELEASE_THIS(); Cancel(rv); } return NS_OK; }