On x86 compilers without fastcall, simulate it when invoking traces and un-simulate...
[wine-gecko.git] / netwerk / base / src / nsLoadGroup.cpp
blob110f1b90cdf7474ae49e23090dd075961ab66c78
1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /* vim: set sw=4 ts=4 sts=4 et cin: */
3 /* ***** BEGIN LICENSE BLOCK *****
4 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
6 * The contents of this file are subject to the Mozilla Public License Version
7 * 1.1 (the "License"); you may not use this file except in compliance with
8 * the License. You may obtain a copy of the License at
9 * http://www.mozilla.org/MPL/
11 * Software distributed under the License is distributed on an "AS IS" basis,
12 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
13 * for the specific language governing rights and limitations under the
14 * License.
16 * The Original Code is mozilla.org code.
18 * The Initial Developer of the Original Code is
19 * Netscape Communications Corporation.
20 * Portions created by the Initial Developer are Copyright (C) 1998
21 * the Initial Developer. All Rights Reserved.
23 * Contributor(s):
24 * Darin Fisher <darin@netscape.com>
26 * Alternatively, the contents of this file may be used under the terms of
27 * either the GNU General Public License Version 2 or later (the "GPL"), or
28 * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
29 * in which case the provisions of the GPL or the LGPL are applicable instead
30 * of those above. If you wish to allow use of your version of this file only
31 * under the terms of either the GPL or the LGPL, and not to allow others to
32 * use your version of this file under the terms of the MPL, indicate your
33 * decision by deleting the provisions above and replace them with the notice
34 * and other provisions required by the GPL or the LGPL. If you do not delete
35 * the provisions above, a recipient may use your version of this file under
36 * the terms of any one of the MPL, the GPL or the LGPL.
38 * ***** END LICENSE BLOCK ***** */
40 #include "nsLoadGroup.h"
41 #include "nsISupportsArray.h"
42 #include "nsEnumeratorUtils.h"
43 #include "nsIServiceManager.h"
44 #include "nsCOMPtr.h"
45 #include "nsIURI.h"
46 #include "prlog.h"
47 #include "nsCRT.h"
48 #include "netCore.h"
49 #include "nsXPIDLString.h"
50 #include "nsReadableUtils.h"
51 #include "nsString.h"
52 #include "nsVoidArray.h"
54 #if defined(PR_LOGGING)
56 // Log module for nsILoadGroup logging...
58 // To enable logging (see prlog.h for full details):
60 // set NSPR_LOG_MODULES=LoadGroup:5
61 // set NSPR_LOG_FILE=nspr.log
63 // this enables PR_LOG_DEBUG level information and places all output in
64 // the file nspr.log
66 static PRLogModuleInfo* gLoadGroupLog = nsnull;
67 #endif
69 #define LOG(args) PR_LOG(gLoadGroupLog, PR_LOG_DEBUG, args)
71 ////////////////////////////////////////////////////////////////////////////////
73 class RequestMapEntry : public PLDHashEntryHdr
75 public:
76 RequestMapEntry(nsIRequest *aRequest) :
77 mKey(aRequest)
81 nsCOMPtr<nsIRequest> mKey;
84 PR_STATIC_CALLBACK(PRBool)
85 RequestHashMatchEntry(PLDHashTable *table, const PLDHashEntryHdr *entry,
86 const void *key)
88 const RequestMapEntry *e =
89 static_cast<const RequestMapEntry *>(entry);
90 const nsIRequest *request = static_cast<const nsIRequest *>(key);
92 return e->mKey == request;
95 PR_STATIC_CALLBACK(void)
96 RequestHashClearEntry(PLDHashTable *table, PLDHashEntryHdr *entry)
98 RequestMapEntry *e = static_cast<RequestMapEntry *>(entry);
100 // An entry is being cleared, let the entry do its own cleanup.
101 e->~RequestMapEntry();
104 PR_STATIC_CALLBACK(PRBool)
105 RequestHashInitEntry(PLDHashTable *table, PLDHashEntryHdr *entry,
106 const void *key)
108 const nsIRequest *const_request = static_cast<const nsIRequest *>(key);
109 nsIRequest *request = const_cast<nsIRequest *>(const_request);
111 // Initialize the entry with placement new
112 new (entry) RequestMapEntry(request);
113 return PR_TRUE;
117 static void
118 RescheduleRequest(nsIRequest *aRequest, PRInt32 delta)
120 nsCOMPtr<nsISupportsPriority> p = do_QueryInterface(aRequest);
121 if (p)
122 p->AdjustPriority(delta);
125 PR_STATIC_CALLBACK(PLDHashOperator)
126 RescheduleRequests(PLDHashTable *table, PLDHashEntryHdr *hdr,
127 PRUint32 number, void *arg)
129 RequestMapEntry *e = static_cast<RequestMapEntry *>(hdr);
130 PRInt32 *delta = static_cast<PRInt32 *>(arg);
132 RescheduleRequest(e->mKey, *delta);
133 return PL_DHASH_NEXT;
137 nsLoadGroup::nsLoadGroup(nsISupports* outer)
138 : mForegroundCount(0)
139 , mLoadFlags(LOAD_NORMAL)
140 , mStatus(NS_OK)
141 , mPriority(PRIORITY_NORMAL)
142 , mIsCanceling(PR_FALSE)
144 NS_INIT_AGGREGATED(outer);
146 #if defined(PR_LOGGING)
147 // Initialize the global PRLogModule for nsILoadGroup logging
148 if (nsnull == gLoadGroupLog)
149 gLoadGroupLog = PR_NewLogModule("LoadGroup");
150 #endif
152 LOG(("LOADGROUP [%x]: Created.\n", this));
154 // Initialize the ops in the hash to null to make sure we get
155 // consistent errors if someone fails to call ::Init() on an
156 // nsLoadGroup.
157 mRequests.ops = nsnull;
160 nsLoadGroup::~nsLoadGroup()
162 nsresult rv;
164 rv = Cancel(NS_BINDING_ABORTED);
165 NS_ASSERTION(NS_SUCCEEDED(rv), "Cancel failed");
167 if (mRequests.ops) {
168 PL_DHashTableFinish(&mRequests);
171 mDefaultLoadRequest = 0;
173 LOG(("LOADGROUP [%x]: Destroyed.\n", this));
177 nsresult nsLoadGroup::Init()
179 static PLDHashTableOps hash_table_ops =
181 PL_DHashAllocTable,
182 PL_DHashFreeTable,
183 PL_DHashVoidPtrKeyStub,
184 RequestHashMatchEntry,
185 PL_DHashMoveEntryStub,
186 RequestHashClearEntry,
187 PL_DHashFinalizeStub,
188 RequestHashInitEntry
191 if (!PL_DHashTableInit(&mRequests, &hash_table_ops, nsnull,
192 sizeof(RequestMapEntry), 16)) {
193 mRequests.ops = nsnull;
195 return NS_ERROR_OUT_OF_MEMORY;
198 return NS_OK;
201 ////////////////////////////////////////////////////////////////////////////////
202 // nsISupports methods:
204 NS_IMPL_AGGREGATED(nsLoadGroup)
205 NS_INTERFACE_MAP_BEGIN_AGGREGATED(nsLoadGroup)
206 NS_INTERFACE_MAP_ENTRY(nsILoadGroup)
207 NS_INTERFACE_MAP_ENTRY(nsIRequest)
208 NS_INTERFACE_MAP_ENTRY(nsISupportsPriority)
209 NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
210 NS_INTERFACE_MAP_END
212 ////////////////////////////////////////////////////////////////////////////////
213 // nsIRequest methods:
215 NS_IMETHODIMP
216 nsLoadGroup::GetName(nsACString &result)
218 // XXX is this the right "name" for a load group?
220 if (!mDefaultLoadRequest) {
221 result.Truncate();
222 return NS_OK;
225 return mDefaultLoadRequest->GetName(result);
228 NS_IMETHODIMP
229 nsLoadGroup::IsPending(PRBool *aResult)
231 *aResult = (mForegroundCount > 0) ? PR_TRUE : PR_FALSE;
232 return NS_OK;
235 NS_IMETHODIMP
236 nsLoadGroup::GetStatus(nsresult *status)
238 if (NS_SUCCEEDED(mStatus) && mDefaultLoadRequest)
239 return mDefaultLoadRequest->GetStatus(status);
241 *status = mStatus;
242 return NS_OK;
245 // PLDHashTable enumeration callback that appends strong references to
246 // all nsIRequest to an nsVoidArray.
247 PR_STATIC_CALLBACK(PLDHashOperator)
248 AppendRequestsToVoidArray(PLDHashTable *table, PLDHashEntryHdr *hdr,
249 PRUint32 number, void *arg)
251 RequestMapEntry *e = static_cast<RequestMapEntry *>(hdr);
252 nsVoidArray *array = static_cast<nsVoidArray *>(arg);
254 nsIRequest *request = e->mKey;
255 NS_ASSERTION(request, "What? Null key in pldhash entry?");
257 PRBool ok = array->AppendElement(request);
259 if (!ok) {
260 return PL_DHASH_STOP;
263 NS_ADDREF(request);
265 return PL_DHASH_NEXT;
268 // nsVoidArray enumeration callback that releases all items in the
269 // nsVoidArray
270 PR_STATIC_CALLBACK(PRBool)
271 ReleaseVoidArrayItems(void *aElement, void *aData)
273 nsISupports *s = static_cast<nsISupports *>(aElement);
275 NS_RELEASE(s);
277 return PR_TRUE;
280 NS_IMETHODIMP
281 nsLoadGroup::Cancel(nsresult status)
283 NS_ASSERTION(NS_FAILED(status), "shouldn't cancel with a success code");
284 nsresult rv;
285 PRUint32 count = mRequests.entryCount;
287 nsAutoVoidArray requests;
289 PL_DHashTableEnumerate(&mRequests, AppendRequestsToVoidArray,
290 static_cast<nsVoidArray *>(&requests));
292 if (requests.Count() != (PRInt32)count) {
293 requests.EnumerateForwards(ReleaseVoidArrayItems, nsnull);
295 return NS_ERROR_OUT_OF_MEMORY;
298 // set the load group status to our cancel status while we cancel
299 // all our requests...once the cancel is done, we'll reset it...
301 mStatus = status;
303 // Set the flag indicating that the loadgroup is being canceled... This
304 // prevents any new channels from being added during the operation.
306 mIsCanceling = PR_TRUE;
308 nsresult firstError = NS_OK;
310 while (count > 0) {
311 nsIRequest* request = static_cast<nsIRequest*>(requests.ElementAt(--count));
313 NS_ASSERTION(request, "NULL request found in list.");
315 RequestMapEntry *entry =
316 static_cast<RequestMapEntry *>
317 (PL_DHashTableOperate(&mRequests, request,
318 PL_DHASH_LOOKUP));
320 if (PL_DHASH_ENTRY_IS_FREE(entry)) {
321 // |request| was removed already
323 NS_RELEASE(request);
325 continue;
328 #if defined(PR_LOGGING)
329 nsCAutoString nameStr;
330 request->GetName(nameStr);
331 LOG(("LOADGROUP [%x]: Canceling request %x %s.\n",
332 this, request, nameStr.get()));
333 #endif
336 // Remove the request from the load group... This may cause
337 // the OnStopRequest notification to fire...
339 // XXX: What should the context be?
341 (void)RemoveRequest(request, nsnull, status);
343 // Cancel the request...
344 rv = request->Cancel(status);
346 // Remember the first failure and return it...
347 if (NS_FAILED(rv) && NS_SUCCEEDED(firstError))
348 firstError = rv;
350 NS_RELEASE(request);
353 #if defined(DEBUG)
354 NS_ASSERTION(mRequests.entryCount == 0, "Request list is not empty.");
355 NS_ASSERTION(mForegroundCount == 0, "Foreground URLs are active.");
356 #endif
358 mStatus = NS_OK;
359 mIsCanceling = PR_FALSE;
361 return firstError;
365 NS_IMETHODIMP
366 nsLoadGroup::Suspend()
368 nsresult rv, firstError;
369 PRUint32 count = mRequests.entryCount;
371 nsAutoVoidArray requests;
373 PL_DHashTableEnumerate(&mRequests, AppendRequestsToVoidArray,
374 static_cast<nsVoidArray *>(&requests));
376 if (requests.Count() != (PRInt32)count) {
377 requests.EnumerateForwards(ReleaseVoidArrayItems, nsnull);
379 return NS_ERROR_OUT_OF_MEMORY;
382 firstError = NS_OK;
384 // Operate the elements from back to front so that if items get
385 // get removed from the list it won't affect our iteration
387 while (count > 0) {
388 nsIRequest* request =
389 static_cast<nsIRequest*>(requests.ElementAt(--count));
391 NS_ASSERTION(request, "NULL request found in list.");
392 if (!request)
393 continue;
395 #if defined(PR_LOGGING)
396 nsCAutoString nameStr;
397 request->GetName(nameStr);
398 LOG(("LOADGROUP [%x]: Suspending request %x %s.\n",
399 this, request, nameStr.get()));
400 #endif
402 // Suspend the request...
403 rv = request->Suspend();
405 // Remember the first failure and return it...
406 if (NS_FAILED(rv) && NS_SUCCEEDED(firstError))
407 firstError = rv;
409 NS_RELEASE(request);
412 return firstError;
416 NS_IMETHODIMP
417 nsLoadGroup::Resume()
419 nsresult rv, firstError;
420 PRUint32 count = mRequests.entryCount;
422 nsAutoVoidArray requests;
424 PL_DHashTableEnumerate(&mRequests, AppendRequestsToVoidArray,
425 static_cast<nsVoidArray *>(&requests));
427 if (requests.Count() != (PRInt32)count) {
428 requests.EnumerateForwards(ReleaseVoidArrayItems, nsnull);
430 return NS_ERROR_OUT_OF_MEMORY;
433 firstError = NS_OK;
435 // Operate the elements from back to front so that if items get
436 // get removed from the list it won't affect our iteration
438 while (count > 0) {
439 nsIRequest* request =
440 static_cast<nsIRequest*>(requests.ElementAt(--count));
442 NS_ASSERTION(request, "NULL request found in list.");
443 if (!request)
444 continue;
446 #if defined(PR_LOGGING)
447 nsCAutoString nameStr;
448 request->GetName(nameStr);
449 LOG(("LOADGROUP [%x]: Resuming request %x %s.\n",
450 this, request, nameStr.get()));
451 #endif
453 // Resume the request...
454 rv = request->Resume();
456 // Remember the first failure and return it...
457 if (NS_FAILED(rv) && NS_SUCCEEDED(firstError))
458 firstError = rv;
460 NS_RELEASE(request);
463 return firstError;
466 NS_IMETHODIMP
467 nsLoadGroup::GetLoadFlags(PRUint32 *aLoadFlags)
469 *aLoadFlags = mLoadFlags;
470 return NS_OK;
473 NS_IMETHODIMP
474 nsLoadGroup::SetLoadFlags(PRUint32 aLoadFlags)
476 mLoadFlags = aLoadFlags;
477 return NS_OK;
480 NS_IMETHODIMP
481 nsLoadGroup::GetLoadGroup(nsILoadGroup **loadGroup)
483 *loadGroup = mLoadGroup;
484 NS_IF_ADDREF(*loadGroup);
485 return NS_OK;
488 NS_IMETHODIMP
489 nsLoadGroup::SetLoadGroup(nsILoadGroup *loadGroup)
491 mLoadGroup = loadGroup;
492 return NS_OK;
495 ////////////////////////////////////////////////////////////////////////////////
496 // nsILoadGroup methods:
498 NS_IMETHODIMP
499 nsLoadGroup::GetDefaultLoadRequest(nsIRequest * *aRequest)
501 *aRequest = mDefaultLoadRequest;
502 NS_IF_ADDREF(*aRequest);
503 return NS_OK;
506 NS_IMETHODIMP
507 nsLoadGroup::SetDefaultLoadRequest(nsIRequest *aRequest)
509 mDefaultLoadRequest = aRequest;
510 // Inherit the group load flags from the default load request
511 if (mDefaultLoadRequest) {
512 mDefaultLoadRequest->GetLoadFlags(&mLoadFlags);
514 // Mask off any bits that are not part of the nsIRequest flags.
515 // in particular, nsIChannel::LOAD_DOCUMENT_URI...
517 mLoadFlags &= 0xFFFF;
519 // Else, do not change the group's load flags (see bug 95981)
520 return NS_OK;
523 NS_IMETHODIMP
524 nsLoadGroup::AddRequest(nsIRequest *request, nsISupports* ctxt)
526 nsresult rv;
528 #if defined(PR_LOGGING)
530 nsCAutoString nameStr;
531 request->GetName(nameStr);
532 LOG(("LOADGROUP [%x]: Adding request %x %s (count=%d).\n",
533 this, request, nameStr.get(), mRequests.entryCount));
535 #endif /* PR_LOGGING */
537 #ifdef DEBUG
539 RequestMapEntry *entry =
540 static_cast<RequestMapEntry *>
541 (PL_DHashTableOperate(&mRequests, request,
542 PL_DHASH_LOOKUP));
544 NS_ASSERTION(PL_DHASH_ENTRY_IS_FREE(entry),
545 "Entry added to loadgroup twice, don't do that");
547 #endif
550 // Do not add the channel, if the loadgroup is being canceled...
552 if (mIsCanceling) {
554 #if defined(PR_LOGGING)
555 LOG(("LOADGROUP [%x]: AddChannel() ABORTED because LoadGroup is"
556 " being canceled!!\n", this));
557 #endif /* PR_LOGGING */
559 return NS_BINDING_ABORTED;
562 nsLoadFlags flags;
563 // if the request is the default load request or if the default
564 // load request is null, then the load group should inherit its
565 // load flags from the request.
566 if (mDefaultLoadRequest == request || !mDefaultLoadRequest)
567 rv = request->GetLoadFlags(&flags);
568 else
569 rv = MergeLoadFlags(request, flags);
570 if (NS_FAILED(rv)) return rv;
573 // Add the request to the list of active requests...
576 RequestMapEntry *entry =
577 static_cast<RequestMapEntry *>
578 (PL_DHashTableOperate(&mRequests, request,
579 PL_DHASH_ADD));
581 if (!entry) {
582 return NS_ERROR_OUT_OF_MEMORY;
585 if (mPriority != 0)
586 RescheduleRequest(request, mPriority);
588 if (!(flags & nsIRequest::LOAD_BACKGROUND)) {
589 // Update the count of foreground URIs..
590 mForegroundCount += 1;
593 // Fire the OnStartRequest notification out to the observer...
595 // If the notification fails then DO NOT add the request to
596 // the load group.
598 nsCOMPtr<nsIRequestObserver> observer = do_QueryReferent(mObserver);
599 if (observer) {
600 LOG(("LOADGROUP [%x]: Firing OnStartRequest for request %x."
601 "(foreground count=%d).\n", this, request, mForegroundCount));
603 rv = observer->OnStartRequest(request, ctxt);
604 if (NS_FAILED(rv)) {
605 LOG(("LOADGROUP [%x]: OnStartRequest for request %x FAILED.\n",
606 this, request));
608 // The URI load has been canceled by the observer. Clean up
609 // the damage...
612 PL_DHashTableOperate(&mRequests, request, PL_DHASH_REMOVE);
614 rv = NS_OK;
616 mForegroundCount -= 1;
620 // Ensure that we're part of our loadgroup while pending
621 if (mForegroundCount == 1 && mLoadGroup) {
622 mLoadGroup->AddRequest(this, nsnull);
627 return rv;
630 NS_IMETHODIMP
631 nsLoadGroup::RemoveRequest(nsIRequest *request, nsISupports* ctxt,
632 nsresult aStatus)
634 NS_ENSURE_ARG_POINTER(request);
635 nsresult rv;
637 #if defined(PR_LOGGING)
639 nsCAutoString nameStr;
640 request->GetName(nameStr);
641 LOG(("LOADGROUP [%x]: Removing request %x %s status %x (count=%d).\n",
642 this, request, nameStr.get(), aStatus, mRequests.entryCount-1));
644 #endif
646 // Make sure we have a owning reference to the request we're about
647 // to remove.
649 nsCOMPtr<nsIRequest> kungFuDeathGrip(request);
652 // Remove the request from the group. If this fails, it means that
653 // the request was *not* in the group so do not update the foreground
654 // count or it will get messed up...
656 RequestMapEntry *entry =
657 static_cast<RequestMapEntry *>
658 (PL_DHashTableOperate(&mRequests, request,
659 PL_DHASH_LOOKUP));
661 if (PL_DHASH_ENTRY_IS_FREE(entry)) {
662 LOG(("LOADGROUP [%x]: Unable to remove request %x. Not in group!\n",
663 this, request));
665 return NS_ERROR_FAILURE;
668 PL_DHashTableRawRemove(&mRequests, entry);
670 // Undo any group priority delta...
671 if (mPriority != 0)
672 RescheduleRequest(request, -mPriority);
674 nsLoadFlags flags;
675 rv = request->GetLoadFlags(&flags);
676 if (NS_FAILED(rv)) return rv;
678 if (!(flags & nsIRequest::LOAD_BACKGROUND)) {
679 NS_ASSERTION(mForegroundCount > 0, "ForegroundCount messed up");
680 mForegroundCount -= 1;
682 // Fire the OnStopRequest out to the observer...
683 nsCOMPtr<nsIRequestObserver> observer = do_QueryReferent(mObserver);
684 if (observer) {
685 LOG(("LOADGROUP [%x]: Firing OnStopRequest for request %x."
686 "(foreground count=%d).\n", this, request, mForegroundCount));
688 rv = observer->OnStopRequest(request, ctxt, aStatus);
690 #if defined(PR_LOGGING)
691 if (NS_FAILED(rv)) {
692 LOG(("LOADGROUP [%x]: OnStopRequest for request %x FAILED.\n",
693 this, request));
695 #endif
698 // If that was the last request -> remove ourselves from loadgroup
699 if (mForegroundCount == 0 && mLoadGroup) {
700 mLoadGroup->RemoveRequest(this, nsnull, aStatus);
704 return rv;
707 // PLDHashTable enumeration callback that appends all items in the
708 // hash to an nsISupportsArray.
709 PR_STATIC_CALLBACK(PLDHashOperator)
710 AppendRequestsToISupportsArray(PLDHashTable *table, PLDHashEntryHdr *hdr,
711 PRUint32 number, void *arg)
713 RequestMapEntry *e = static_cast<RequestMapEntry *>(hdr);
714 nsISupportsArray *array = static_cast<nsISupportsArray *>(arg);
716 PRBool ok = array->AppendElement(e->mKey);
718 if (!ok) {
719 return PL_DHASH_STOP;
722 return PL_DHASH_NEXT;
725 NS_IMETHODIMP
726 nsLoadGroup::GetRequests(nsISimpleEnumerator * *aRequests)
728 nsCOMPtr<nsISupportsArray> array;
729 nsresult rv = NS_NewISupportsArray(getter_AddRefs(array));
730 NS_ENSURE_SUCCESS(rv, rv);
732 PL_DHashTableEnumerate(&mRequests, AppendRequestsToISupportsArray,
733 array.get());
735 PRUint32 count;
736 array->Count(&count);
738 if (count != mRequests.entryCount) {
739 return NS_ERROR_OUT_OF_MEMORY;
742 return NS_NewArrayEnumerator(aRequests, array);
745 NS_IMETHODIMP
746 nsLoadGroup::SetGroupObserver(nsIRequestObserver* aObserver)
748 mObserver = do_GetWeakReference(aObserver);
749 return NS_OK;
752 NS_IMETHODIMP
753 nsLoadGroup::GetGroupObserver(nsIRequestObserver* *aResult)
755 nsCOMPtr<nsIRequestObserver> observer = do_QueryReferent(mObserver);
756 *aResult = observer;
757 NS_IF_ADDREF(*aResult);
758 return NS_OK;
761 NS_IMETHODIMP
762 nsLoadGroup::GetActiveCount(PRUint32* aResult)
764 *aResult = mForegroundCount;
765 return NS_OK;
768 NS_IMETHODIMP
769 nsLoadGroup::GetNotificationCallbacks(nsIInterfaceRequestor **aCallbacks)
771 NS_ENSURE_ARG_POINTER(aCallbacks);
772 *aCallbacks = mCallbacks;
773 NS_IF_ADDREF(*aCallbacks);
774 return NS_OK;
777 NS_IMETHODIMP
778 nsLoadGroup::SetNotificationCallbacks(nsIInterfaceRequestor *aCallbacks)
780 mCallbacks = aCallbacks;
781 return NS_OK;
784 ////////////////////////////////////////////////////////////////////////////////
785 // nsISupportsPriority methods:
787 NS_IMETHODIMP
788 nsLoadGroup::GetPriority(PRInt32 *aValue)
790 *aValue = mPriority;
791 return NS_OK;
794 NS_IMETHODIMP
795 nsLoadGroup::SetPriority(PRInt32 aValue)
797 return AdjustPriority(aValue - mPriority);
800 NS_IMETHODIMP
801 nsLoadGroup::AdjustPriority(PRInt32 aDelta)
803 // Update the priority for each request that supports nsISupportsPriority
804 if (aDelta != 0) {
805 mPriority += aDelta;
806 PL_DHashTableEnumerate(&mRequests, RescheduleRequests, &aDelta);
808 return NS_OK;
811 ////////////////////////////////////////////////////////////////////////////////
813 nsresult nsLoadGroup::MergeLoadFlags(nsIRequest *aRequest, nsLoadFlags& outFlags)
815 nsresult rv;
816 nsLoadFlags flags, oldFlags;
818 rv = aRequest->GetLoadFlags(&flags);
819 if (NS_FAILED(rv))
820 return rv;
822 oldFlags = flags;
824 // Inherit the following bits...
825 flags |= (mLoadFlags & (LOAD_BACKGROUND |
826 LOAD_BYPASS_CACHE |
827 LOAD_FROM_CACHE |
828 VALIDATE_ALWAYS |
829 VALIDATE_ONCE_PER_SESSION |
830 VALIDATE_NEVER));
832 if (flags != oldFlags)
833 rv = aRequest->SetLoadFlags(flags);
835 outFlags = flags;
836 return rv;