/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* ***** 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.org code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * 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 "nsIFileTransportService.h" #include "nsIStreamListener.h" #include "nsIServiceManager.h" #include "nsIComponentRegistrar.h" #include "nsIInputStream.h" #include "nsIEventQueue.h" #include "nsIEventQueueService.h" #include "prinrval.h" #include "prmon.h" #include "prcmon.h" #include "prio.h" #include "nsIFileStreams.h" #include "nsILocalFile.h" #include "nsNetUtil.h" #include "nsIPipe.h" #include "nsIOutputStream.h" #include "nsIInputStream.h" #include "nsIRunnable.h" #include "nsIThread.h" #include "nsCOMArray.h" #include "nsIChannel.h" #include "nsCOMPtr.h" #include #include "nsInt64.h" static NS_DEFINE_CID(kFileTransportServiceCID, NS_FILETRANSPORTSERVICE_CID); static NS_DEFINE_CID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID); #if defined(XP_MAC) #define FILE_PATH_SEPARATOR ":" #elif defined(XP_WIN) || defined(XP_OS2) #define FILE_PATH_SEPARATOR "\\" #elif defined(XP_UNIX) || defined(XP_BEOS) #define FILE_PATH_SEPARATOR "/" #else #error need_to_define_your_file_path_separator_and_illegal_characters #endif PRIntervalTime gDuration = 0; PRUint32 gVolume = 0; class nsReader : public nsIRunnable, public nsIStreamListener { public: NS_DECL_ISUPPORTS NS_IMETHOD Run() { printf("waiting\n"); if (!mMonitor) return NS_ERROR_OUT_OF_MEMORY; PR_EnterMonitor(mMonitor); if (mEventQueue == nsnull) PR_CWait(this, PR_INTERVAL_NO_TIMEOUT); PR_ExitMonitor(mMonitor); printf("running\n"); mEventQueue->EventLoop(); printf("quitting\n"); return NS_OK; } nsReader() : mEventQueue(nsnull), mStartTime(0), mThread(nsnull), mBytesRead(0) { mMonitor = PR_NewMonitor(); } virtual ~nsReader() { NS_IF_RELEASE(mThread); NS_IF_RELEASE(mEventQueue); PR_DestroyMonitor(mMonitor); } nsresult Init(nsIThread* thread) { nsresult rv; mThread = thread; NS_ADDREF(mThread); PRThread* prthread; thread->GetPRThread(&prthread); PR_EnterMonitor(mMonitor); nsCOMPtr eventQService = do_GetService(kEventQueueServiceCID, &rv); if (NS_SUCCEEDED(rv)) { rv = eventQService->GetThreadEventQueue(NS_CURRENT_THREAD, &mEventQueue); } if (NS_FAILED(rv)) return rv; // wake up event loop PR_Notify(mMonitor); PR_ExitMonitor(mMonitor); return NS_OK; } NS_IMETHOD OnStartRequest(nsIRequest *request, nsISupports* context) { PR_EnterMonitor(mMonitor); printf("start binding\n"); mStartTime = PR_IntervalNow(); PR_ExitMonitor(mMonitor); return NS_OK; } NS_IMETHOD OnDataAvailable(nsIRequest *request, nsISupports* context, nsIInputStream *aIStream, PRUint32 aSourceOffset, PRUint32 aLength) { PR_EnterMonitor(mMonitor); char buf[1025]; while (aLength > 0) { PRUint32 amt; /*nsresult rv = */aIStream->Read(buf, 1024, &amt); if (amt == 0) break; buf[amt] = '\0'; printf(buf); aLength -= amt; mBytesRead += amt; gVolume += amt; } PR_ExitMonitor(mMonitor); return NS_OK; } NS_IMETHOD OnStopRequest(nsIRequest *request, nsISupports* context, nsresult aStatus) { nsresult rv; PR_EnterMonitor(mMonitor); PRIntervalTime endTime = PR_IntervalNow(); gDuration += (endTime - mStartTime); printf("stop binding, %d\n", aStatus); if (NS_FAILED(aStatus)) printf("channel failed.\n"); printf("read %d bytes\n", mBytesRead); PR_ExitMonitor(mMonitor); // get me out of my event loop rv = mThread->Interrupt(); if (NS_FAILED(rv)) return rv; return rv; } protected: nsIEventQueue* mEventQueue; PRIntervalTime mStartTime; nsIThread* mThread; PRUint32 mBytesRead; private: PRMonitor* mMonitor; }; NS_IMPL_ADDREF(nsReader) NS_IMPL_RELEASE(nsReader) NS_IMETHODIMP nsReader::QueryInterface(const nsIID& aIID, void* *aInstancePtr) { if (NULL == aInstancePtr) { return NS_ERROR_NULL_POINTER; } if (aIID.Equals(NS_GET_IID(nsIRunnable)) || aIID.Equals(NS_GET_IID(nsISupports))) { *aInstancePtr = NS_STATIC_CAST(nsIRunnable*, this); NS_ADDREF_THIS(); return NS_OK; } if (aIID.Equals(NS_GET_IID(nsIStreamListener))) { *aInstancePtr = NS_STATIC_CAST(nsIStreamListener*, this); NS_ADDREF_THIS(); return NS_OK; } return NS_NOINTERFACE; } nsresult Simulated_nsFileTransport_Run(nsReader* reader, const char* path) { // duplicate the work of nsFileTransport::Run here #define NS_FILE_TRANSPORT_BUFFER_SIZE (4*1024) nsresult rv; nsCOMPtr fileStr; nsIInputStream* bufStr = nsnull; PRUint32 sourceOffset = 0; nsCOMPtr out; nsCOMPtr file; rv = reader->OnStartRequest(nsnull, nsnull); if (NS_FAILED(rv)) goto done; // XXX should this abort the transfer? rv = NS_NewNativeLocalFile(nsDependentCString(path), PR_FALSE, getter_AddRefs(file)); if (NS_FAILED(rv)) goto done; rv = NS_NewLocalFileInputStream(getter_AddRefs(fileStr), file); if (NS_FAILED(rv)) goto done; rv = NS_NewPipe(&bufStr, getter_AddRefs(out), NS_FILE_TRANSPORT_BUFFER_SIZE, NS_FILE_TRANSPORT_BUFFER_SIZE); if (NS_FAILED(rv)) goto done; /* if ( spec.GetFileSize() == 0) goto done; */ while (PR_TRUE) { PRUint32 amt; /* id'l change to FillFrom... */ PRInt64 size; rv = file->GetFileSize(&size); if (NS_FAILED(rv)) break; rv = out->WriteFrom(fileStr, nsInt64(size), &amt); if (NS_FAILED(rv) || amt == 0) break; rv = reader->OnDataAvailable(nsnull, nsnull, bufStr, sourceOffset, amt); if (NS_FAILED(rv)) break; sourceOffset += amt; } done: NS_IF_RELEASE(bufStr); rv = reader->OnStopRequest(nsnull, nsnull, rv); return rv; } void SerialReadTest(char* dirName) { nsresult rv; PRStatus status; PRDir* dir = PR_OpenDir(dirName); NS_ASSERTION(dir, "bad dir"); nsCOMArray threads; PRIntervalTime startTime = PR_IntervalNow(); PRDirEntry* entry; while ((entry = PR_ReadDir(dir, PR_SKIP_BOTH)) != nsnull) { char buffer[1024]; sprintf(buffer, "%s%s%s" dirName, FILE_PATH_SEPARATOR, entry->name); nsReader* reader = new nsReader(); NS_ASSERTION(reader, "out of memory"); NS_ADDREF(reader); nsIThread* readerThread; rv = NS_NewThread(&readerThread, reader, 0, PR_JOINABLE_THREAD); NS_ASSERTION(NS_SUCCEEDED(rv), "new thread failed"); rv = reader->Init(readerThread); NS_ASSERTION(NS_SUCCEEDED(rv), "init failed"); // XXXbz shouldn't we put these in our array? nsIStreamListener* listener; reader->QueryInterface(NS_GET_IID(nsIStreamListener), (void**)&listener); NS_ASSERTION(listener, "QI failed"); rv = Simulated_nsFileTransport_Run(reader, buffer); NS_ASSERTION(NS_SUCCEEDED(rv), "Simulated_nsFileTransport_Run failed"); // the reader thread will hang on to these objects until it quits NS_RELEASE(listener); NS_RELEASE(readerThread); NS_RELEASE(reader); } PRIntervalTime endTime = PR_IntervalNow(); printf("duration %d ms, volume %d\n", PR_IntervalToMilliseconds(endTime - startTime), gVolume); gVolume = 0; // now that we've forked all the async requests, wait until they're done PRUint32 threadCount = threads.Count(); for (PRUint32 i = 0; i < threadCount; i++) { nsIThread* thread = threads.ObjectAt(i); thread->Join(); } status = PR_CloseDir(dir); NS_ASSERTION(status == PR_SUCCESS, "can't close dir"); } void ParallelReadTest(char* dirName, nsIFileTransportService* fts) { nsresult rv; PRStatus status; PRDir* dir = PR_OpenDir(dirName); NS_ASSERTION(dir, "bad dir"); nsCOMArray threads; PRDirEntry* entry; while ((entry = PR_ReadDir(dir, PR_SKIP_BOTH)) != nsnull) { nsCOMPtr localFile; rv = NS_NewNativeLocalFile(nsDependentCString(dirName), PR_FALSE, getter_AddRefs(localFile)); NS_ASSERTION(NS_SUCCEEDED(rv), "NS_NewNativeLocalFile failed"); rv = localFile->AppendNative(nsDependentCString(entry->name)); NS_ASSERTION(NS_SUCCEEDED(rv), "AppendPath failed"); nsReader* reader = new nsReader(); NS_ASSERTION(reader, "out of memory"); NS_ADDREF(reader); nsCOMPtr readerThread; rv = NS_NewThread(getter_AddRefs(readerThread), reader, 0, PR_JOINABLE_THREAD); NS_ASSERTION(NS_SUCCEEDED(rv), "new thread failed"); rv = reader->Init(readerThread); NS_ASSERTION(NS_SUCCEEDED(rv), "init failed"); nsIStreamListener* listener; reader->QueryInterface(NS_GET_IID(nsIStreamListener), (void**)&listener); NS_ASSERTION(listener, "QI failed"); nsITransport* trans; rv = fts->CreateTransport(localFile, PR_RDONLY, 0, PR_TRUE, &trans); NS_ASSERTION(NS_SUCCEEDED(rv), "create failed"); nsCOMPtr request; rv = trans->AsyncRead(nsnull, listener, 0, -1, 0, getter_AddRefs(request)); NS_ASSERTION(NS_SUCCEEDED(rv), "AsyncRead failed"); // the reader thread will hang on to these objects until it quits NS_RELEASE(trans); NS_RELEASE(listener); NS_RELEASE(reader); rv = threads.AppendObject(readerThread) ? NS_OK : NS_ERROR_FAILURE; NS_ASSERTION(NS_SUCCEEDED(rv), "AppendElement failed"); } // now that we've forked all the async requests, wait until they're done PRUint32 threadCount = threads.Count(); for (PRUint32 i = 0; i < threadCount; i++) { nsIThread* thread = threads.ObjectAt(i); thread->Join(); } status = PR_CloseDir(dir); NS_ASSERTION(status == PR_SUCCESS, "can't close dir"); } int main(int argc, char* argv[]) { nsresult rv; if (argc < 2) { printf("usage: %s \n", argv[0]); return -1; } char* dirName = argv[1]; { nsCOMPtr servMan; NS_InitXPCOM2(getter_AddRefs(servMan), nsnull, nsnull); nsCOMPtr registrar = do_QueryInterface(servMan); NS_ASSERTION(registrar, "Null nsIComponentRegistrar"); if (registrar) registrar->AutoRegister(nsnull); nsCOMPtr fts = do_GetService(kFileTransportServiceCID, &rv); if (NS_FAILED(rv)) return rv; SerialReadTest(dirName); //ParallelReadTest(dirName, fts); fts->ProcessPendingRequests(); printf("duration %d ms, volume %d\n", PR_IntervalToMilliseconds(gDuration), gVolume); gVolume = 0; } // this scopes the nsCOMPtrs // no nsCOMPtrs are allowed to be alive when you call NS_ShutdownXPCOM rv = NS_ShutdownXPCOM(nsnull); NS_ASSERTION(NS_SUCCEEDED(rv), "NS_ShutdownXPCOM failed"); return 0; }