diff options
author | amit@chromium.org <amit@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-05-07 18:29:24 +0000 |
---|---|---|
committer | amit@chromium.org <amit@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-05-07 18:29:24 +0000 |
commit | b31b9ff8646154d3e5c21e113d5afc634c61e9a2 (patch) | |
tree | 3b04d0a3988efa1a80c204e788f1620ac5dec951 | |
parent | b13205648a9c45ba4d5e6fa5db3012bc8a1cf75e (diff) | |
download | chromium_src-b31b9ff8646154d3e5c21e113d5afc634c61e9a2.zip chromium_src-b31b9ff8646154d3e5c21e113d5afc634c61e9a2.tar.gz chromium_src-b31b9ff8646154d3e5c21e113d5afc634c61e9a2.tar.bz2 |
Cache progress notifications only if necessary
We cache and suppress sending progress notifications till
we get the first OnDataAvailable. This is to prevent
mshtml from making up its mind about the mime type.
However, this is the invasive part of the patch and
could trip other software that's due to mistimed progress
notifications. It is probably not good to hide redirect
and some cookie notifications.
We only need to suppress data notifications like
BINDSTATUS_MIMETYPEAVAILABLE,
BINDSTATUS_CACHEFILENAMEAVAILABLE etc.
This is an atempt to reduce the exposure by starting to
cache only when we receive one of the interesting progress
notification.
BUG=42611
TEST=none
Review URL: http://codereview.chromium.org/2046003
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@46715 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | chrome_frame/urlmon_bind_status_callback.cc | 67 | ||||
-rw-r--r-- | chrome_frame/urlmon_bind_status_callback.h | 4 |
2 files changed, 63 insertions, 8 deletions
diff --git a/chrome_frame/urlmon_bind_status_callback.cc b/chrome_frame/urlmon_bind_status_callback.cc index 354ab3f..8fb6ebd 100644 --- a/chrome_frame/urlmon_bind_status_callback.cc +++ b/chrome_frame/urlmon_bind_status_callback.cc @@ -88,7 +88,13 @@ HRESULT SniffData::InitializeCache(const std::wstring& url) { DCHECK(mem) << "GlobalAlloc failed: " << GetLastError(); HRESULT hr = CreateStreamOnHGlobal(mem, TRUE, cache_.Receive()); - DLOG_IF(ERROR, FAILED(hr)) << "CreateStreamOnHGlobal failed: " << hr; + if (SUCCEEDED(hr)) { + ULARGE_INTEGER size = {0}; + cache_->SetSize(size); + } else { + DLOG(ERROR) << "CreateStreamOnHGlobal failed: " << hr; + } + return hr; } @@ -212,16 +218,21 @@ STDMETHODIMP BSCBStorageBind::OnProgress(ULONG progress, ULONG progress_max, status_code, PlatformThread::CurrentId(), status_text); HRESULT hr = S_OK; - if (data_sniffer_.is_undetermined()) { + + // Remember the last redirected URL in case we get switched into + // chrome frame + if (status_code == BINDSTATUS_REDIRECTING) { + scoped_refptr<BindContextInfo> info = + BindContextInfo::FromBindContext(bind_ctx_); + DCHECK(info); + if (info) + info->set_url(status_text); + } + + if (ShouldCacheProgress(status_code)) { Progress new_progress = { progress, progress_max, status_code, status_text ? status_text : std::wstring() }; saved_progress_.push_back(new_progress); - if (status_code == BINDSTATUS_REDIRECTING) { - scoped_refptr<BindContextInfo> info = - BindContextInfo::FromBindContext(bind_ctx_); - DCHECK(info); - info->set_url(status_text); - } } else { hr = CallbackImpl::OnProgress(progress, progress_max, status_code, status_text); @@ -338,3 +349,43 @@ HRESULT BSCBStorageBind::MayPlayBack(DWORD flags) { return hr; } +// We cache and suppress sending progress notifications till +// we get the first OnDataAvailable. This is to prevent +// mshtml from making up its mind about the mime type. +// However, this is the invasive part of the patch and +// could trip other software that's due to mistimed progress +// notifications. It is probably not a good idea to hide redirects +// and some cookie notifications. +// +// We only need to suppress data notifications like +// BINDSTATUS_MIMETYPEAVAILABLE, +// BINDSTATUS_CACHEFILENAMEAVAILABLE etc. +// +// This is an atempt to reduce the exposure by starting to +// cache only when we receive one of the interesting progress +// notification. +bool BSCBStorageBind::ShouldCacheProgress(unsigned long status_code) const { + // We need to cache progress notifications only if we haven't yet figured + // out which way the request is going. + if (data_sniffer_.is_undetermined()) { + // If we are already caching then continue. + if (!saved_progress_.empty()) + return true; + // Start caching only if we see one of the interesting progress + // notifications. + switch (status_code) { + case BINDSTATUS_BEGINDOWNLOADDATA: + case BINDSTATUS_DOWNLOADINGDATA: + case BINDSTATUS_USINGCACHEDCOPY: + case BINDSTATUS_MIMETYPEAVAILABLE: + case BINDSTATUS_CACHEFILENAMEAVAILABLE: + case BINDSTATUS_SERVER_MIMETYPEAVAILABLE: + return true; + default: + break; + } + } + + return false; +} + diff --git a/chrome_frame/urlmon_bind_status_callback.h b/chrome_frame/urlmon_bind_status_callback.h index 8b6fd3b..33e5a51 100644 --- a/chrome_frame/urlmon_bind_status_callback.h +++ b/chrome_frame/urlmon_bind_status_callback.h @@ -111,6 +111,10 @@ END_COM_MAP() STDMETHOD(OnStopBinding)(HRESULT hresult, LPCWSTR error); protected: + // is it a good time to start caching progress notifications + bool ShouldCacheProgress(ULONG status_code) const; + + protected: SniffData data_sniffer_; // A structure to cache the progress notifications |