blob: 16b739afdcc670f0b8c3ea3fe106ecd66e291f71 [file] [log] [blame]
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/web_cache/renderer/web_cache_impl.h"
#include <limits>
#include <memory>
#include "base/bind.h"
#include "base/feature_list.h"
#include "base/numerics/safe_conversions.h"
#include "components/web_cache/public/features.h"
#include "content/public/renderer/render_thread.h"
#include "third_party/blink/public/common/features.h"
#include "third_party/blink/public/platform/web_cache.h"
namespace web_cache {
WebCacheImpl::WebCacheImpl() {
// The cache implementation is a blink::MemoryCache, which already owns a
// memory pressure listener. This listener is only enabled for low end devices
// (with 512MB or less of RAM), so for this type of device this memory
// pressure listener is redundant as we'll clear the cache twice. Preventing
// this would require exposing the |kTrimWebCacheOnMemoryPressureOnly| as a
// blink feature and this adds too much overhead for the sake of this
// experiment. If this feature gets enabled by default this should be
// optimized (the low end device check should be removed from
// blink::MemoryCache and this listener should be removed).
if (base::FeatureList::IsEnabled(kTrimWebCacheOnMemoryPressureOnly)) {
memory_pressure_listener_.emplace(
FROM_HERE,
base::BindRepeating(
[](WebCacheImpl* cache_impl,
base::MemoryPressureListener::MemoryPressureLevel level) {
if (level == base::MemoryPressureListener::MemoryPressureLevel::
MEMORY_PRESSURE_LEVEL_CRITICAL) {
cache_impl->ClearCache(false);
}
},
// Using unretained is safe because the memory pressure listener is
// owned by this object and so this can't cause an use-after-free.
base::Unretained(this)));
}
}
WebCacheImpl::~WebCacheImpl() = default;
void WebCacheImpl::BindReceiver(
mojo::PendingReceiver<mojom::WebCache> web_cache_receiver) {
receivers_.Add(this, std::move(web_cache_receiver));
}
void WebCacheImpl::ExecutePendingClearCache() {
switch (clear_cache_state_) {
case kInit:
clear_cache_state_ = kNavigate_Pending;
break;
case kNavigate_Pending:
break;
case kClearCache_Pending:
blink::WebCache::Clear();
clear_cache_state_ = kInit;
break;
}
}
void WebCacheImpl::SetCacheCapacity(uint64_t capacity64) {
DCHECK(!base::FeatureList::IsEnabled(
blink::features::kNoCentralWebCacheLimitControl));
size_t capacity = base::checked_cast<size_t>(capacity64);
blink::WebCache::SetCapacity(capacity);
}
void WebCacheImpl::ClearCache(bool on_navigation) {
if (!on_navigation) {
blink::WebCache::Clear();
return;
}
switch (clear_cache_state_) {
case kInit:
clear_cache_state_ = kClearCache_Pending;
break;
case kNavigate_Pending:
blink::WebCache::Clear();
clear_cache_state_ = kInit;
break;
case kClearCache_Pending:
break;
}
}
} // namespace web_cache