mirror of
https://github.com/LadybirdBrowser/ladybird
synced 2026-04-25 17:25:08 +02:00
LibWeb+LibWebView+WebContent: Implement versioning for document cookies
This patch introduces a cookie cache in the WebContent process to reduce blocking IPC calls when JS accesses document.cookie. The UI process now maintains a cookie version counter per-domain in shared memory. When JS reads document.cookie, we check whether we have a valid cached cookie by comparing the current shared version to the last used version. If they match, the cached cookie is returned without IPC. This optimization is based on Chromium's shared versioning, in which it was observed that 87% of document.cookie accesses were redundant. See: https://blog.chromium.org/2024/06/introducing-shared-memory-versioning-to.html Note that this cache only supports document.cookie, not HTTP Cookie headers. HTTP cookies are attached to requests with varying URLs and paths. The cookies that match the document URL might not match the request URL, which we wouldn't know from WebContent. So attaching the cached document cookie would be incorrect. On https://twinings.co.uk, we see approximately 600 document.cookie requests while the page loads. This patch reduces the time spent in the document.cookie getter from ~45ms to 2-3ms.
This commit is contained in:
Notes:
github-actions[bot]
2026-02-06 10:32:49 +00:00
Author: https://github.com/trflynn89 Commit: https://github.com/LadybirdBrowser/ladybird/commit/0482b6bb57b Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/7717 Reviewed-by: https://github.com/gmta
@@ -48,7 +48,8 @@ Optional<ViewImplementation&> ViewImplementation::find_view_by_id(u64 id)
|
||||
}
|
||||
|
||||
ViewImplementation::ViewImplementation()
|
||||
: m_view_id(s_view_count++)
|
||||
: m_document_cookie_version_buffer(Core::create_shared_version_buffer())
|
||||
, m_view_id(s_view_count++)
|
||||
{
|
||||
s_all_views.set(m_view_id, this);
|
||||
|
||||
@@ -259,9 +260,42 @@ void ViewImplementation::set_preferred_motion(Web::CSS::PreferredMotion motion)
|
||||
client().async_set_preferred_motion(page_id(), motion);
|
||||
}
|
||||
|
||||
void ViewImplementation::notify_cookies_changed(ReadonlySpan<Web::Cookie::Cookie> cookies)
|
||||
void ViewImplementation::notify_cookies_changed(HashTable<String> const& changed_domains, ReadonlySpan<Web::Cookie::Cookie> cookies)
|
||||
{
|
||||
client().async_cookies_changed(page_id(), cookies);
|
||||
for (auto const& domain : changed_domains) {
|
||||
if (auto document_index = m_document_cookie_version_indices.get(domain); document_index.has_value())
|
||||
Core::increment_shared_version(m_document_cookie_version_buffer, *document_index);
|
||||
}
|
||||
|
||||
if (!cookies.is_empty())
|
||||
client().async_cookies_changed(page_id(), cookies);
|
||||
}
|
||||
|
||||
ErrorOr<Core::SharedVersionIndex> ViewImplementation::ensure_document_cookie_version_index(Badge<WebContentClient>, String const& domain)
|
||||
{
|
||||
return m_document_cookie_version_indices.try_ensure(domain, [&]() -> ErrorOr<Core::SharedVersionIndex> {
|
||||
Core::SharedVersionIndex document_index = m_document_cookie_version_indices.size();
|
||||
|
||||
if (!Core::initialize_shared_version(m_document_cookie_version_buffer, document_index)) {
|
||||
dbgln("Reached maximum document cookie version count for {}, cannot create new version for {}", m_url, domain);
|
||||
return Error::from_string_literal("Reached maximum document cookie version count");
|
||||
}
|
||||
|
||||
return document_index;
|
||||
});
|
||||
}
|
||||
|
||||
Optional<Core::SharedVersion> ViewImplementation::document_cookie_version(URL::URL const& url) const
|
||||
{
|
||||
auto domain = Web::Cookie::canonicalize_domain(url);
|
||||
if (!domain.has_value())
|
||||
return {};
|
||||
|
||||
auto document_index = m_document_cookie_version_indices.get(*domain);
|
||||
if (!document_index.has_value())
|
||||
return {};
|
||||
|
||||
return Core::get_shared_version(m_document_cookie_version_buffer, *document_index);
|
||||
}
|
||||
|
||||
ByteString ViewImplementation::selected_text()
|
||||
@@ -592,6 +626,7 @@ void ViewImplementation::initialize_client(CreateNewClient create_new_client)
|
||||
client().async_set_device_pixel_ratio(m_client_state.page_index, m_device_pixel_ratio);
|
||||
client().async_set_maximum_frames_per_second(m_client_state.page_index, m_maximum_frames_per_second);
|
||||
client().async_set_system_visibility_state(m_client_state.page_index, m_system_visibility_state);
|
||||
client().async_set_document_cookie_version_buffer(m_client_state.page_index, m_document_cookie_version_buffer);
|
||||
|
||||
if (auto webdriver_content_ipc_path = Application::browser_options().webdriver_content_ipc_path; webdriver_content_ipc_path.has_value())
|
||||
client().async_connect_to_webdriver(m_client_state.page_index, *webdriver_content_ipc_path);
|
||||
|
||||
Reference in New Issue
Block a user