blob: b99538f3814052c32949f927c7c5f1de9e4c8aae [file] [log] [blame]
/*
* Copyright (C) 2006, 2007, 2008, 2009, 2011 Apple Inc. All rights reserved.
* Copyright (C) 2008, 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/)
* Copyright (C) Research In Motion Limited 2009. All rights reserved.
* Copyright (C) 2011 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef FrameLoader_h
#define FrameLoader_h
#include "core/CoreExport.h"
#include "core/dom/IconURL.h"
#include "core/dom/SandboxFlags.h"
#include "core/dom/SecurityContext.h"
#include "core/fetch/ResourceLoaderOptions.h"
#include "core/frame/FrameTypes.h"
#include "core/loader/FrameLoaderStateMachine.h"
#include "core/loader/FrameLoaderTypes.h"
#include "core/loader/HistoryItem.h"
#include "core/loader/NavigationPolicy.h"
#include "platform/Timer.h"
#include "platform/heap/Handle.h"
#include "platform/network/ResourceRequest.h"
#include "wtf/Forward.h"
#include "wtf/HashSet.h"
namespace blink {
class DocumentLoader;
class Frame;
class FrameLoaderClient;
class ProgressTracker;
class ResourceError;
class SerializedScriptValue;
class SubstituteData;
struct FrameLoadRequest;
CORE_EXPORT bool isBackForwardLoadType(FrameLoadType);
class CORE_EXPORT FrameLoader final {
WTF_MAKE_NONCOPYABLE(FrameLoader);
DISALLOW_NEW();
public:
static ResourceRequest resourceRequestFromHistoryItem(HistoryItem*, ResourceRequestCachePolicy);
FrameLoader(LocalFrame*);
~FrameLoader();
void init();
ResourceRequest resourceRequestForReload(FrameLoadType, const KURL& overrideURL = KURL(),
ClientRedirectPolicy = NotClientRedirect);
ProgressTracker& progress() const { return *m_progressTracker; }
// Starts a load. It will eventually call startLoad() or
// loadInSameDocument(). For history navigations or reloads, an appropriate
// FrameLoadType should be given. Otherwise, FrameLoadTypeStandard should be
// used (and the final FrameLoadType will be computed). For history
// navigations, a history item and a HistoryLoadType should also be provided.
void load(const FrameLoadRequest&, FrameLoadType = FrameLoadTypeStandard,
HistoryItem* = nullptr, HistoryLoadType = HistoryDifferentDocumentLoad);
static void reportLocalLoadFailed(LocalFrame*, const String& url);
// Warning: stopAllLoaders can and will detach the LocalFrame out from under you. All callers need to either protect the LocalFrame
// or guarantee they won't in any way access the LocalFrame after stopAllLoaders returns.
void stopAllLoaders();
// FIXME: clear() is trying to do too many things. We should break it down into smaller functions.
void clear();
void replaceDocumentWhileExecutingJavaScriptURL(const String& source, Document* ownerDocument);
// Sets a timer to notify the client that the initial empty document has
// been accessed, and thus it is no longer safe to show a provisional URL
// above the document without risking a URL spoof.
void didAccessInitialDocument();
// If the initial empty document is showing and has been accessed, this
// cancels the timer and immediately notifies the client in cases that
// waiting to notify would allow a URL spoof.
void notifyIfInitialDocumentAccessed();
DocumentLoader* documentLoader() const { return m_documentLoader.get(); }
DocumentLoader* provisionalDocumentLoader() const { return m_provisionalDocumentLoader.get(); }
void receivedMainResourceError(DocumentLoader*, const ResourceError&);
bool isLoadingMainFrame() const;
bool shouldTreatURLAsSameAsCurrent(const KURL&) const;
bool shouldTreatURLAsSrcdocDocument(const KURL&) const;
FrameLoadType loadType() const;
void setLoadType(FrameLoadType loadType) { m_loadType = loadType; }
FrameLoaderClient* client() const;
void setDefersLoading(bool);
void didExplicitOpen();
// Callbacks from DocumentWriter
void didBeginDocument(bool dispatchWindowObjectAvailable);
void receivedFirstData();
String userAgent() const;
void dispatchDidClearWindowObjectInMainWorld();
void dispatchDidClearDocumentOfWindowObject();
void dispatchDocumentElementAvailable();
// The following sandbox flags will be forced, regardless of changes to
// the sandbox attribute of any parent frames.
void forceSandboxFlags(SandboxFlags flags) { m_forcedSandboxFlags |= flags; }
SandboxFlags effectiveSandboxFlags() const;
bool shouldEnforceStrictMixedContentChecking() const;
SecurityContext::InsecureRequestsPolicy insecureRequestsPolicy() const;
SecurityContext::InsecureNavigationsSet* insecureNavigationsToUpgrade() const;
Frame* opener();
void setOpener(LocalFrame*);
void detach();
void finishedParsing();
void checkCompleted();
void receivedMainResourceRedirect(const KURL& newURL);
// This prepares the FrameLoader for the next commit. It will dispatch
// unload events, abort XHR requests and detach the document. Returns true
// if the frame is ready to receive the next commit, or false otherwise.
bool prepareForCommit();
void commitProvisionalLoad();
FrameLoaderStateMachine* stateMachine() const { return &m_stateMachine; }
void applyUserAgent(ResourceRequest&);
bool shouldInterruptLoadForXFrameOptions(const String&, const KURL&, unsigned long requestIdentifier);
bool allAncestorsAreComplete() const; // including this
bool shouldClose(bool isReload = false);
void dispatchUnloadEvent();
bool allowPlugins(ReasonForCallingAllowPlugins);
void updateForSameDocumentNavigation(const KURL&, SameDocumentNavigationSource, PassRefPtr<SerializedScriptValue>, HistoryScrollRestorationType, FrameLoadType);
HistoryItem* currentItem() const { return m_currentItem.get(); }
void saveScrollState();
void restoreScrollPositionAndViewState();
bool shouldContinueForNavigationPolicy(const ResourceRequest&, const SubstituteData&, DocumentLoader*, ContentSecurityPolicyDisposition,
NavigationType, NavigationPolicy, bool shouldReplaceCurrentEntry, bool isClientRedirect);
DECLARE_TRACE();
private:
void checkTimerFired(Timer<FrameLoader>*);
void didAccessInitialDocumentTimerFired(Timer<FrameLoader>*);
bool prepareRequestForThisFrame(FrameLoadRequest&);
static void setReferrerForFrameRequest(ResourceRequest&, ShouldSendReferrer, Document*);
FrameLoadType determineFrameLoadType(const FrameLoadRequest&);
SubstituteData defaultSubstituteDataForURL(const KURL&);
bool shouldPerformFragmentNavigation(bool isFormSubmission, const String& httpMethod, FrameLoadType, const KURL&);
void processFragment(const KURL&, LoadStartType);
void startLoad(FrameLoadRequest&, FrameLoadType, NavigationPolicy);
enum class HistoryNavigationType {
DifferentDocument,
Fragment,
HistoryApi
};
void setHistoryItemStateForCommit(HistoryCommitType, HistoryNavigationType);
void loadInSameDocument(const KURL&, PassRefPtr<SerializedScriptValue> stateObject, FrameLoadType, HistoryLoadType, ClientRedirectPolicy);
void scheduleCheckCompleted();
void detachDocumentLoader(RefPtrWillBeMember<DocumentLoader>&);
RawPtrWillBeMember<LocalFrame> m_frame;
// FIXME: These should be OwnPtr<T> to reduce build times and simplify
// header dependencies unless performance testing proves otherwise.
// Some of these could be lazily created for memory savings on devices.
mutable FrameLoaderStateMachine m_stateMachine;
OwnPtrWillBeMember<ProgressTracker> m_progressTracker;
FrameLoadType m_loadType;
// Document loaders for the three phases of frame loading. Note that while
// a new request is being loaded, the old document loader may still be referenced.
// E.g. while a new request is in the "policy" state, the old document loader may
// be consulted in particular as it makes sense to imply certain settings on the new loader.
RefPtrWillBeMember<DocumentLoader> m_documentLoader;
RefPtrWillBeMember<DocumentLoader> m_provisionalDocumentLoader;
RefPtrWillBeMember<HistoryItem> m_currentItem;
RefPtrWillBeMember<HistoryItem> m_provisionalItem;
class DeferredHistoryLoad : public NoBaseWillBeGarbageCollectedFinalized<DeferredHistoryLoad> {
WTF_MAKE_NONCOPYABLE(DeferredHistoryLoad);
public:
static PassOwnPtrWillBeRawPtr<DeferredHistoryLoad> create(ResourceRequest request, HistoryItem* item, FrameLoadType loadType, HistoryLoadType historyLoadType)
{
return adoptPtrWillBeNoop(new DeferredHistoryLoad(request, item, loadType, historyLoadType));
}
DeferredHistoryLoad(ResourceRequest request, HistoryItem* item, FrameLoadType loadType,
HistoryLoadType historyLoadType)
: m_request(request)
, m_item(item)
, m_loadType(loadType)
, m_historyLoadType(historyLoadType)
{
}
DEFINE_INLINE_TRACE()
{
visitor->trace(m_item);
}
ResourceRequest m_request;
RefPtrWillBeMember<HistoryItem> m_item;
FrameLoadType m_loadType;
HistoryLoadType m_historyLoadType;
};
OwnPtrWillBeMember<DeferredHistoryLoad> m_deferredHistoryLoad;
bool m_inStopAllLoaders;
Timer<FrameLoader> m_checkTimer;
bool m_didAccessInitialDocument;
Timer<FrameLoader> m_didAccessInitialDocumentTimer;
SandboxFlags m_forcedSandboxFlags;
bool m_dispatchingDidClearWindowObjectInMainWorld;
};
} // namespace blink
#endif // FrameLoader_h