Skip to content

Commit

Permalink
Bug 1599160 - Better integration of the shared stylesheet cache with …
Browse files Browse the repository at this point in the history
…the network cache. r=tnikkel,mayhemer,heycam

Make the stylesheet cache respect the same headers as the image cache
does. This makes no-cache stylesheets work as they do now, which is
useful for developers that want to develop sites locally, and for
shift-reloads, etc.

Differential Revision: https://phabricator.services.mozilla.com/D78659
  • Loading branch information
emilio committed Jun 11, 2020
1 parent d999791 commit 5048e0e
Show file tree
Hide file tree
Showing 11 changed files with 176 additions and 78 deletions.
35 changes: 35 additions & 0 deletions dom/base/nsContentUtils.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,7 @@
#include "nsHTMLTags.h"
#include "nsIAnonymousContentCreator.h"
#include "nsIAsyncVerifyRedirectCallback.h"
#include "nsICacheInfoChannel.h"
#include "nsICategoryManager.h"
#include "nsIChannelEventSink.h"
#include "nsIConsoleService.h"
Expand Down Expand Up @@ -10357,3 +10358,37 @@ ScreenIntMargin nsContentUtils::GetWindowSafeAreaInsets(

return windowSafeAreaInsets;
}

/* static */
nsContentUtils::SubresourceCacheValidationInfo
nsContentUtils::GetSubresourceCacheValidationInfo(nsIRequest* aRequest) {
SubresourceCacheValidationInfo info;
if (nsCOMPtr<nsICacheInfoChannel> cache = do_QueryInterface(aRequest)) {
uint32_t value = 0;
if (NS_SUCCEEDED(cache->GetCacheTokenExpirationTime(&value))) {
info.mExpirationTime.emplace(value);
}
}

// Determine whether the cache entry must be revalidated when we try to use
// it. Currently, only HTTP specifies this information...
if (nsCOMPtr<nsIHttpChannel> httpChannel = do_QueryInterface(aRequest)) {
Unused << httpChannel->IsNoStoreResponse(&info.mMustRevalidate);

if (!info.mMustRevalidate) {
Unused << httpChannel->IsNoCacheResponse(&info.mMustRevalidate);
}

// FIXME(bug 1644173): Why this check?
if (!info.mMustRevalidate) {
nsAutoCString cacheHeader;
Unused << httpChannel->GetResponseHeader(
NS_LITERAL_CSTRING("Cache-Control"), cacheHeader);
if (PL_strcasestr(cacheHeader.get(), "must-revalidate")) {
info.mMustRevalidate = true;
}
}
}

return info;
}
18 changes: 18 additions & 0 deletions dom/base/nsContentUtils.h
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@
#include "mozilla/dom/Document.h"
#include "nsPIDOMWindow.h"
#include "nsRFPService.h"
#include "prtime.h"

#if defined(XP_WIN)
// Undefine LoadImage to prevent naming conflict with Windows.
Expand Down Expand Up @@ -3246,6 +3247,23 @@ class nsContentUtils {
nsIScreen* aScreen, const mozilla::ScreenIntMargin& aSafeareaInsets,
const mozilla::LayoutDeviceIntRect& aWindowRect);

struct SubresourceCacheValidationInfo {
// The expiration time, in seconds, if known.
Maybe<uint32_t> mExpirationTime;
bool mMustRevalidate = false;
};

/**
* Gets cache validation info for subresources such as images or CSS
* stylesheets.
*/
static SubresourceCacheValidationInfo GetSubresourceCacheValidationInfo(
nsIRequest*);

static uint32_t SecondsFromPRTime(PRTime aTime) {
return uint32_t(int64_t(aTime) / int64_t(PR_USEC_PER_SEC));
}

private:
static bool InitializeEventTable();

Expand Down
13 changes: 6 additions & 7 deletions image/imgLoader.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -960,11 +960,11 @@ static nsresult NewImageChannel(
return NS_OK;
}

/* static */
uint32_t imgCacheEntry::SecondsFromPRTime(PRTime prTime) {
return uint32_t(int64_t(prTime) / int64_t(PR_USEC_PER_SEC));
static uint32_t SecondsFromPRTime(PRTime aTime) {
return nsContentUtils::SecondsFromPRTime(aTime);
}

/* static */
imgCacheEntry::imgCacheEntry(imgLoader* loader, imgRequest* request,
bool forcePrincipalCheck)
: mLoader(loader),
Expand Down Expand Up @@ -1822,8 +1822,8 @@ bool imgLoader::ValidateEntry(
// If the expiration time is zero, then the request has not gotten far enough
// to know when it will expire.
uint32_t expiryTime = aEntry->GetExpiryTime();
bool hasExpired = expiryTime != 0 &&
expiryTime <= imgCacheEntry::SecondsFromPRTime(PR_Now());
bool hasExpired =
expiryTime != 0 && expiryTime <= SecondsFromPRTime(PR_Now());

nsresult rv;

Expand All @@ -1840,8 +1840,7 @@ bool imgLoader::ValidateEntry(
if (NS_SUCCEEDED(rv)) {
// nsIFile uses millisec, NSPR usec
fileLastMod *= 1000;
hasExpired =
imgCacheEntry::SecondsFromPRTime((PRTime)fileLastMod) > lastModTime;
hasExpired = SecondsFromPRTime((PRTime)fileLastMod) > lastModTime;
}
}
}
Expand Down
2 changes: 0 additions & 2 deletions image/imgLoader.h
Original file line number Diff line number Diff line change
Expand Up @@ -39,8 +39,6 @@ namespace image {} // namespace image

class imgCacheEntry {
public:
static uint32_t SecondsFromPRTime(PRTime prTime);

imgCacheEntry(imgLoader* loader, imgRequest* request,
bool aForcePrincipalCheck);
~imgCacheEntry();
Expand Down
63 changes: 18 additions & 45 deletions image/imgRequest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -517,53 +517,26 @@ void imgRequest::UpdateCacheEntrySize() {
void imgRequest::SetCacheValidation(imgCacheEntry* aCacheEntry,
nsIRequest* aRequest) {
/* get the expires info */
if (aCacheEntry) {
// Expiration time defaults to 0. We set the expiration time on our
// entry if it hasn't been set yet.
if (aCacheEntry->GetExpiryTime() == 0) {
uint32_t expiration = 0;
nsCOMPtr<nsICacheInfoChannel> cacheChannel(do_QueryInterface(aRequest));
if (cacheChannel) {
/* get the expiration time from the caching channel's token */
cacheChannel->GetCacheTokenExpirationTime(&expiration);
}
if (expiration == 0) {
// If the channel doesn't support caching, then ensure this expires the
// next time it is used.
expiration = imgCacheEntry::SecondsFromPRTime(PR_Now()) - 1;
}
aCacheEntry->SetExpiryTime(expiration);
}

// Determine whether the cache entry must be revalidated when we try to use
// it. Currently, only HTTP specifies this information...
nsCOMPtr<nsIHttpChannel> httpChannel(do_QueryInterface(aRequest));
if (httpChannel) {
bool bMustRevalidate = false;

Unused << httpChannel->IsNoStoreResponse(&bMustRevalidate);

if (!bMustRevalidate) {
Unused << httpChannel->IsNoCacheResponse(&bMustRevalidate);
}

if (!bMustRevalidate) {
nsAutoCString cacheHeader;
if (!aCacheEntry || aCacheEntry->GetExpiryTime() != 0) {
return;
}

Unused << httpChannel->GetResponseHeader(
NS_LITERAL_CSTRING("Cache-Control"), cacheHeader);
if (PL_strcasestr(cacheHeader.get(), "must-revalidate")) {
bMustRevalidate = true;
}
}
auto info = nsContentUtils::GetSubresourceCacheValidationInfo(aRequest);

// Cache entries default to not needing to validate. We ensure that
// multiple calls to this function don't override an earlier decision to
// validate by making validation a one-way decision.
if (bMustRevalidate) {
aCacheEntry->SetMustValidate(bMustRevalidate);
}
}
// Expiration time defaults to 0. We set the expiration time on our entry if
// it hasn't been set yet.
if (!info.mExpirationTime) {
// If the channel doesn't support caching, then ensure this expires the
// next time it is used.
info.mExpirationTime.emplace(nsContentUtils::SecondsFromPRTime(PR_Now()) -
1);
}
aCacheEntry->SetExpiryTime(*info.mExpirationTime);
// Cache entries default to not needing to validate. We ensure that
// multiple calls to this function don't override an earlier decision to
// validate by making validation a one-way decision.
if (info.mMustRevalidate) {
aCacheEntry->SetMustValidate(info.mMustRevalidate);
}
}

Expand Down
22 changes: 21 additions & 1 deletion layout/style/Loader.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
#include "mozilla/URLPreloader.h"
#include "nsIRunnable.h"
#include "nsITimedChannel.h"
#include "nsICachingChannel.h"
#include "nsSyncLoadService.h"
#include "nsCOMPtr.h"
#include "nsString.h"
Expand Down Expand Up @@ -900,7 +901,7 @@ std::tuple<RefPtr<StyleSheet>, Loader::SheetState> Loader::CreateSheet(
GetFallbackEncoding(*this, aLinkingContent,
aPreloadOrParentDataEncoding),
aCORSMode, aParsingMode, mCompatMode, sriMetadata, aIsPreload);
auto cacheResult = mSheets->Lookup(key, aSyncLoad);
auto cacheResult = mSheets->Lookup(*this, key, aSyncLoad);
if (const auto& [styleSheet, sheetState] = cacheResult; styleSheet) {
LOG((" Hit cache with state: %s", gStateStrings[size_t(sheetState)]));
return cacheResult;
Expand Down Expand Up @@ -1410,6 +1411,9 @@ Loader::Completed Loader::ParseSheet(const nsACString& aBytes,
SheetLoadData& aLoadData,
AllowAsyncParse aAllowAsync) {
LOG(("css::Loader::ParseSheet"));
if (aLoadData.mURI) {
LOG_URI(" Load succeeded for URI: '%s', parsing", aLoadData.mURI);
}
AUTO_PROFILER_LABEL("css::Loader::ParseSheet", LAYOUT_CSSParsing);

++mParsedSheetCount;
Expand Down Expand Up @@ -2143,6 +2147,22 @@ nsIPrincipal* Loader::LoaderPrincipal() const {
return nsContentUtils::GetSystemPrincipal();
}

bool Loader::ShouldBypassCache() const {
if (!mDocument) {
return false;
}
RefPtr<nsILoadGroup> lg = mDocument->GetDocumentLoadGroup();
if (!lg) {
return false;
}
nsLoadFlags flags;
if (NS_FAILED(lg->GetLoadFlags(&flags))) {
return false;
}
return flags & (nsIRequest::LOAD_BYPASS_CACHE |
nsICachingChannel::LOAD_BYPASS_LOCAL_CACHE);
}

void Loader::BlockOnload() {
if (mDocument) {
mDocument->BlockOnload();
Expand Down
2 changes: 2 additions & 0 deletions layout/style/Loader.h
Original file line number Diff line number Diff line change
Expand Up @@ -505,6 +505,8 @@ class Loader final {
// owned by a document, or the system principal otherwise.
nsIPrincipal* LoaderPrincipal() const;

bool ShouldBypassCache() const;

private:
friend class mozilla::SharedStyleSheetCache;
friend class SheetLoadData;
Expand Down
74 changes: 53 additions & 21 deletions layout/style/SharedStyleSheetCache.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -79,9 +79,16 @@ static void AssertIncompleteSheetMatches(const SheetLoadData& aData,
"CSSOM shouldn't allow access to incomplete sheets");
}

auto SharedStyleSheetCache::Lookup(SheetLoadDataHashKey& aKey, bool aSyncLoad)
-> CacheResult {
bool SharedStyleSheetCache::CompleteSheet::Expired() const {
return mExpirationTime &&
mExpirationTime <= nsContentUtils::SecondsFromPRTime(PR_Now());
}

SharedStyleSheetCache::CacheResult SharedStyleSheetCache::Lookup(
css::Loader& aLoader, const SheetLoadDataHashKey& aKey, bool aSyncLoad) {
nsIURI* uri = aKey.URI();
LOG(("SharedStyleSheetCache::Lookup(%s)", uri->GetSpecOrDefault().get()));

// Try to find first in the XUL prototype cache.
if (dom::IsChromeURI(uri)) {
nsXULPrototypeCache* cache = nsXULPrototypeCache::GetInstance();
Expand All @@ -100,28 +107,35 @@ auto SharedStyleSheetCache::Lookup(SheetLoadDataHashKey& aKey, bool aSyncLoad)
return {CloneSheet(*sheet), SheetState::Complete};
}

LOG(
(" Not cloning due to forced unique inner or mismatched "
"parsing mode"));
LOG((" Not cloning due to mismatched parsing mode"));
}
}
}

// Now complete sheets.
if (auto lookup = mCompleteSheets.Lookup(aKey)) {
LOG((" From completed: %p", lookup.Data().get()));
AssertComplete(*lookup.Data());
MOZ_ASSERT(lookup.Data()->ParsingMode() == aKey.ParsingMode());
const CompleteSheet& completeSheet = lookup.Data();
// We can assert the stylesheet has not been modified, as we clone it on
// insertion.
StyleSheet* cachedSheet = lookup.Data();
MOZ_ASSERT(!cachedSheet->HasForcedUniqueInner());
MOZ_ASSERT(!cachedSheet->HasModifiedRules());

RefPtr<StyleSheet> clone = CloneSheet(*cachedSheet);
MOZ_ASSERT(!clone->HasForcedUniqueInner());
MOZ_ASSERT(!clone->HasModifiedRules());
return {std::move(clone), SheetState::Complete};
StyleSheet& cachedSheet = *completeSheet.mSheet;
LOG((" From completed: %p", &cachedSheet));

if ((!aLoader.ShouldBypassCache() && !completeSheet.Expired()) ||
aLoader.mLoadsPerformed.Contains(aKey)) {
LOG(
(" Not expired yet, or previously loaded already in "
"that document"));

AssertComplete(cachedSheet);
MOZ_ASSERT(cachedSheet.ParsingMode() == aKey.ParsingMode());
MOZ_ASSERT(!cachedSheet.HasForcedUniqueInner());
MOZ_ASSERT(!cachedSheet.HasModifiedRules());

RefPtr<StyleSheet> clone = CloneSheet(cachedSheet);
MOZ_ASSERT(!clone->HasForcedUniqueInner());
MOZ_ASSERT(!clone->HasModifiedRules());
return {std::move(clone), SheetState::Complete};
}
}

if (aSyncLoad) {
Expand Down Expand Up @@ -189,7 +203,7 @@ size_t SharedStyleSheetCache::SizeOfIncludingThis(

n += mCompleteSheets.ShallowSizeOfExcludingThis(aMallocSizeOf);
for (auto iter = mCompleteSheets.ConstIter(); !iter.Done(); iter.Next()) {
n += iter.UserData()->SizeOfIncludingThis(aMallocSizeOf);
n += iter.UserData().mSheet->SizeOfIncludingThis(aMallocSizeOf);
}

// Measurement of the following members may be added later if DMD finds it is
Expand Down Expand Up @@ -312,19 +326,30 @@ void SharedStyleSheetCache::InsertIntoCompleteCacheIfNeeded(
SheetLoadData& aData) {
MOZ_ASSERT(aData.mLoader->GetDocument(),
"We only cache document-associated sheets");
LOG(("SharedStyleSheetCache::InsertIntoCompleteCacheIfNeeded"));
// If we ever start doing this for failed loads, we'll need to adjust the
// PostLoadEvent code that thinks anything already complete must have loaded
// succesfully.
if (aData.mLoadFailed) {
LOG((" Load failed, bailing"));
return;
}

// If this sheet came from the cache already, there's no need to override
// anything.
if (aData.mSheetAlreadyComplete) {
LOG((" Sheet came from the cache, bailing"));
return;
}

if (!aData.mURI) {
LOG((" Inline style sheet, bailing"));
// Inline sheet caching happens in Loader::mInlineSheets.
return;
}

if (aData.mSheet->IsConstructed()) {
LOG((" Constructable style sheet, bailing"));
// Constructable sheets are not worth caching, they're always unique.
return;
}
Expand Down Expand Up @@ -352,18 +377,25 @@ void SharedStyleSheetCache::InsertIntoCompleteCacheIfNeeded(
}
}
} else {
LOG((" Putting style sheet in shared cache: %s",
aData.mURI->GetSpecOrDefault().get()));
SheetLoadDataHashKey key(aData);
MOZ_ASSERT(sheet->IsComplete(), "Should only be caching complete sheets");

#ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
for (const auto& entry : mCompleteSheets) {
MOZ_DIAGNOSTIC_ASSERT(
entry.GetData() != sheet || key.KeyEquals(entry.GetKey()),
"Same sheet, different keys?");
if (!key.KeyEquals(entry.GetKey())) {
MOZ_DIAGNOSTIC_ASSERT(entry.GetData().mSheet != sheet,
"Same sheet, different keys?");
} else {
MOZ_DIAGNOSTIC_ASSERT(
entry.GetData().Expired() || aData.mLoader->ShouldBypassCache(),
"Overriding existing complete entry?");
}
}
#endif

mCompleteSheets.Put(key, std::move(sheet));
mCompleteSheets.Put(key, {aData.mExpirationTime, std::move(sheet)});
}
}

Expand Down
Loading

0 comments on commit 5048e0e

Please sign in to comment.