mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2026-06-29 04:30:24 +00:00
Enable -Wexit-time-destructors for all in-tree library targets and update process-lifetime library statics so they no longer register exit-time destructors. Long-lived caches, lookup tables, singleton registries, and generated constants now use NeverDestroyed or leaked references where the data is intended to live until process exit. Update LibWeb, LibLine, and the binding generators so regenerated sources follow the same rule instead of reintroducing destructed statics.
95 lines
2.8 KiB
C++
95 lines
2.8 KiB
C++
/*
|
|
* Copyright (c) 2022, Andreas Kling <andreas@ladybird.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/HashTable.h>
|
|
#include <AK/NeverDestroyed.h>
|
|
#include <LibWeb/DOM/QualifiedName.h>
|
|
|
|
namespace Web::DOM {
|
|
|
|
static unsigned hash_impl(FlyString const& local_name, Optional<FlyString> const& prefix, Optional<FlyString> const& namespace_)
|
|
{
|
|
unsigned hash = local_name.hash();
|
|
if (prefix.has_value())
|
|
hash = pair_int_hash(hash, prefix->hash());
|
|
if (namespace_.has_value())
|
|
hash = pair_int_hash(hash, namespace_->hash());
|
|
return hash;
|
|
}
|
|
|
|
struct ImplTraits : public Traits<QualifiedName::Impl*> {
|
|
static unsigned hash(QualifiedName::Impl* impl)
|
|
{
|
|
return hash_impl(impl->local_name, impl->prefix, impl->namespace_);
|
|
}
|
|
|
|
static bool equals(QualifiedName::Impl* a, QualifiedName::Impl* b)
|
|
{
|
|
return a->local_name == b->local_name
|
|
&& a->prefix == b->prefix
|
|
&& a->namespace_ == b->namespace_;
|
|
}
|
|
};
|
|
|
|
static HashTable<QualifiedName::Impl*, ImplTraits>& impls()
|
|
{
|
|
static NeverDestroyed<HashTable<QualifiedName::Impl*, ImplTraits>> impls;
|
|
return *impls;
|
|
}
|
|
|
|
static NonnullRefPtr<QualifiedName::Impl> ensure_impl(FlyString const& local_name, Optional<FlyString> const& prefix, Optional<FlyString> const& namespace_)
|
|
{
|
|
unsigned hash = hash_impl(local_name, prefix, namespace_);
|
|
|
|
auto it = impls().find(hash, [&](QualifiedName::Impl* entry) {
|
|
return entry->local_name == local_name
|
|
&& entry->prefix == prefix
|
|
&& entry->namespace_ == namespace_;
|
|
});
|
|
if (it != impls().end())
|
|
return *(*it);
|
|
return adopt_ref(*new QualifiedName::Impl(local_name, prefix, namespace_));
|
|
}
|
|
|
|
QualifiedName::QualifiedName(FlyString const& local_name, Optional<FlyString> const& prefix, Optional<FlyString> const& namespace_)
|
|
: m_impl(ensure_impl(local_name, prefix, namespace_))
|
|
{
|
|
}
|
|
|
|
QualifiedName::Impl::Impl(FlyString const& a_local_name, Optional<FlyString> const& a_prefix, Optional<FlyString> const& a_namespace)
|
|
: local_name(a_local_name)
|
|
, lowercased_local_name(local_name.to_ascii_lowercase())
|
|
, prefix(a_prefix)
|
|
, namespace_(a_namespace)
|
|
{
|
|
impls().set(this);
|
|
make_internal_string();
|
|
}
|
|
|
|
QualifiedName::Impl::~Impl()
|
|
{
|
|
impls().remove(this);
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#concept-attribute-qualified-name
|
|
// https://dom.spec.whatwg.org/#concept-element-qualified-name
|
|
void QualifiedName::Impl::make_internal_string()
|
|
{
|
|
// This is possible to do according to the spec: "User agents could have this as an internal slot as an optimization."
|
|
if (!prefix.has_value()) {
|
|
as_string = local_name;
|
|
return;
|
|
}
|
|
|
|
as_string = MUST(String::formatted("{}:{}", prefix.value(), local_name));
|
|
}
|
|
|
|
void QualifiedName::set_prefix(Optional<FlyString> value)
|
|
{
|
|
m_impl->prefix = move(value);
|
|
}
|
|
|
|
}
|