From ced222fdbabacc695f8f081c5b009afc9be6b8d0 Mon Sep 17 00:00:00 2001
From: Haled Odat <8566042+HalidOdat@users.noreply.github.com>
Date: Sun, 12 May 2024 21:48:06 +0200
Subject: [PATCH] Separate `JsString` into its own crate (#3837)
---
ABOUT.md | 2 +
Cargo.lock | 16 +-
Cargo.toml | 1 +
cli/ABOUT.md | 2 +
core/ast/ABOUT.md | 2 +
core/engine/ABOUT.md | 2 +
core/engine/Cargo.toml | 5 +-
core/engine/src/builtins/array/mod.rs | 2 +-
core/engine/src/builtins/array_buffer/mod.rs | 2 +-
.../src/builtins/array_buffer/shared.rs | 2 +-
.../engine/src/builtins/async_function/mod.rs | 2 +-
.../src/builtins/async_generator/mod.rs | 2 +-
.../builtins/async_generator_function/mod.rs | 2 +-
core/engine/src/builtins/atomics/mod.rs | 2 +-
core/engine/src/builtins/bigint/mod.rs | 2 +-
core/engine/src/builtins/boolean/mod.rs | 2 +-
core/engine/src/builtins/builder.rs | 2 +-
core/engine/src/builtins/dataview/mod.rs | 2 +-
core/engine/src/builtins/date/mod.rs | 2 +-
core/engine/src/builtins/error/aggregate.rs | 2 +-
core/engine/src/builtins/error/eval.rs | 2 +-
core/engine/src/builtins/error/mod.rs | 2 +-
core/engine/src/builtins/error/range.rs | 2 +-
core/engine/src/builtins/error/reference.rs | 2 +-
core/engine/src/builtins/error/syntax.rs | 2 +-
core/engine/src/builtins/error/type.rs | 2 +-
core/engine/src/builtins/error/uri.rs | 2 +-
core/engine/src/builtins/escape/mod.rs | 4 +-
core/engine/src/builtins/eval/mod.rs | 2 +-
core/engine/src/builtins/function/mod.rs | 2 +-
core/engine/src/builtins/generator/mod.rs | 2 +-
.../src/builtins/generator_function/mod.rs | 2 +-
core/engine/src/builtins/intl/collator/mod.rs | 2 +-
.../src/builtins/intl/date_time_format.rs | 2 +-
.../src/builtins/intl/list_format/mod.rs | 2 +-
core/engine/src/builtins/intl/locale/mod.rs | 2 +-
core/engine/src/builtins/intl/mod.rs | 2 +-
.../src/builtins/intl/number_format/mod.rs | 2 +-
.../src/builtins/intl/plural_rules/mod.rs | 2 +-
.../engine/src/builtins/intl/segmenter/mod.rs | 2 +-
core/engine/src/builtins/json/mod.rs | 2 +-
core/engine/src/builtins/map/mod.rs | 2 +-
core/engine/src/builtins/math/mod.rs | 4 +-
core/engine/src/builtins/number/globals.rs | 2 +-
core/engine/src/builtins/number/mod.rs | 2 +-
core/engine/src/builtins/object/mod.rs | 2 +-
core/engine/src/builtins/promise/mod.rs | 2 +-
core/engine/src/builtins/proxy/mod.rs | 2 +-
core/engine/src/builtins/reflect/mod.rs | 2 +-
core/engine/src/builtins/regexp/mod.rs | 2 +-
core/engine/src/builtins/set/mod.rs | 2 +-
core/engine/src/builtins/string/mod.rs | 19 +-
core/engine/src/builtins/symbol/mod.rs | 2 +-
.../src/builtins/temporal/calendar/mod.rs | 2 +-
.../src/builtins/temporal/duration/mod.rs | 2 +-
.../src/builtins/temporal/instant/mod.rs | 2 +-
core/engine/src/builtins/temporal/mod.rs | 2 +-
core/engine/src/builtins/temporal/now.rs | 2 +-
.../src/builtins/temporal/plain_date/mod.rs | 2 +-
.../builtins/temporal/plain_date_time/mod.rs | 2 +-
.../builtins/temporal/plain_month_day/mod.rs | 2 +-
.../src/builtins/temporal/plain_time/mod.rs | 2 +-
.../builtins/temporal/plain_year_month/mod.rs | 2 +-
.../src/builtins/temporal/time_zone/mod.rs | 2 +-
.../builtins/temporal/zoned_date_time/mod.rs | 2 +-
.../src/builtins/typed_array/builtin.rs | 2 +-
core/engine/src/builtins/typed_array/mod.rs | 2 +-
core/engine/src/builtins/uri/mod.rs | 2 +-
core/engine/src/builtins/weak/weak_ref.rs | 2 +-
core/engine/src/builtins/weak_map/mod.rs | 2 +-
core/engine/src/builtins/weak_set/mod.rs | 2 +-
core/engine/src/object/mod.rs | 2 +-
core/engine/src/object/operations.rs | 2 +-
core/engine/src/string.rs | 211 ++++
core/engine/src/string/common.rs | 950 -----------------
core/engine/src/symbol.rs | 2 +-
core/engine/src/tagged.rs | 1 +
core/engine/src/vm/opcode/push/array.rs | 2 +-
core/gc/ABOUT.md | 2 +
core/gc/Cargo.toml | 3 +
core/gc/src/trace.rs | 12 +
core/icu_provider/ABOUT.md | 2 +
core/interner/ABOUT.md | 2 +
core/interop/ABOUT.md | 2 +
core/macros/ABOUT.md | 2 +
core/parser/ABOUT.md | 2 +
core/profiler/ABOUT.md | 2 +
core/runtime/ABOUT.md | 2 +
core/string/ABOUT.md | 34 +
core/string/Cargo.toml | 28 +
core/string/src/common.rs | 951 ++++++++++++++++++
.../{engine/src/string => string/src}/iter.rs | 4 +
.../src/string/mod.rs => string/src/lib.rs} | 369 ++-----
core/{engine/src/string => string/src}/str.rs | 17 +-
core/string/src/tagged.rs | 109 ++
core/string/src/tests.rs | 166 +++
96 files changed, 1750 insertions(+), 1308 deletions(-)
create mode 100644 core/engine/src/string.rs
delete mode 100644 core/engine/src/string/common.rs
create mode 100644 core/string/ABOUT.md
create mode 100644 core/string/Cargo.toml
create mode 100644 core/string/src/common.rs
rename core/{engine/src/string => string/src}/iter.rs (97%)
rename core/{engine/src/string/mod.rs => string/src/lib.rs} (82%)
rename core/{engine/src/string => string/src}/str.rs (96%)
create mode 100644 core/string/src/tagged.rs
create mode 100644 core/string/src/tests.rs
diff --git a/ABOUT.md b/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/ABOUT.md
+++ b/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/Cargo.lock b/Cargo.lock
index 38894767745..2a012fc6025 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -395,6 +395,7 @@ dependencies = [
"boa_macros",
"boa_parser",
"boa_profiler",
+ "boa_string",
"bytemuck",
"cfg-if",
"criterion",
@@ -427,7 +428,6 @@ dependencies = [
"num-traits",
"num_enum",
"once_cell",
- "paste",
"pollster",
"portable-atomic",
"rand",
@@ -437,7 +437,6 @@ dependencies = [
"serde",
"serde_json",
"sptr",
- "static_assertions",
"sys-locale",
"tap",
"temporal_rs",
@@ -475,6 +474,7 @@ version = "0.18.0"
dependencies = [
"boa_macros",
"boa_profiler",
+ "boa_string",
"hashbrown 0.14.5",
"icu_locid",
"thin-vec",
@@ -571,6 +571,18 @@ dependencies = [
"textwrap",
]
+[[package]]
+name = "boa_string"
+version = "0.18.0"
+dependencies = [
+ "boa_macros",
+ "fast-float",
+ "paste",
+ "rustc-hash",
+ "sptr",
+ "static_assertions",
+]
+
[[package]]
name = "boa_tester"
version = "0.18.0"
diff --git a/Cargo.toml b/Cargo.toml
index 75650ecd0b9..ac1c3bf75d1 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -47,6 +47,7 @@ boa_macros = { version = "~0.18.0", path = "core/macros" }
boa_parser = { version = "~0.18.0", path = "core/parser" }
boa_profiler = { version = "~0.18.0", path = "core/profiler" }
boa_runtime = { version = "~0.18.0", path = "core/runtime" }
+boa_string = { version = "~0.18.0", path = "core/string" }
# Shared deps
arbitrary = "1"
diff --git a/cli/ABOUT.md b/cli/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/cli/ABOUT.md
+++ b/cli/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/ast/ABOUT.md b/core/ast/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/ast/ABOUT.md
+++ b/core/ast/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/engine/ABOUT.md b/core/engine/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/engine/ABOUT.md
+++ b/core/engine/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/engine/Cargo.toml b/core/engine/Cargo.toml
index 40883df1bd9..73502507c17 100644
--- a/core/engine/Cargo.toml
+++ b/core/engine/Cargo.toml
@@ -69,11 +69,12 @@ js = ["dep:web-time"]
[dependencies]
boa_interner.workspace = true
-boa_gc = { workspace = true, features = ["thin-vec"] }
+boa_gc = { workspace = true, features = ["thin-vec", "boa_string"] }
boa_profiler.workspace = true
boa_macros.workspace = true
boa_ast.workspace = true
boa_parser.workspace = true
+boa_string.workspace = true
serde = { workspace = true, features = ["derive", "rc"] }
serde_json.workspace = true
rand = "0.8.5"
@@ -89,7 +90,6 @@ fast-float.workspace = true
once_cell = { workspace = true, features = ["std"] }
tap = "1.0.1"
sptr = "0.3.2"
-static_assertions.workspace = true
thiserror = "1.0.59"
dashmap = "5.5.3"
num_enum = "0.7.2"
@@ -97,7 +97,6 @@ pollster.workspace = true
thin-vec.workspace = true
itertools = { version = "0.12.1", default-features = false }
icu_normalizer = { workspace = true, features = ["compiled_data"] }
-paste = "1.0"
portable-atomic = "1.6.0"
bytemuck = { version = "1.15.0", features = ["derive"] }
arrayvec = "0.7.4"
diff --git a/core/engine/src/builtins/array/mod.rs b/core/engine/src/builtins/array/mod.rs
index aa962470408..31472c605fc 100644
--- a/core/engine/src/builtins/array/mod.rs
+++ b/core/engine/src/builtins/array/mod.rs
@@ -32,7 +32,7 @@ use crate::{
},
property::{Attribute, PropertyDescriptor, PropertyKey, PropertyNameKind},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::{IntegerOrInfinity, JsValue},
Context, JsArgs, JsResult, JsString,
diff --git a/core/engine/src/builtins/array_buffer/mod.rs b/core/engine/src/builtins/array_buffer/mod.rs
index 1145c26f826..d5ea40ab554 100644
--- a/core/engine/src/builtins/array_buffer/mod.rs
+++ b/core/engine/src/builtins/array_buffer/mod.rs
@@ -29,7 +29,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject, Object},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsData, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/array_buffer/shared.rs b/core/engine/src/builtins/array_buffer/shared.rs
index 790a15eda0a..10c1092ee76 100644
--- a/core/engine/src/builtins/array_buffer/shared.rs
+++ b/core/engine/src/builtins/array_buffer/shared.rs
@@ -18,7 +18,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/builtins/async_function/mod.rs b/core/engine/src/builtins/async_function/mod.rs
index 865510b4ea0..f8db6134e9a 100644
--- a/core/engine/src/builtins/async_function/mod.rs
+++ b/core/engine/src/builtins/async_function/mod.rs
@@ -12,7 +12,7 @@ use crate::{
context::intrinsics::{Intrinsics, StandardConstructor, StandardConstructors},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/async_generator/mod.rs b/core/engine/src/builtins/async_generator/mod.rs
index 976dab93cfe..6fa423ddbdc 100644
--- a/core/engine/src/builtins/async_generator/mod.rs
+++ b/core/engine/src/builtins/async_generator/mod.rs
@@ -19,7 +19,7 @@ use crate::{
object::{FunctionObjectBuilder, JsObject, CONSTRUCTOR},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
vm::{CompletionRecord, GeneratorResumeKind},
diff --git a/core/engine/src/builtins/async_generator_function/mod.rs b/core/engine/src/builtins/async_generator_function/mod.rs
index 975bb842cd3..c0ca21b09ec 100644
--- a/core/engine/src/builtins/async_generator_function/mod.rs
+++ b/core/engine/src/builtins/async_generator_function/mod.rs
@@ -11,7 +11,7 @@ use crate::{
object::{JsObject, PROTOTYPE},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
Context, JsResult, JsString,
diff --git a/core/engine/src/builtins/atomics/mod.rs b/core/engine/src/builtins/atomics/mod.rs
index db74e6fbca8..2d4ff957aee 100644
--- a/core/engine/src/builtins/atomics/mod.rs
+++ b/core/engine/src/builtins/atomics/mod.rs
@@ -16,7 +16,7 @@ use std::sync::atomic::Ordering;
use crate::{
builtins::BuiltInObject, context::intrinsics::Intrinsics, js_string, object::JsObject,
- property::Attribute, realm::Realm, string::common::StaticJsStrings, symbol::JsSymbol,
+ property::Attribute, realm::Realm, string::StaticJsStrings, symbol::JsSymbol,
sys::time::Duration, value::IntegerOrInfinity, Context, JsArgs, JsNativeError, JsResult,
JsString, JsValue,
};
diff --git a/core/engine/src/builtins/bigint/mod.rs b/core/engine/src/builtins/bigint/mod.rs
index 3dc75356b18..947183afa54 100644
--- a/core/engine/src/builtins/bigint/mod.rs
+++ b/core/engine/src/builtins/bigint/mod.rs
@@ -20,7 +20,7 @@ use crate::{
object::JsObject,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::{IntegerOrInfinity, PreferredType},
Context, JsArgs, JsBigInt, JsResult, JsString, JsValue,
diff --git a/core/engine/src/builtins/boolean/mod.rs b/core/engine/src/builtins/boolean/mod.rs
index 3ae156b44b2..b33525a0238 100644
--- a/core/engine/src/builtins/boolean/mod.rs
+++ b/core/engine/src/builtins/boolean/mod.rs
@@ -19,7 +19,7 @@ use crate::{
js_string,
object::{internal_methods::get_prototype_from_constructor, JsObject},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsResult, JsString, JsValue,
};
use boa_profiler::Profiler;
diff --git a/core/engine/src/builtins/builder.rs b/core/engine/src/builtins/builder.rs
index c87b9ab7290..610a910db33 100644
--- a/core/engine/src/builtins/builder.rs
+++ b/core/engine/src/builtins/builder.rs
@@ -9,7 +9,7 @@ use crate::{
},
property::{Attribute, PropertyDescriptor, PropertyKey},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
JsObject, JsString, JsValue, NativeFunction,
};
diff --git a/core/engine/src/builtins/dataview/mod.rs b/core/engine/src/builtins/dataview/mod.rs
index 7274cc1fd94..dd8dfef3091 100644
--- a/core/engine/src/builtins/dataview/mod.rs
+++ b/core/engine/src/builtins/dataview/mod.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
Context, JsArgs, JsData, JsResult, JsString,
diff --git a/core/engine/src/builtins/date/mod.rs b/core/engine/src/builtins/date/mod.rs
index 49be23c5f88..17304c6e5ee 100644
--- a/core/engine/src/builtins/date/mod.rs
+++ b/core/engine/src/builtins/date/mod.rs
@@ -27,7 +27,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::{JsValue, PreferredType},
Context, JsArgs, JsData, JsError, JsResult, JsString,
diff --git a/core/engine/src/builtins/error/aggregate.rs b/core/engine/src/builtins/error/aggregate.rs
index bc4b9c32928..6a8042df31f 100644
--- a/core/engine/src/builtins/error/aggregate.rs
+++ b/core/engine/src/builtins/error/aggregate.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::{Attribute, PropertyDescriptorBuilder},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/error/eval.rs b/core/engine/src/builtins/error/eval.rs
index 266fd820530..88f6d21f0d1 100644
--- a/core/engine/src/builtins/error/eval.rs
+++ b/core/engine/src/builtins/error/eval.rs
@@ -18,7 +18,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/error/mod.rs b/core/engine/src/builtins/error/mod.rs
index ed005d0e380..39911f71156 100644
--- a/core/engine/src/builtins/error/mod.rs
+++ b/core/engine/src/builtins/error/mod.rs
@@ -18,7 +18,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsResult, JsString, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/error/range.rs b/core/engine/src/builtins/error/range.rs
index cc192396c33..15cf63d64f4 100644
--- a/core/engine/src/builtins/error/range.rs
+++ b/core/engine/src/builtins/error/range.rs
@@ -16,7 +16,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/error/reference.rs b/core/engine/src/builtins/error/reference.rs
index eb8a1158623..f56f6b403a7 100644
--- a/core/engine/src/builtins/error/reference.rs
+++ b/core/engine/src/builtins/error/reference.rs
@@ -16,7 +16,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/error/syntax.rs b/core/engine/src/builtins/error/syntax.rs
index 9074856694d..70333aaa10f 100644
--- a/core/engine/src/builtins/error/syntax.rs
+++ b/core/engine/src/builtins/error/syntax.rs
@@ -18,7 +18,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/error/type.rs b/core/engine/src/builtins/error/type.rs
index 3b3223d0fb0..364afba24dc 100644
--- a/core/engine/src/builtins/error/type.rs
+++ b/core/engine/src/builtins/error/type.rs
@@ -24,7 +24,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue, NativeFunction,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/error/uri.rs b/core/engine/src/builtins/error/uri.rs
index fcf7b1d208b..36e6f275bc8 100644
--- a/core/engine/src/builtins/error/uri.rs
+++ b/core/engine/src/builtins/error/uri.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsString, JsValue,
};
use boa_macros::js_str;
diff --git a/core/engine/src/builtins/escape/mod.rs b/core/engine/src/builtins/escape/mod.rs
index e3fcf1cb2b0..dc11cd0ae31 100644
--- a/core/engine/src/builtins/escape/mod.rs
+++ b/core/engine/src/builtins/escape/mod.rs
@@ -11,8 +11,8 @@
//! [spec]: https://tc39.es/ecma262/#sec-additional-properties-of-the-global-object
use crate::{
- context::intrinsics::Intrinsics, js_string, realm::Realm, string::common::StaticJsStrings,
- Context, JsArgs, JsObject, JsResult, JsString, JsValue,
+ context::intrinsics::Intrinsics, js_string, realm::Realm, string::StaticJsStrings, Context,
+ JsArgs, JsObject, JsResult, JsString, JsValue,
};
use super::{BuiltInBuilder, BuiltInObject, IntrinsicObject};
diff --git a/core/engine/src/builtins/eval/mod.rs b/core/engine/src/builtins/eval/mod.rs
index 2026208ab7e..8982c716c07 100644
--- a/core/engine/src/builtins/eval/mod.rs
+++ b/core/engine/src/builtins/eval/mod.rs
@@ -20,7 +20,7 @@ use crate::{
js_string,
object::JsObject,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
vm::{CallFrame, CallFrameFlags, Opcode},
Context, JsArgs, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/function/mod.rs b/core/engine/src/builtins/function/mod.rs
index 653a402ec77..1bf76b5e82c 100644
--- a/core/engine/src/builtins/function/mod.rs
+++ b/core/engine/src/builtins/function/mod.rs
@@ -30,7 +30,7 @@ use crate::{
},
property::{Attribute, PropertyDescriptor, PropertyKey},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::IntegerOrInfinity,
vm::{ActiveRunnable, CallFrame, CallFrameFlags, CodeBlock},
diff --git a/core/engine/src/builtins/generator/mod.rs b/core/engine/src/builtins/generator/mod.rs
index 3a9d56aa5b6..da8e38d181d 100644
--- a/core/engine/src/builtins/generator/mod.rs
+++ b/core/engine/src/builtins/generator/mod.rs
@@ -17,7 +17,7 @@ use crate::{
object::{JsObject, CONSTRUCTOR},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
vm::{CallFrame, CallFrameFlags, CompletionRecord, GeneratorResumeKind},
diff --git a/core/engine/src/builtins/generator_function/mod.rs b/core/engine/src/builtins/generator_function/mod.rs
index 300600d1ec9..c4687f1b051 100644
--- a/core/engine/src/builtins/generator_function/mod.rs
+++ b/core/engine/src/builtins/generator_function/mod.rs
@@ -16,7 +16,7 @@ use crate::{
object::PROTOTYPE,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
Context, JsResult, JsString,
diff --git a/core/engine/src/builtins/intl/collator/mod.rs b/core/engine/src/builtins/intl/collator/mod.rs
index a99aec57a68..edbb9314362 100644
--- a/core/engine/src/builtins/intl/collator/mod.rs
+++ b/core/engine/src/builtins/intl/collator/mod.rs
@@ -29,7 +29,7 @@ use crate::{
},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsData, JsNativeError, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/intl/date_time_format.rs b/core/engine/src/builtins/intl/date_time_format.rs
index 0dff2fe4dd8..f6dca66bfc7 100644
--- a/core/engine/src/builtins/intl/date_time_format.rs
+++ b/core/engine/src/builtins/intl/date_time_format.rs
@@ -17,7 +17,7 @@ use crate::{
js_string,
object::{internal_methods::get_prototype_from_constructor, JsObject},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsData, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/intl/list_format/mod.rs b/core/engine/src/builtins/intl/list_format/mod.rs
index c76889ffaed..6bc069570ef 100644
--- a/core/engine/src/builtins/intl/list_format/mod.rs
+++ b/core/engine/src/builtins/intl/list_format/mod.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsData, JsNativeError, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/intl/locale/mod.rs b/core/engine/src/builtins/intl/locale/mod.rs
index a70ab770f63..9c1afe0bd2a 100644
--- a/core/engine/src/builtins/intl/locale/mod.rs
+++ b/core/engine/src/builtins/intl/locale/mod.rs
@@ -1,4 +1,4 @@
-use crate::{builtins::options::get_option, realm::Realm, string::common::StaticJsStrings};
+use crate::{builtins::options::get_option, realm::Realm, string::StaticJsStrings};
use boa_macros::js_str;
use boa_profiler::Profiler;
use icu_collator::CaseFirst;
diff --git a/core/engine/src/builtins/intl/mod.rs b/core/engine/src/builtins/intl/mod.rs
index 149babe56b9..5f033a75682 100644
--- a/core/engine/src/builtins/intl/mod.rs
+++ b/core/engine/src/builtins/intl/mod.rs
@@ -20,7 +20,7 @@ use crate::{
object::JsObject,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsData, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/intl/number_format/mod.rs b/core/engine/src/builtins/intl/number_format/mod.rs
index 3b63fca375d..a94d8e56ccd 100644
--- a/core/engine/src/builtins/intl/number_format/mod.rs
+++ b/core/engine/src/builtins/intl/number_format/mod.rs
@@ -32,7 +32,7 @@ use crate::{
},
property::{Attribute, PropertyDescriptor},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
value::PreferredType,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
NativeFunction,
diff --git a/core/engine/src/builtins/intl/plural_rules/mod.rs b/core/engine/src/builtins/intl/plural_rules/mod.rs
index d0dad83b308..917e85c1917 100644
--- a/core/engine/src/builtins/intl/plural_rules/mod.rs
+++ b/core/engine/src/builtins/intl/plural_rules/mod.rs
@@ -21,7 +21,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, ObjectInitializer},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsStr, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/builtins/intl/segmenter/mod.rs b/core/engine/src/builtins/intl/segmenter/mod.rs
index 517e359b688..7353acfef5f 100644
--- a/core/engine/src/builtins/intl/segmenter/mod.rs
+++ b/core/engine/src/builtins/intl/segmenter/mod.rs
@@ -18,7 +18,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject, ObjectInitializer},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsResult, JsStr, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/builtins/json/mod.rs b/core/engine/src/builtins/json/mod.rs
index af6a4513baa..52e8debde26 100644
--- a/core/engine/src/builtins/json/mod.rs
+++ b/core/engine/src/builtins/json/mod.rs
@@ -27,7 +27,7 @@ use crate::{
object::{internal_methods::InternalMethodContext, JsObject},
property::{Attribute, PropertyNameKind},
realm::Realm,
- string::{common::StaticJsStrings, CodePoint},
+ string::{CodePoint, StaticJsStrings},
symbol::JsSymbol,
value::IntegerOrInfinity,
vm::{CallFrame, CallFrameFlags},
diff --git a/core/engine/src/builtins/map/mod.rs b/core/engine/src/builtins/map/mod.rs
index c5c4edee468..ef8eaad1e58 100644
--- a/core/engine/src/builtins/map/mod.rs
+++ b/core/engine/src/builtins/map/mod.rs
@@ -18,7 +18,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::{Attribute, PropertyNameKind},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/math/mod.rs b/core/engine/src/builtins/math/mod.rs
index 13d5748c071..424aed1e2dc 100644
--- a/core/engine/src/builtins/math/mod.rs
+++ b/core/engine/src/builtins/math/mod.rs
@@ -13,8 +13,8 @@
use crate::{
builtins::BuiltInObject, context::intrinsics::Intrinsics, js_string, object::JsObject,
- property::Attribute, realm::Realm, string::common::StaticJsStrings, symbol::JsSymbol, Context,
- JsArgs, JsResult, JsString, JsValue,
+ property::Attribute, realm::Realm, string::StaticJsStrings, symbol::JsSymbol, Context, JsArgs,
+ JsResult, JsString, JsValue,
};
use boa_profiler::Profiler;
diff --git a/core/engine/src/builtins/number/globals.rs b/core/engine/src/builtins/number/globals.rs
index 1900fa850c0..9564d4de924 100644
--- a/core/engine/src/builtins/number/globals.rs
+++ b/core/engine/src/builtins/number/globals.rs
@@ -3,7 +3,7 @@ use crate::{
context::intrinsics::Intrinsics,
object::JsObject,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsResult, JsStr, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/number/mod.rs b/core/engine/src/builtins/number/mod.rs
index 025c25acf6e..73345912782 100644
--- a/core/engine/src/builtins/number/mod.rs
+++ b/core/engine/src/builtins/number/mod.rs
@@ -21,7 +21,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
value::{AbstractRelation, IntegerOrInfinity, JsValue},
Context, JsArgs, JsResult, JsString,
};
diff --git a/core/engine/src/builtins/object/mod.rs b/core/engine/src/builtins/object/mod.rs
index a0fe6351913..92d2abe8352 100644
--- a/core/engine/src/builtins/object/mod.rs
+++ b/core/engine/src/builtins/object/mod.rs
@@ -28,7 +28,7 @@ use crate::{
},
property::{Attribute, PropertyDescriptor, PropertyKey, PropertyNameKind},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
Context, JsArgs, JsData, JsResult, JsString,
diff --git a/core/engine/src/builtins/promise/mod.rs b/core/engine/src/builtins/promise/mod.rs
index b3fc4d103fa..0578ad98eb4 100644
--- a/core/engine/src/builtins/promise/mod.rs
+++ b/core/engine/src/builtins/promise/mod.rs
@@ -17,7 +17,7 @@ use crate::{
},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
Context, JsArgs, JsError, JsResult, JsString,
diff --git a/core/engine/src/builtins/proxy/mod.rs b/core/engine/src/builtins/proxy/mod.rs
index ea0d0329107..ff6327c63c5 100644
--- a/core/engine/src/builtins/proxy/mod.rs
+++ b/core/engine/src/builtins/proxy/mod.rs
@@ -26,7 +26,7 @@ use crate::{
},
property::{PropertyDescriptor, PropertyKey},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
value::Type,
Context, JsArgs, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/reflect/mod.rs b/core/engine/src/builtins/reflect/mod.rs
index f4d405adf76..841a24d4b3d 100644
--- a/core/engine/src/builtins/reflect/mod.rs
+++ b/core/engine/src/builtins/reflect/mod.rs
@@ -19,7 +19,7 @@ use crate::{
object::{internal_methods::InternalMethodContext, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/regexp/mod.rs b/core/engine/src/builtins/regexp/mod.rs
index 34be1ef9654..beb782af1ac 100644
--- a/core/engine/src/builtins/regexp/mod.rs
+++ b/core/engine/src/builtins/regexp/mod.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject, CONSTRUCTOR},
property::Attribute,
realm::Realm,
- string::{common::StaticJsStrings, CodePoint, JsStrVariant},
+ string::{CodePoint, JsStrVariant, StaticJsStrings},
symbol::JsSymbol,
value::JsValue,
Context, JsArgs, JsData, JsResult, JsString,
diff --git a/core/engine/src/builtins/set/mod.rs b/core/engine/src/builtins/set/mod.rs
index a82f8c973fc..2927a9e8466 100644
--- a/core/engine/src/builtins/set/mod.rs
+++ b/core/engine/src/builtins/set/mod.rs
@@ -26,7 +26,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::{Attribute, PropertyNameKind},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/string/mod.rs b/core/engine/src/builtins/string/mod.rs
index a49f08a1866..f0a3d8ce34b 100644
--- a/core/engine/src/builtins/string/mod.rs
+++ b/core/engine/src/builtins/string/mod.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::{Attribute, PropertyDescriptor},
realm::Realm,
- string::{common::StaticJsStrings, CodePoint},
+ string::{CodePoint, StaticJsStrings},
symbol::JsSymbol,
value::IntegerOrInfinity,
Context, JsArgs, JsResult, JsString, JsValue,
@@ -73,23 +73,6 @@ pub(crate) const fn is_trimmable_whitespace(c: char) -> bool {
)
}
-/// Helper function to check if a `u8` latin1 character is trimmable.
-pub(crate) const fn is_trimmable_whitespace_latin1(c: u8) -> bool {
- // The rust implementation of `trim` does not regard the same characters whitespace as ecma standard does
- //
- // Rust uses \p{White_Space} by default, which also includes:
- // `\u{0085}' (next line)
- // And does not include:
- // '\u{FEFF}' (zero width non-breaking space)
- // Explicit whitespace: https://tc39.es/ecma262/#sec-white-space
- matches!(
- c,
- 0x09 | 0x0B | 0x0C | 0x20 | 0xA0 |
- // Line terminators: https://tc39.es/ecma262/#sec-line-terminators
- 0x0A | 0x0D
- )
-}
-
/// JavaScript `String` implementation.
#[derive(Debug, Clone, Copy)]
pub(crate) struct String;
diff --git a/core/engine/src/builtins/symbol/mod.rs b/core/engine/src/builtins/symbol/mod.rs
index dd50afb1048..ca18aee702a 100644
--- a/core/engine/src/builtins/symbol/mod.rs
+++ b/core/engine/src/builtins/symbol/mod.rs
@@ -28,7 +28,7 @@ use crate::{
object::JsObject,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::JsValue,
Context, JsArgs, JsResult, JsString,
diff --git a/core/engine/src/builtins/temporal/calendar/mod.rs b/core/engine/src/builtins/temporal/calendar/mod.rs
index 2da4335a92b..489e8f0163d 100644
--- a/core/engine/src/builtins/temporal/calendar/mod.rs
+++ b/core/engine/src/builtins/temporal/calendar/mod.rs
@@ -18,7 +18,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{custom_trace, Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/duration/mod.rs b/core/engine/src/builtins/temporal/duration/mod.rs
index 8e242ae44ae..f22ef7599c1 100644
--- a/core/engine/src/builtins/temporal/duration/mod.rs
+++ b/core/engine/src/builtins/temporal/duration/mod.rs
@@ -10,7 +10,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/instant/mod.rs b/core/engine/src/builtins/temporal/instant/mod.rs
index 48335b5db36..d4c927e4ca4 100644
--- a/core/engine/src/builtins/temporal/instant/mod.rs
+++ b/core/engine/src/builtins/temporal/instant/mod.rs
@@ -14,7 +14,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsBigInt, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol,
JsValue,
};
diff --git a/core/engine/src/builtins/temporal/mod.rs b/core/engine/src/builtins/temporal/mod.rs
index adb26887809..d8aa749aaaf 100644
--- a/core/engine/src/builtins/temporal/mod.rs
+++ b/core/engine/src/builtins/temporal/mod.rs
@@ -33,7 +33,7 @@ use crate::{
js_string,
property::{Attribute, PropertyKey},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
value::Type,
Context, JsBigInt, JsError, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/builtins/temporal/now.rs b/core/engine/src/builtins/temporal/now.rs
index 4e13a3424d8..05782441442 100644
--- a/core/engine/src/builtins/temporal/now.rs
+++ b/core/engine/src/builtins/temporal/now.rs
@@ -9,7 +9,7 @@ use crate::{
js_string,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
sys::time::SystemTime,
Context, JsBigInt, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/builtins/temporal/plain_date/mod.rs b/core/engine/src/builtins/temporal/plain_date/mod.rs
index aac77d7e019..558d8121a67 100644
--- a/core/engine/src/builtins/temporal/plain_date/mod.rs
+++ b/core/engine/src/builtins/temporal/plain_date/mod.rs
@@ -13,7 +13,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/plain_date_time/mod.rs b/core/engine/src/builtins/temporal/plain_date_time/mod.rs
index 462b1caf338..55e4d815621 100644
--- a/core/engine/src/builtins/temporal/plain_date_time/mod.rs
+++ b/core/engine/src/builtins/temporal/plain_date_time/mod.rs
@@ -11,7 +11,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/plain_month_day/mod.rs b/core/engine/src/builtins/temporal/plain_month_day/mod.rs
index 3cdc17652bf..62e483ab2bd 100644
--- a/core/engine/src/builtins/temporal/plain_month_day/mod.rs
+++ b/core/engine/src/builtins/temporal/plain_month_day/mod.rs
@@ -6,7 +6,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/plain_time/mod.rs b/core/engine/src/builtins/temporal/plain_time/mod.rs
index 846b23cc63c..adba3b53ed6 100644
--- a/core/engine/src/builtins/temporal/plain_time/mod.rs
+++ b/core/engine/src/builtins/temporal/plain_time/mod.rs
@@ -10,7 +10,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/plain_year_month/mod.rs b/core/engine/src/builtins/temporal/plain_year_month/mod.rs
index c651534b5d4..4b06c7ae46c 100644
--- a/core/engine/src/builtins/temporal/plain_year_month/mod.rs
+++ b/core/engine/src/builtins/temporal/plain_year_month/mod.rs
@@ -7,7 +7,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/time_zone/mod.rs b/core/engine/src/builtins/temporal/time_zone/mod.rs
index 826cf94bb52..a1e35e93e6a 100644
--- a/core/engine/src/builtins/temporal/time_zone/mod.rs
+++ b/core/engine/src/builtins/temporal/time_zone/mod.rs
@@ -11,7 +11,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, CONSTRUCTOR},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsArgs, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{custom_trace, Finalize, Trace};
diff --git a/core/engine/src/builtins/temporal/zoned_date_time/mod.rs b/core/engine/src/builtins/temporal/zoned_date_time/mod.rs
index d95f7de02f3..3c0e8691d50 100644
--- a/core/engine/src/builtins/temporal/zoned_date_time/mod.rs
+++ b/core/engine/src/builtins/temporal/zoned_date_time/mod.rs
@@ -4,7 +4,7 @@ use crate::{
context::intrinsics::{Intrinsics, StandardConstructor, StandardConstructors},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsBigInt, JsData, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
use boa_gc::{custom_trace, Finalize, Trace};
diff --git a/core/engine/src/builtins/typed_array/builtin.rs b/core/engine/src/builtins/typed_array/builtin.rs
index 9119edd361c..faa4961e6e1 100644
--- a/core/engine/src/builtins/typed_array/builtin.rs
+++ b/core/engine/src/builtins/typed_array/builtin.rs
@@ -24,7 +24,7 @@ use crate::{
object::internal_methods::get_prototype_from_constructor,
property::{Attribute, PropertyNameKind},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
value::IntegerOrInfinity,
Context, JsArgs, JsNativeError, JsObject, JsResult, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/builtins/typed_array/mod.rs b/core/engine/src/builtins/typed_array/mod.rs
index 15f63e0fe37..8bbdf820a6d 100644
--- a/core/engine/src/builtins/typed_array/mod.rs
+++ b/core/engine/src/builtins/typed_array/mod.rs
@@ -23,7 +23,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
value::{JsValue, Numeric},
Context, JsArgs, JsResult, JsString,
diff --git a/core/engine/src/builtins/uri/mod.rs b/core/engine/src/builtins/uri/mod.rs
index 3389a36f6db..0e7fb3d4764 100644
--- a/core/engine/src/builtins/uri/mod.rs
+++ b/core/engine/src/builtins/uri/mod.rs
@@ -25,7 +25,7 @@ use crate::{
js_string,
object::{JsFunction, JsObject},
realm::Realm,
- string::{common::StaticJsStrings, CodePoint},
+ string::{CodePoint, StaticJsStrings},
Context, JsArgs, JsNativeError, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/weak/weak_ref.rs b/core/engine/src/builtins/weak/weak_ref.rs
index 8442e6ac1bc..a19ec15a7cc 100644
--- a/core/engine/src/builtins/weak/weak_ref.rs
+++ b/core/engine/src/builtins/weak/weak_ref.rs
@@ -8,7 +8,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, ErasedVTableObject, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsNativeError, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/weak_map/mod.rs b/core/engine/src/builtins/weak_map/mod.rs
index 01cd63c0a98..57e2bd588e9 100644
--- a/core/engine/src/builtins/weak_map/mod.rs
+++ b/core/engine/src/builtins/weak_map/mod.rs
@@ -17,7 +17,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, ErasedVTableObject, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsNativeError, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/builtins/weak_set/mod.rs b/core/engine/src/builtins/weak_set/mod.rs
index d90a8257725..493051b873a 100644
--- a/core/engine/src/builtins/weak_set/mod.rs
+++ b/core/engine/src/builtins/weak_set/mod.rs
@@ -14,7 +14,7 @@ use crate::{
object::{internal_methods::get_prototype_from_constructor, ErasedVTableObject, JsObject},
property::Attribute,
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
symbol::JsSymbol,
Context, JsArgs, JsNativeError, JsResult, JsString, JsValue,
};
diff --git a/core/engine/src/object/mod.rs b/core/engine/src/object/mod.rs
index ffffaa5579f..b6696e6e665 100644
--- a/core/engine/src/object/mod.rs
+++ b/core/engine/src/object/mod.rs
@@ -23,7 +23,7 @@ use crate::{
native_function::{NativeFunction, NativeFunctionObject},
property::{Attribute, PropertyDescriptor, PropertyKey},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
Context, JsStr, JsString, JsSymbol, JsValue,
};
diff --git a/core/engine/src/object/operations.rs b/core/engine/src/object/operations.rs
index 6cee0140e11..0301d6c10b6 100644
--- a/core/engine/src/object/operations.rs
+++ b/core/engine/src/object/operations.rs
@@ -9,7 +9,7 @@ use crate::{
object::{JsObject, PrivateElement, PrivateName, CONSTRUCTOR, PROTOTYPE},
property::{PropertyDescriptor, PropertyDescriptorBuilder, PropertyKey, PropertyNameKind},
realm::Realm,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
value::Type,
Context, JsResult, JsSymbol, JsValue,
};
diff --git a/core/engine/src/string.rs b/core/engine/src/string.rs
new file mode 100644
index 00000000000..cfddb9328ce
--- /dev/null
+++ b/core/engine/src/string.rs
@@ -0,0 +1,211 @@
+//! This module contains the [`js_string`][crate::js_string] macro and the
+//! [`js_str`][crate::js_str] macro.
+//!
+//! The [`js_string`][crate::js_string] macro is used when you need to create a new [`JsString`],
+//! and the [`js_str`][crate::js_str] macro is used for const conversions of string literals to [`JsStr`].
+
+#[doc(inline)]
+pub use boa_string::*;
+
+/// Utility macro to create a [`JsString`].
+///
+/// # Examples
+///
+/// You can call the macro without arguments to create an empty `JsString`:
+///
+/// ```
+/// use boa_engine::js_string;
+///
+/// let empty_str = js_string!();
+/// assert!(empty_str.is_empty());
+/// ```
+///
+///
+/// You can create a `JsString` from a string literal, which completely skips the runtime
+/// conversion from [`&str`] to [&\[u16\]][slice]
:
+///
+/// ```
+/// # use boa_engine::js_string;
+/// let hw = js_string!("Hello, world!");
+/// assert_eq!(&hw, "Hello, world!");
+/// ```
+///
+/// Any `&[u16]` slice is a valid `JsString`, including unpaired surrogates:
+///
+/// ```
+/// # use boa_engine::js_string;
+/// let array = js_string!(&[0xD8AFu16, 0x00A0, 0xD8FF, 0x00F0]);
+/// ```
+///
+/// You can also pass it any number of `&[u16]` as arguments to create a new `JsString` with
+/// the concatenation of every slice:
+///
+/// ```
+/// # use boa_engine::{js_string, js_str, JsStr};
+/// const NAME: JsStr<'_> = js_str!("human! ");
+/// let greeting = js_string!("Hello, ");
+/// let msg = js_string!(&greeting, NAME, js_str!("Nice to meet you!"));
+///
+/// assert_eq!(&msg, "Hello, human! Nice to meet you!");
+/// ```
+#[macro_export]
+#[allow(clippy::module_name_repetitions)]
+macro_rules! js_string {
+ () => {
+ $crate::string::JsString::default()
+ };
+ ($s:literal) => {
+ $crate::string::JsString::from($crate::js_str!($s))
+ };
+ ($s:expr) => {
+ $crate::string::JsString::from($s)
+ };
+ ( $x:expr, $y:expr ) => {
+ $crate::string::JsString::concat($crate::string::JsStr::from($x), $crate::string::JsStr::from($y))
+ };
+ ( $( $s:expr ),+ ) => {
+ $crate::string::JsString::concat_array(&[ $( $crate::string::JsStr::from($s) ),+ ])
+ };
+}
+
+#[allow(clippy::redundant_clone)]
+#[cfg(test)]
+mod tests {
+ use std::hash::{BuildHasher, BuildHasherDefault, Hash};
+
+ use crate::{string::StaticJsStrings, JsStr};
+
+ use super::JsString;
+ use boa_macros::{js_str, utf16};
+ use rustc_hash::FxHasher;
+
+ fn hash_value(value: &T) -> u64 {
+ BuildHasherDefault::::default().hash_one(value)
+ }
+
+ #[test]
+ fn empty() {
+ let s = js_string!();
+ assert_eq!(&s, utf16!(""));
+ }
+
+ #[test]
+ fn refcount() {
+ let x = js_string!("Hello world");
+ assert_eq!(x.refcount(), Some(1));
+
+ {
+ let y = x.clone();
+ assert_eq!(x.refcount(), Some(2));
+ assert_eq!(y.refcount(), Some(2));
+
+ {
+ let z = y.clone();
+ assert_eq!(x.refcount(), Some(3));
+ assert_eq!(y.refcount(), Some(3));
+ assert_eq!(z.refcount(), Some(3));
+ }
+
+ assert_eq!(x.refcount(), Some(2));
+ assert_eq!(y.refcount(), Some(2));
+ }
+
+ assert_eq!(x.refcount(), Some(1));
+ }
+
+ #[test]
+ fn static_refcount() {
+ let x = js_string!();
+ assert_eq!(x.refcount(), None);
+
+ {
+ let y = x.clone();
+ assert_eq!(x.refcount(), None);
+ assert_eq!(y.refcount(), None);
+ };
+
+ assert_eq!(x.refcount(), None);
+ }
+
+ #[test]
+ fn as_str() {
+ const HELLO: &[u16] = utf16!("Hello");
+ let x = js_string!(HELLO);
+
+ assert_eq!(&x, HELLO);
+ }
+
+ #[test]
+ fn hash() {
+ const HELLOWORLD: JsStr<'_> = js_str!("Hello World!");
+ let x = js_string!(HELLOWORLD);
+
+ assert_eq!(x.as_str(), HELLOWORLD);
+
+ assert!(HELLOWORLD.is_latin1());
+ assert!(x.as_str().is_latin1());
+
+ let s_hash = hash_value(&HELLOWORLD);
+ let x_hash = hash_value(&x);
+
+ assert_eq!(s_hash, x_hash);
+ }
+
+ #[test]
+ fn concat() {
+ const Y: &[u16] = utf16!(", ");
+ const W: &[u16] = utf16!("!");
+
+ let x = js_string!("hello");
+ let z = js_string!("world");
+
+ let xy = js_string!(&x, &JsString::from(Y));
+ assert_eq!(&xy, utf16!("hello, "));
+ assert_eq!(xy.refcount(), Some(1));
+
+ let xyz = js_string!(&xy, &z);
+ assert_eq!(&xyz, utf16!("hello, world"));
+ assert_eq!(xyz.refcount(), Some(1));
+
+ let xyzw = js_string!(&xyz, &JsString::from(W));
+ assert_eq!(&xyzw, utf16!("hello, world!"));
+ assert_eq!(xyzw.refcount(), Some(1));
+ }
+
+ #[test]
+ fn trim_start_non_ascii_to_ascii() {
+ let s = "\u{2029}abc";
+ let x = js_string!(s);
+
+ let y = js_string!(x.trim_start());
+
+ assert_eq!(&y, s.trim_start());
+ }
+
+ #[test]
+ fn conversion_to_known_static_js_string() {
+ const JS_STR_U8: &JsStr<'_> = &js_str!("length");
+ const JS_STR_U16: &JsStr<'_> = &JsStr::utf16(utf16!("length"));
+
+ assert!(JS_STR_U8.is_latin1());
+ assert!(!JS_STR_U16.is_latin1());
+
+ assert_eq!(JS_STR_U8, JS_STR_U8);
+ assert_eq!(JS_STR_U16, JS_STR_U16);
+
+ assert_eq!(JS_STR_U8, JS_STR_U16);
+ assert_eq!(JS_STR_U16, JS_STR_U8);
+
+ assert_eq!(hash_value(JS_STR_U8), hash_value(JS_STR_U16));
+
+ let string = StaticJsStrings::get_string(JS_STR_U8);
+
+ assert!(string.is_some());
+ assert!(string.unwrap().as_str().is_latin1());
+
+ let string = StaticJsStrings::get_string(JS_STR_U16);
+
+ assert!(string.is_some());
+ assert!(string.unwrap().as_str().is_latin1());
+ }
+}
diff --git a/core/engine/src/string/common.rs b/core/engine/src/string/common.rs
deleted file mode 100644
index ba7785d7f63..00000000000
--- a/core/engine/src/string/common.rs
+++ /dev/null
@@ -1,950 +0,0 @@
-//! List of commonly used strings in Javascript code.
-
-use crate::{tagged::Tagged, JsStr};
-
-use super::JsString;
-use boa_macros::js_str;
-use paste::paste;
-use rustc_hash::{FxHashMap, FxHasher};
-use std::hash::BuildHasherDefault;
-
-macro_rules! well_known_statics {
- ( $( $(#[$attr:meta])* ($name:ident, $string:literal) ),+$(,)? ) => {
- $(
- paste!{
- #[doc = "Gets the static `JsString` for `\"" $string "\"`."]
- pub const $name: JsString = JsString {
- ptr: Tagged::from_tag(
- Self::find_index($string),
- ),
- };
- }
- )+
- };
-}
-
-/// List of commonly used strings in Javascript code.
-///
-/// Any strings defined here are used as a static [`JsString`] instead of allocating on the heap.
-#[derive(Debug, Clone, Copy)]
-pub struct StaticJsStrings;
-
-impl StaticJsStrings {
- // useful to search at compile time a certain string in the array
- const fn find_index(candidate: &str) -> usize {
- const fn const_eq(lhs: &[u8], rhs: &[u8]) -> bool {
- if lhs.len() != rhs.len() {
- return false;
- }
-
- let mut i = 0;
- while i < lhs.len() {
- if lhs[i] != rhs[i] {
- return false;
- }
- i += 1;
- }
- true
- }
-
- let len = RAW_STATICS.len();
- let mut i = 0;
- while i < len {
- let Some(s) = RAW_STATICS[i].as_latin1() else {
- // All static strings are latin1 encoded
- unreachable!()
- };
- if const_eq(s, candidate.as_bytes()) {
- return i;
- }
- i += 1;
- }
-
- panic!("couldn't find the required string on the common string array");
- }
-
- /// Gets the `JsString` corresponding to `string`, or `None` if the string
- /// doesn't exist inside the static array.
- pub(crate) fn get_string(string: &JsStr<'_>) -> Option {
- if string.len() > MAX_STATIC_LENGTH {
- return None;
- }
-
- let index = RAW_STATICS_CACHE.with(|map| map.get(string).copied())?;
-
- Some(JsString {
- ptr: Tagged::from_tag(index),
- })
- }
-
- /// Gets the `&[u16]` slice corresponding to the provided index, or `None` if the index
- /// provided exceeds the size of the static array.
- pub(crate) fn get(index: usize) -> Option> {
- RAW_STATICS.get(index).copied()
- }
-
- // Some consts are only used on certain features, which triggers the unused lint.
- well_known_statics! {
- (EMPTY_STRING, ""),
- (LENGTH, "length"),
- // Symbols
- (SYMBOL_ASYNC_ITERATOR, "Symbol.asyncIterator"),
- (SYMBOL_HAS_INSTANCE, "Symbol.hasInstance"),
- (SYMBOL_IS_CONCAT_SPREADABLE, "Symbol.isConcatSpreadable"),
- (SYMBOL_ITERATOR, "Symbol.iterator"),
- (SYMBOL_MATCH, "Symbol.match"),
- (SYMBOL_MATCH_ALL, "Symbol.matchAll"),
- (SYMBOL_REPLACE, "Symbol.replace"),
- (SYMBOL_SEARCH, "Symbol.search"),
- (SYMBOL_SPECIES, "Symbol.species"),
- (SYMBOL_SPLIT, "Symbol.split"),
- (SYMBOL_TO_PRIMITIVE, "Symbol.toPrimitive"),
- (SYMBOL_TO_STRING_TAG, "Symbol.toStringTag"),
- (SYMBOL_UNSCOPABLES, "Symbol.unscopables"),
- (FN_SYMBOL_ASYNC_ITERATOR, "[Symbol.asyncIterator]"),
- (FN_SYMBOL_HAS_INSTANCE, "[Symbol.hasInstance]"),
- (FN_SYMBOL_IS_CONCAT_SPREADABLE, "[Symbol.isConcatSpreadable]"),
- (FN_SYMBOL_ITERATOR, "[Symbol.iterator]"),
- (FN_SYMBOL_MATCH, "[Symbol.match]"),
- (FN_SYMBOL_MATCH_ALL, "[Symbol.matchAll]"),
- (FN_SYMBOL_REPLACE, "[Symbol.replace]"),
- (FN_SYMBOL_SEARCH, "[Symbol.search]"),
- (FN_SYMBOL_SPECIES, "[Symbol.species]"),
- (FN_SYMBOL_SPLIT, "[Symbol.split]"),
- (FN_SYMBOL_TO_PRIMITIVE, "[Symbol.toPrimitive]"),
- (FN_SYMBOL_TO_STRING_TAG, "[Symbol.toStringTag]"),
- (FN_SYMBOL_UNSCOPABLES, "[Symbol.unscopables]"),
- // Builtins
- (ARRAY, "Array"),
- (ARRAY_BUFFER, "ArrayBuffer"),
- (SHARED_ARRAY_BUFFER, "SharedArrayBuffer"),
- (ASYNC_FUNCTION, "AsyncFunction"),
- (ASYNC_GENERATOR, "AsyncGenerator"),
- (ASYNC_GENERATOR_FUNCTION, "AsyncGeneratorFunction"),
- (ATOMICS, "Atomics"),
- (BIG_INT, "BigInt"),
- (BOOLEAN, "Boolean"),
- (DATA_VIEW, "DataView"),
- (DATE, "Date"),
- (ERROR, "Error"),
- (AGGREGATE_ERROR, "AggregateError"),
- (EVAL_ERROR, "EvalError"),
- (RANGE_ERROR, "RangeError"),
- (REFERENCE_ERROR, "ReferenceError"),
- (SYNTAX_ERROR, "SyntaxError"),
- (TYPE_ERROR, "TypeError"),
- (URI_ERROR, "URIError"),
- (ESCAPE, "escape"),
- (UNESCAPE, "unescape"),
- (EVAL, "eval"),
- (FUNCTION, "Function"),
- (GENERATOR, "Generator"),
- (GENERATOR_FUNCTION, "GeneratorFunction"),
- (INTL, "Intl"),
- (COLLATOR, "Collator"),
- (LIST_FORMAT, "ListFormat"),
- (LOCALE, "Locale"),
- (PLURAL_RULES, "PluralRules"),
- (SEGMENTER, "Segmenter"),
- (DATE_TIME_FORMAT, "DateTimeFormat"),
- (JSON, "JSON"),
- (MAP, "Map"),
- (MATH, "Math"),
- (NUMBER, "Number"),
- (NUMBER_FORMAT, "NumberFormat"),
- (IS_FINITE, "isFinite"),
- (IS_NAN, "isNaN"),
- (PARSE_INT, "parseInt"),
- (PARSE_FLOAT, "parseFloat"),
- (OBJECT, "Object"),
- (PROMISE, "Promise"),
- (PROXY, "Proxy"),
- (REFLECT, "Reflect"),
- (REG_EXP, "RegExp"),
- (SET, "Set"),
- (STRING, "String"),
- (SYMBOL, "Symbol"),
- (TYPED_ARRAY, "TypedArray"),
- (INT8_ARRAY, "Int8Array"),
- (UINT8_ARRAY, "Uint8Array"),
- (UINT8_CLAMPED_ARRAY, "Uint8ClampedArray"),
- (INT16_ARRAY, "Int16Array"),
- (UINT16_ARRAY, "Uint16Array"),
- (INT32_ARRAY, "Int32Array"),
- (UINT32_ARRAY, "Uint32Array"),
- (BIG_INT64_ARRAY, "BigInt64Array"),
- (BIG_UINT64_ARRAY, "BigUint64Array"),
- (FLOAT32_ARRAY, "Float32Array"),
- (FLOAT64_ARRAY, "Float64Array"),
- (ENCODE_URI, "encodeURI"),
- (ENCODE_URI_COMPONENT, "encodeURIComponent"),
- (DECODE_URI, "decodeURI"),
- (DECODE_URI_COMPONENT, "decodeURIComponent"),
- (WEAK_REF, "WeakRef"),
- (WEAK_MAP, "WeakMap"),
- (WEAK_SET, "WeakSet"),
- (TEMPORAL, "Temporal"),
- (NOW, "Temporal.Now"),
- (INSTANT, "Temporal.Instant"),
- (DURATION, "Temporal.Duration"),
- (PLAIN_DATE, "Temporal.PlainDate"),
- (PLAIN_DATETIME, "Temporal.PlainDateTime"),
- (PLAIN_TIME, "Temporal.PlainTime"),
- (PLAIN_YM, "Temporal.PlainYearMonth"),
- (PLAIN_MD, "Temporal.PlainMonthDay"),
- (CALENDAR, "Temporal.Calendar"),
- (TIMEZONE, "Temporal.TimeZone"),
- (ZONED_DT, "Temporal.ZonedDateTime"),
- }
-}
-
-const MAX_STATIC_LENGTH: usize = {
- let mut max = 0;
- let mut i = 0;
- while i < RAW_STATICS.len() {
- // TOOD: Because `get_index` is not const, we are accessing doc hidden stuff, that may change.
- let len = RAW_STATICS[i].len();
- if len > max {
- max = len;
- }
- i += 1;
- }
- max
-};
-
-thread_local! {
- /// Map from a string inside [`RAW_STATICS`] to its corresponding static index on `RAW_STATICS`.
- static RAW_STATICS_CACHE: FxHashMap, usize> = {
- let mut constants = FxHashMap::with_capacity_and_hasher(
- RAW_STATICS.len(),
- BuildHasherDefault::::default(),
- );
-
- for (idx, &s) in RAW_STATICS.iter().enumerate() {
- constants.insert(s, idx);
- }
-
- constants
- };
-}
-
-/// Array of raw static strings that aren't reference counted.
-const RAW_STATICS: &[JsStr<'_>] = &[
- js_str!(""),
- // Well known symbols
- js_str!("Symbol.asyncIterator"),
- js_str!("[Symbol.asyncIterator]"),
- js_str!("Symbol.hasInstance"),
- js_str!("[Symbol.hasInstance]"),
- js_str!("Symbol.isConcatSpreadable"),
- js_str!("[Symbol.isConcatSpreadable]"),
- js_str!("Symbol.iterator"),
- js_str!("[Symbol.iterator]"),
- js_str!("Symbol.match"),
- js_str!("[Symbol.match]"),
- js_str!("Symbol.matchAll"),
- js_str!("[Symbol.matchAll]"),
- js_str!("Symbol.replace"),
- js_str!("[Symbol.replace]"),
- js_str!("Symbol.search"),
- js_str!("[Symbol.search]"),
- js_str!("Symbol.species"),
- js_str!("[Symbol.species]"),
- js_str!("Symbol.split"),
- js_str!("[Symbol.split]"),
- js_str!("Symbol.toPrimitive"),
- js_str!("[Symbol.toPrimitive]"),
- js_str!("Symbol.toStringTag"),
- js_str!("[Symbol.toStringTag]"),
- js_str!("Symbol.unscopables"),
- js_str!("[Symbol.unscopables]"),
- js_str!("get [Symbol.species]"),
- js_str!("get [Symbol.toStringTag]"),
- // Well known builtins
- js_str!("Array"),
- js_str!("ArrayBuffer"),
- js_str!("SharedArrayBuffer"),
- js_str!("AsyncFunction"),
- js_str!("AsyncGenerator"),
- js_str!("AsyncGeneratorFunction"),
- js_str!("Atomics"),
- js_str!("BigInt"),
- js_str!("Boolean"),
- js_str!("DataView"),
- js_str!("Date"),
- js_str!("Error"),
- js_str!("AggregateError"),
- js_str!("EvalError"),
- js_str!("RangeError"),
- js_str!("ReferenceError"),
- js_str!("SyntaxError"),
- js_str!("TypeError"),
- js_str!("URIError"),
- js_str!("escape"),
- js_str!("unescape"),
- js_str!("eval"),
- js_str!("Function"),
- js_str!("Generator"),
- js_str!("GeneratorFunction"),
- js_str!("Intl"),
- js_str!("Collator"),
- js_str!("ListFormat"),
- js_str!("Locale"),
- js_str!("PluralRules"),
- js_str!("Segmenter"),
- js_str!("DateTimeFormat"),
- js_str!("JSON"),
- js_str!("Map"),
- js_str!("Math"),
- js_str!("Number"),
- js_str!("NumberFormat"),
- js_str!("isFinite"),
- js_str!("isNaN"),
- js_str!("parseInt"),
- js_str!("parseFloat"),
- js_str!("Object"),
- js_str!("Promise"),
- js_str!("Proxy"),
- js_str!("Reflect"),
- js_str!("RegExp"),
- js_str!("Set"),
- js_str!("String"),
- js_str!("Symbol"),
- js_str!("TypedArray"),
- js_str!("Int8Array"),
- js_str!("Uint8Array"),
- js_str!("Uint8ClampedArray"),
- js_str!("Int16Array"),
- js_str!("Uint16Array"),
- js_str!("Int32Array"),
- js_str!("Uint32Array"),
- js_str!("BigInt64Array"),
- js_str!("BigUint64Array"),
- js_str!("Float32Array"),
- js_str!("Float64Array"),
- js_str!("encodeURI"),
- js_str!("encodeURIComponent"),
- js_str!("decodeURI"),
- js_str!("decodeURIComponent"),
- js_str!("WeakRef"),
- js_str!("WeakMap"),
- js_str!("WeakSet"),
- js_str!("Temporal"),
- js_str!("Temporal.Now"),
- js_str!("Temporal.Instant"),
- js_str!("Temporal.Duration"),
- js_str!("Temporal.Calendar"),
- js_str!("Temporal.PlainDate"),
- js_str!("Temporal.PlainDateTime"),
- js_str!("Temporal.PlainMonthDay"),
- js_str!("Temporal.PlainYearMonth"),
- js_str!("Temporal.PlainTime"),
- js_str!("Temporal.TimeZone"),
- js_str!("Temporal.ZonedDateTime"),
- // Misc
- js_str!(","),
- js_str!(":"),
- // Generic use
- js_str!("name"),
- js_str!("length"),
- js_str!("arguments"),
- js_str!("prototype"),
- js_str!("constructor"),
- js_str!("return"),
- js_str!("throw"),
- js_str!("global"),
- js_str!("globalThis"),
- // typeof
- js_str!("null"),
- js_str!("undefined"),
- js_str!("number"),
- js_str!("string"),
- js_str!("symbol"),
- js_str!("bigint"),
- js_str!("object"),
- js_str!("function"),
- // Property descriptor
- js_str!("value"),
- js_str!("get"),
- js_str!("set"),
- js_str!("writable"),
- js_str!("enumerable"),
- js_str!("configurable"),
- // Object object
- js_str!("assign"),
- js_str!("create"),
- js_str!("toString"),
- js_str!("valueOf"),
- js_str!("is"),
- js_str!("seal"),
- js_str!("isSealed"),
- js_str!("freeze"),
- js_str!("isFrozen"),
- js_str!("isExtensible"),
- js_str!("hasOwnProperty"),
- js_str!("isPrototypeOf"),
- js_str!("setPrototypeOf"),
- js_str!("getPrototypeOf"),
- js_str!("defineProperty"),
- js_str!("defineProperties"),
- js_str!("deleteProperty"),
- js_str!("construct"),
- js_str!("hasOwn"),
- js_str!("ownKeys"),
- js_str!("keys"),
- js_str!("values"),
- js_str!("entries"),
- js_str!("fromEntries"),
- js_str!("propertyIsEnumerable"),
- js_str!("preventExtensions"),
- js_str!("getOwnPropertyDescriptor"),
- js_str!("getOwnPropertyDescriptors"),
- js_str!("getOwnPropertyNames"),
- js_str!("getOwnPropertySymbols"),
- js_str!("__defineGetter__"),
- js_str!("__defineSetter__"),
- js_str!("__lookupGetter__"),
- js_str!("__lookupSetter__"),
- js_str!("__proto__"),
- js_str!("get __proto__"),
- js_str!("set __proto__"),
- // Function object
- js_str!("apply"),
- js_str!("bind"),
- js_str!("call"),
- js_str!("caller"),
- // Arguments object
- js_str!("callee"),
- // Array object
- js_str!("at"),
- js_str!("from"),
- js_str!("isArray"),
- js_str!("of"),
- js_str!("copyWithin"),
- js_str!("every"),
- js_str!("fill"),
- js_str!("filter"),
- js_str!("find"),
- js_str!("findIndex"),
- js_str!("findLast"),
- js_str!("findLastIndex"),
- js_str!("flat"),
- js_str!("flatMap"),
- js_str!("forEach"),
- js_str!("includes"),
- js_str!("indexOf"),
- js_str!("join"),
- js_str!("map"),
- js_str!("next"),
- js_str!("reduce"),
- js_str!("reduceRight"),
- js_str!("reverse"),
- js_str!("shift"),
- js_str!("slice"),
- js_str!("splice"),
- js_str!("some"),
- js_str!("sort"),
- js_str!("unshift"),
- js_str!("push"),
- js_str!("pop"),
- js_str!("groupBy"),
- js_str!("toReversed"),
- js_str!("toSorted"),
- js_str!("toSpliced"),
- js_str!("with"),
- // String object
- js_str!("charAt"),
- js_str!("charCodeAt"),
- js_str!("codePointAt"),
- js_str!("concat"),
- js_str!("endsWith"),
- js_str!("fromCharCode"),
- js_str!("fromCodePoint"),
- js_str!("lastIndexOf"),
- js_str!("match"),
- js_str!("matchAll"),
- js_str!("normalize"),
- js_str!("padEnd"),
- js_str!("padStart"),
- js_str!("raw"),
- js_str!("repeat"),
- js_str!("replace"),
- js_str!("replaceAll"),
- js_str!("search"),
- js_str!("split"),
- js_str!("startsWith"),
- js_str!("substr"),
- js_str!("substring"),
- js_str!("toLocaleString"),
- js_str!("toLowerCase"),
- js_str!("toUpperCase"),
- js_str!("trim"),
- js_str!("trimEnd"),
- js_str!("trimStart"),
- js_str!("isWellFormed"),
- js_str!("localeCompare"),
- js_str!("toWellFormed"),
- js_str!("toLocaleLowerCase"),
- js_str!("toLocaleUpperCase"),
- js_str!("trimLeft"),
- js_str!("trimRight"),
- js_str!("anchor"),
- js_str!("big"),
- js_str!("blink"),
- js_str!("bold"),
- js_str!("fixed"),
- js_str!("fontcolor"),
- js_str!("fontsize"),
- js_str!("italics"),
- js_str!("link"),
- js_str!("small"),
- js_str!("strike"),
- js_str!("sub"),
- js_str!("sup"),
- // Number object
- js_str!("Infinity"),
- js_str!("NaN"),
- js_str!("EPSILON"),
- js_str!("MAX_SAFE_INTEGER"),
- js_str!("MIN_SAFE_INTEGER"),
- js_str!("MAX_VALUE"),
- js_str!("MIN_VALUE"),
- js_str!("NEGATIVE_INFINITY"),
- js_str!("POSITIVE_INFINITY"),
- js_str!("isSafeInteger"),
- js_str!("isInteger"),
- js_str!("toExponential"),
- js_str!("toFixed"),
- js_str!("toPrecision"),
- // BigInt object
- js_str!("asIntN"),
- js_str!("asUintN"),
- // RegExp object
- js_str!("exec"),
- js_str!("test"),
- js_str!("compile"),
- js_str!("flags"),
- js_str!("index"),
- js_str!("lastIndex"),
- js_str!("hasIndices"),
- js_str!("ignoreCase"),
- js_str!("multiline"),
- js_str!("dotAll"),
- js_str!("unicode"),
- js_str!("sticky"),
- js_str!("source"),
- js_str!("get hasIndices"),
- js_str!("get global"),
- js_str!("get ignoreCase"),
- js_str!("get multiline"),
- js_str!("get dotAll"),
- js_str!("get unicode"),
- js_str!("get sticky"),
- js_str!("get flags"),
- js_str!("get source"),
- // Symbol object
- js_str!("for"),
- js_str!("keyFor"),
- js_str!("description"),
- js_str!("asyncIterator"),
- js_str!("hasInstance"),
- js_str!("species"),
- js_str!("unscopables"),
- js_str!("iterator"),
- js_str!("toStringTag"),
- js_str!("toPrimitive"),
- js_str!("isConcatSpreadable"),
- js_str!("get description"),
- // Map object
- js_str!("clear"),
- js_str!("delete"),
- js_str!("has"),
- js_str!("size"),
- // Set object
- js_str!("add"),
- // Reflect object
- // Proxy object
- js_str!("revocable"),
- // Error objects
- js_str!("message"),
- // Date object
- js_str!("toJSON"),
- js_str!("getDate"),
- js_str!("getDay"),
- js_str!("getFullYear"),
- js_str!("getHours"),
- js_str!("getMilliseconds"),
- js_str!("getMinutes"),
- js_str!("getMonth"),
- js_str!("getSeconds"),
- js_str!("getTime"),
- js_str!("getYear"),
- js_str!("getUTCDate"),
- js_str!("getUTCDay"),
- js_str!("getUTCFullYear"),
- js_str!("getUTCHours"),
- js_str!("getUTCMinutes"),
- js_str!("getUTCMonth"),
- js_str!("getUTCSeconds"),
- js_str!("setDate"),
- js_str!("setFullYear"),
- js_str!("setHours"),
- js_str!("setMilliseconds"),
- js_str!("setMinutes"),
- js_str!("setMonth"),
- js_str!("setSeconds"),
- js_str!("setYear"),
- js_str!("setTime"),
- js_str!("setUTCDate"),
- js_str!("setUTCFullYear"),
- js_str!("setUTCHours"),
- js_str!("setUTCMinutes"),
- js_str!("setUTCMonth"),
- js_str!("setUTCSeconds"),
- js_str!("toDateString"),
- js_str!("toGMTString"),
- js_str!("toISOString"),
- js_str!("toTimeString"),
- js_str!("toUTCString"),
- js_str!("now"),
- js_str!("UTC"),
- js_str!("getTimezoneOffset"),
- js_str!("getUTCMilliseconds"),
- js_str!("setUTCMilliseconds"),
- js_str!("toLocaleDateString"),
- js_str!("toLocaleTimeString"),
- // JSON object
- js_str!("parse"),
- js_str!("stringify"),
- // Promise object
- js_str!("promise"),
- js_str!("resolve"),
- js_str!("reject"),
- js_str!("all"),
- js_str!("allSettled"),
- js_str!("any"),
- js_str!("race"),
- js_str!("then"),
- js_str!("catch"),
- js_str!("finally"),
- js_str!("withResolvers"),
- // Iterator object
- js_str!("Array Iterator"),
- js_str!("Set Iterator"),
- js_str!("String Iterator"),
- js_str!("Map Iterator"),
- js_str!("For In Iterator"),
- js_str!("RegExp String Iterator"),
- // Iterator result object
- js_str!("done"),
- // Math object
- js_str!("LN10"),
- js_str!("LN2"),
- js_str!("LOG10E"),
- js_str!("LOG2E"),
- js_str!("PI"),
- js_str!("SQRT1_2"),
- js_str!("SQRT2"),
- js_str!("abs"),
- js_str!("acos"),
- js_str!("acosh"),
- js_str!("asin"),
- js_str!("asinh"),
- js_str!("atan"),
- js_str!("atanh"),
- js_str!("atan2"),
- js_str!("cbrt"),
- js_str!("ceil"),
- js_str!("clz32"),
- js_str!("cos"),
- js_str!("cosh"),
- js_str!("exp"),
- js_str!("expm1"),
- js_str!("floor"),
- js_str!("fround"),
- js_str!("hypot"),
- js_str!("imul"),
- js_str!("log"),
- js_str!("log1p"),
- js_str!("log10"),
- js_str!("log2"),
- js_str!("max"),
- js_str!("min"),
- js_str!("pow"),
- js_str!("random"),
- js_str!("round"),
- js_str!("sign"),
- js_str!("sin"),
- js_str!("sinh"),
- js_str!("sqrt"),
- js_str!("tan"),
- js_str!("tanh"),
- js_str!("trunc"),
- // TypedArray object
- js_str!("BYTES_PER_ELEMENT"),
- js_str!("buffer"),
- js_str!("byteLength"),
- js_str!("byteOffset"),
- js_str!("isView"),
- js_str!("subarray"),
- js_str!("get byteLength"),
- js_str!("get buffer"),
- js_str!("get byteOffset"),
- js_str!("get size"),
- js_str!("get length"),
- // DataView object
- js_str!("getBigInt64"),
- js_str!("getBigUint64"),
- js_str!("getFloat32"),
- js_str!("getFloat64"),
- js_str!("getInt8"),
- js_str!("getInt16"),
- js_str!("getInt32"),
- js_str!("getUint8"),
- js_str!("getUint16"),
- js_str!("getUint32"),
- js_str!("setBigInt64"),
- js_str!("setBigUint64"),
- js_str!("setFloat32"),
- js_str!("setFloat64"),
- js_str!("setInt8"),
- js_str!("setInt16"),
- js_str!("setInt32"),
- js_str!("setUint8"),
- js_str!("setUint16"),
- js_str!("setUint32"),
- // WeakRef object
- js_str!("deref"),
- // Atomic object
- js_str!("and"),
- js_str!("compareExchange"),
- js_str!("exchange"),
- js_str!("isLockFree"),
- js_str!("load"),
- js_str!("or"),
- js_str!("store"),
- js_str!("wait"),
- js_str!("notify"),
- js_str!("xor"),
- // Intl object
- js_str!("getCanonicalLocales"),
- js_str!("get compare"),
- js_str!("supportedLocalesOf"),
- js_str!("Intl.Collator"),
- js_str!("compare"),
- js_str!("resolvedOptions"),
- js_str!("Intl.ListFormat"),
- js_str!("format"),
- js_str!("formatToParts"),
- js_str!("get baseName"),
- js_str!("get calendar"),
- js_str!("get caseFirst"),
- js_str!("get collation"),
- js_str!("get hourCycle"),
- js_str!("get numeric"),
- js_str!("get numberingSystem"),
- js_str!("get language"),
- js_str!("get script"),
- js_str!("get region"),
- js_str!("Intl.Locale"),
- js_str!("maximize"),
- js_str!("minimize"),
- js_str!("baseName"),
- js_str!("calendar"),
- js_str!("caseFirst"),
- js_str!("collation"),
- js_str!("hourCycle"),
- js_str!("numeric"),
- js_str!("numberingSystem"),
- js_str!("language"),
- js_str!("script"),
- js_str!("region"),
- js_str!("Intl.Segmenter"),
- js_str!("segment"),
- js_str!("containing"),
- js_str!("Segmenter String Iterator"),
- js_str!("Intl.PluralRules"),
- js_str!("select"),
- // Temporal object
- js_str!("get Id"),
- js_str!("getOffsetNanosecondsFor"),
- js_str!("getOffsetStringFor"),
- js_str!("getPlainDateTimeFor"),
- js_str!("getInstantFor"),
- js_str!("getPossibleInstantFor"),
- js_str!("getNextTransition"),
- js_str!("getPreviousTransition"),
- js_str!("id"),
- js_str!("Now"),
- js_str!("Calendar"),
- js_str!("Duration"),
- js_str!("Instant"),
- js_str!("PlainDate"),
- js_str!("PlainDateTime"),
- js_str!("PlainMonthDay"),
- js_str!("PlainTime"),
- js_str!("PlainYearMonth"),
- js_str!("TimeZone"),
- js_str!("ZonedDateTime"),
- js_str!("timeZoneId"),
- js_str!("instant"),
- js_str!("plainDateTime"),
- js_str!("plainDateTimeISO"),
- js_str!("zonedDateTime"),
- js_str!("zonedDateTimeISO"),
- js_str!("plainDate"),
- js_str!("plainDateISO"),
- js_str!("get epochSeconds"),
- js_str!("get epochMilliseconds"),
- js_str!("get epochMicroseconds"),
- js_str!("get epochNanoseconds"),
- js_str!("epochSeconds"),
- js_str!("epochMilliseconds"),
- js_str!("epochMicroseconds"),
- js_str!("epochNanoseconds"),
- js_str!("subtract"),
- js_str!("until"),
- js_str!("since"),
- js_str!("equals"),
- js_str!("toZonedDateTime"),
- js_str!("toZonedDateTimeISO"),
- js_str!("get Years"),
- js_str!("get Months"),
- js_str!("get Weeks"),
- js_str!("get Days"),
- js_str!("get Hours"),
- js_str!("get Minutes"),
- js_str!("get Seconds"),
- js_str!("get Milliseconds"),
- js_str!("get Microseconds"),
- js_str!("get Nanoseconds"),
- js_str!("get Sign"),
- js_str!("get blank"),
- js_str!("years"),
- js_str!("months"),
- js_str!("weeks"),
- js_str!("days"),
- js_str!("hours"),
- js_str!("minutes"),
- js_str!("seconds"),
- js_str!("milliseconds"),
- js_str!("microseconds"),
- js_str!("nanoseconds"),
- js_str!("blank"),
- js_str!("negated"),
- js_str!("total"),
- js_str!("get calendarId"),
- js_str!("get year"),
- js_str!("get month"),
- js_str!("get monthCode"),
- js_str!("get day"),
- js_str!("get dayOfWeek"),
- js_str!("get dayOfYear"),
- js_str!("get weekOfYear"),
- js_str!("get yearOfWeek"),
- js_str!("get daysInWeek"),
- js_str!("get daysInMonth"),
- js_str!("get daysInYear"),
- js_str!("get monthsInYear"),
- js_str!("get inLeapYear"),
- js_str!("calendarId"),
- js_str!("year"),
- js_str!("month"),
- js_str!("monthCode"),
- js_str!("day"),
- js_str!("dayOfWeek"),
- js_str!("dayOfYear"),
- js_str!("weekOfYear"),
- js_str!("yearOfWeek"),
- js_str!("daysInWeek"),
- js_str!("daysInMonth"),
- js_str!("daysInYear"),
- js_str!("monthsInYear"),
- js_str!("inLeapYear"),
- js_str!("toPlainYearMonth"),
- js_str!("toPlainMonthDay"),
- js_str!("getISOFields"),
- js_str!("getCalendar"),
- js_str!("withCalendar"),
- js_str!("dateFromFields"),
- js_str!("yearMonthFromFields"),
- js_str!("monthDayFromFields"),
- js_str!("dateAdd"),
- js_str!("dateUntil"),
- js_str!("era"),
- js_str!("eraYear"),
- js_str!("fields"),
- js_str!("mergeFields"),
- // Console object
- js_str!("console"),
- js_str!("assert"),
- js_str!("debug"),
- js_str!("error"),
- js_str!("info"),
- js_str!("trace"),
- js_str!("warn"),
- js_str!("exception"),
- js_str!("count"),
- js_str!("countReset"),
- js_str!("group"),
- js_str!("groupCollapsed"),
- js_str!("groupEnd"),
- js_str!("time"),
- js_str!("timeLog"),
- js_str!("timeEnd"),
- js_str!("dir"),
- js_str!("dirxml"),
- // Minified name
- js_str!("a"),
- js_str!("c"),
- js_str!("d"),
- js_str!("e"),
- js_str!("f"),
- js_str!("g"),
- js_str!("h"),
- js_str!("i"),
- js_str!("j"),
- js_str!("k"),
- js_str!("l"),
- js_str!("m"),
- js_str!("n"),
- js_str!("o"),
- js_str!("p"),
- js_str!("q"),
- js_str!("r"),
- js_str!("s"),
- js_str!("t"),
- js_str!("u"),
- js_str!("v"),
- js_str!("w"),
- js_str!("x"),
- js_str!("y"),
- js_str!("z"),
- js_str!("A"),
- js_str!("C"),
- js_str!("D"),
- js_str!("E"),
- js_str!("F"),
- js_str!("G"),
- js_str!("H"),
- js_str!("I"),
- js_str!("J"),
- js_str!("K"),
- js_str!("L"),
- js_str!("M"),
- js_str!("N"),
- js_str!("O"),
- js_str!("P"),
- js_str!("Q"),
- js_str!("R"),
- js_str!("S"),
- js_str!("T"),
- js_str!("U"),
- js_str!("V"),
- js_str!("W"),
- js_str!("X"),
- js_str!("Y"),
- js_str!("Z"),
- js_str!("_"),
- js_str!("$"),
-];
diff --git a/core/engine/src/symbol.rs b/core/engine/src/symbol.rs
index 11b04dd9e69..a25cf08855b 100644
--- a/core/engine/src/symbol.rs
+++ b/core/engine/src/symbol.rs
@@ -23,7 +23,7 @@
use crate::{
js_string,
- string::{common::StaticJsStrings, JsString},
+ string::{JsString, StaticJsStrings},
tagged::{Tagged, UnwrappedTagged},
};
use boa_gc::{Finalize, Trace};
diff --git a/core/engine/src/tagged.rs b/core/engine/src/tagged.rs
index 83ec17c24b4..3415b38ed81 100644
--- a/core/engine/src/tagged.rs
+++ b/core/engine/src/tagged.rs
@@ -39,6 +39,7 @@ impl Clone for Tagged {
impl Copy for Tagged {}
+#[allow(dead_code)]
impl Tagged {
/// Creates a new, tagged `Tagged` pointer from an integer.
///
diff --git a/core/engine/src/vm/opcode/push/array.rs b/core/engine/src/vm/opcode/push/array.rs
index 9135f4109ba..480e979ecf8 100644
--- a/core/engine/src/vm/opcode/push/array.rs
+++ b/core/engine/src/vm/opcode/push/array.rs
@@ -1,6 +1,6 @@
use crate::{
builtins::Array,
- string::common::StaticJsStrings,
+ string::StaticJsStrings,
vm::{opcode::Operation, CompletionType},
Context, JsResult, JsValue,
};
diff --git a/core/gc/ABOUT.md b/core/gc/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/gc/ABOUT.md
+++ b/core/gc/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/gc/Cargo.toml b/core/gc/Cargo.toml
index b21b61c44cb..af223d07677 100644
--- a/core/gc/Cargo.toml
+++ b/core/gc/Cargo.toml
@@ -15,12 +15,15 @@ rust-version.workspace = true
thin-vec = ["dep:thin-vec"]
# Enable default implementations of trace and finalize for some `ICU4X` types
icu = ["dep:icu_locid"]
+# Enable default implementations of trace and finalize for the `boa_string` crate
+boa_string = ["dep:boa_string"]
[dependencies]
boa_profiler.workspace = true
boa_macros.workspace = true
hashbrown = { workspace = true, features = ["ahash", "raw"] }
+boa_string = { workspace = true, optional = true }
thin-vec = { workspace = true, optional = true }
icu_locid = { workspace = true, optional = true }
diff --git a/core/gc/src/trace.rs b/core/gc/src/trace.rs
index b28d976e8d1..41ac9d91446 100644
--- a/core/gc/src/trace.rs
+++ b/core/gc/src/trace.rs
@@ -521,3 +521,15 @@ mod icu {
empty_trace!();
}
}
+
+#[cfg(feature = "boa_string")]
+mod boa_string_trace {
+ use crate::{Finalize, Trace};
+
+ // SAFETY: `boa_string::JsString` doesn't have any traceable data.
+ unsafe impl Trace for boa_string::JsString {
+ empty_trace!();
+ }
+
+ impl Finalize for boa_string::JsString {}
+}
diff --git a/core/icu_provider/ABOUT.md b/core/icu_provider/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/icu_provider/ABOUT.md
+++ b/core/icu_provider/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/interner/ABOUT.md b/core/interner/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/interner/ABOUT.md
+++ b/core/interner/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/interop/ABOUT.md b/core/interop/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/interop/ABOUT.md
+++ b/core/interop/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/macros/ABOUT.md b/core/macros/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/macros/ABOUT.md
+++ b/core/macros/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/parser/ABOUT.md b/core/parser/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/parser/ABOUT.md
+++ b/core/parser/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/profiler/ABOUT.md b/core/profiler/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/profiler/ABOUT.md
+++ b/core/profiler/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/runtime/ABOUT.md b/core/runtime/ABOUT.md
index c1de9c5a19f..c8c37206ac0 100644
--- a/core/runtime/ABOUT.md
+++ b/core/runtime/ABOUT.md
@@ -18,6 +18,7 @@ Try out the most recent release with Boa's live demo
- [**`boa_profiler`**][profiler] - Boa's code profiler.
- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
[boa-conformance]: https://boajs.dev/conformance
[boa-web]: https://boajs.dev/
@@ -30,3 +31,4 @@ Try out the most recent release with Boa's live demo
[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/string/ABOUT.md b/core/string/ABOUT.md
new file mode 100644
index 00000000000..c8c37206ac0
--- /dev/null
+++ b/core/string/ABOUT.md
@@ -0,0 +1,34 @@
+# About Boa
+
+Boa is an open-source, experimental ECMAScript Engine written in Rust for
+lexing, parsing and executing ECMAScript/JavaScript. Currently, Boa supports some
+of the [language][boa-conformance]. More information can be viewed at [Boa's
+website][boa-web].
+
+Try out the most recent release with Boa's live demo
+[playground][boa-playground].
+
+## Boa Crates
+
+- [**`boa_ast`**][ast] - Boa's ECMAScript Abstract Syntax Tree.
+- [**`boa_engine`**][engine] - Boa's implementation of ECMAScript builtin objects and execution.
+- [**`boa_gc`**][gc] - Boa's garbage collector.
+- [**`boa_interner`**][interner] - Boa's string interner.
+- [**`boa_parser`**][parser] - Boa's lexer and parser.
+- [**`boa_profiler`**][profiler] - Boa's code profiler.
+- [**`boa_icu_provider`**][icu] - Boa's ICU4X data provider.
+- [**`boa_runtime`**][runtime] - Boa's WebAPI features.
+- [**`boa_string`**][string] - Boa's ECMAScript string implementation.
+
+[boa-conformance]: https://boajs.dev/conformance
+[boa-web]: https://boajs.dev/
+[boa-playground]: https://boajs.dev/playground
+[ast]: https://docs.rs/boa_ast/latest/boa_ast/index.html
+[engine]: https://docs.rs/boa_engine/latest/boa_engine/index.html
+[gc]: https://docs.rs/boa_gc/latest/boa_gc/index.html
+[interner]: https://docs.rs/boa_interner/latest/boa_interner/index.html
+[parser]: https://docs.rs/boa_parser/latest/boa_parser/index.html
+[profiler]: https://docs.rs/boa_profiler/latest/boa_profiler/index.html
+[icu]: https://docs.rs/boa_icu_provider/latest/boa_icu_provider/index.html
+[runtime]: https://docs.rs/boa_runtime/latest/boa_runtime/index.html
+[string]: https://docs.rs/boa_string/latest/boa_string/index.html
diff --git a/core/string/Cargo.toml b/core/string/Cargo.toml
new file mode 100644
index 00000000000..2a36d739c09
--- /dev/null
+++ b/core/string/Cargo.toml
@@ -0,0 +1,28 @@
+[package]
+name = "boa_string"
+keywords = ["javascript", "js", "string"]
+categories = ["parser-implementations", "compilers"]
+readme = "../../README.md"
+description.workspace = true
+version.workspace = true
+edition.workspace = true
+authors.workspace = true
+license.workspace = true
+repository.workspace = true
+rust-version.workspace = true
+
+[dependencies]
+rustc-hash = { workspace = true, features = ["std"] }
+sptr = "0.3.2"
+static_assertions.workspace = true
+paste = "1.0"
+fast-float.workspace = true
+
+[dev-dependencies]
+boa_macros.workspace = true
+
+[lints]
+workspace = true
+
+[package.metadata.docs.rs]
+all-features = true
diff --git a/core/string/src/common.rs b/core/string/src/common.rs
new file mode 100644
index 00000000000..8b420f7c682
--- /dev/null
+++ b/core/string/src/common.rs
@@ -0,0 +1,951 @@
+//! List of commonly used strings in Javascript code.
+
+use crate::{tagged::Tagged, JsStr};
+
+use super::JsString;
+use paste::paste;
+use rustc_hash::{FxHashMap, FxHasher};
+use std::hash::BuildHasherDefault;
+
+macro_rules! well_known_statics {
+ ( $( $(#[$attr:meta])* ($name:ident, $string:literal) ),+$(,)? ) => {
+ $(
+ paste!{
+ #[doc = "Gets the static `JsString` for `\"" $string "\"`."]
+ pub const $name: JsString = JsString {
+ ptr: Tagged::from_tag(
+ Self::find_index($string),
+ ),
+ };
+ }
+ )+
+ };
+}
+
+/// List of commonly used strings in Javascript code.
+///
+/// Any strings defined here are used as a static [`JsString`] instead of allocating on the heap.
+#[derive(Debug, Clone, Copy)]
+pub struct StaticJsStrings;
+
+impl StaticJsStrings {
+ // useful to search at compile time a certain string in the array
+ const fn find_index(candidate: &str) -> usize {
+ const fn const_eq(lhs: &[u8], rhs: &[u8]) -> bool {
+ if lhs.len() != rhs.len() {
+ return false;
+ }
+
+ let mut i = 0;
+ while i < lhs.len() {
+ if lhs[i] != rhs[i] {
+ return false;
+ }
+ i += 1;
+ }
+ true
+ }
+
+ let len = RAW_STATICS.len();
+ let mut i = 0;
+ while i < len {
+ let Some(s) = RAW_STATICS[i].as_latin1() else {
+ // All static strings are latin1 encoded
+ unreachable!()
+ };
+ if const_eq(s, candidate.as_bytes()) {
+ return i;
+ }
+ i += 1;
+ }
+
+ panic!("couldn't find the required string on the common string array");
+ }
+
+ /// Gets the `JsString` corresponding to `string`, or `None` if the string
+ /// doesn't exist inside the static array.
+ #[inline]
+ #[must_use]
+ pub fn get_string(string: &JsStr<'_>) -> Option {
+ if string.len() > MAX_STATIC_LENGTH {
+ return None;
+ }
+
+ let index = RAW_STATICS_CACHE.with(|map| map.get(string).copied())?;
+
+ Some(JsString {
+ ptr: Tagged::from_tag(index),
+ })
+ }
+
+ /// Gets the `&[u16]` slice corresponding to the provided index, or `None` if the index
+ /// provided exceeds the size of the static array.
+ pub(crate) fn get(index: usize) -> Option> {
+ RAW_STATICS.get(index).copied()
+ }
+
+ // Some consts are only used on certain features, which triggers the unused lint.
+ well_known_statics! {
+ (EMPTY_STRING, ""),
+ (LENGTH, "length"),
+ // Symbols
+ (SYMBOL_ASYNC_ITERATOR, "Symbol.asyncIterator"),
+ (SYMBOL_HAS_INSTANCE, "Symbol.hasInstance"),
+ (SYMBOL_IS_CONCAT_SPREADABLE, "Symbol.isConcatSpreadable"),
+ (SYMBOL_ITERATOR, "Symbol.iterator"),
+ (SYMBOL_MATCH, "Symbol.match"),
+ (SYMBOL_MATCH_ALL, "Symbol.matchAll"),
+ (SYMBOL_REPLACE, "Symbol.replace"),
+ (SYMBOL_SEARCH, "Symbol.search"),
+ (SYMBOL_SPECIES, "Symbol.species"),
+ (SYMBOL_SPLIT, "Symbol.split"),
+ (SYMBOL_TO_PRIMITIVE, "Symbol.toPrimitive"),
+ (SYMBOL_TO_STRING_TAG, "Symbol.toStringTag"),
+ (SYMBOL_UNSCOPABLES, "Symbol.unscopables"),
+ (FN_SYMBOL_ASYNC_ITERATOR, "[Symbol.asyncIterator]"),
+ (FN_SYMBOL_HAS_INSTANCE, "[Symbol.hasInstance]"),
+ (FN_SYMBOL_IS_CONCAT_SPREADABLE, "[Symbol.isConcatSpreadable]"),
+ (FN_SYMBOL_ITERATOR, "[Symbol.iterator]"),
+ (FN_SYMBOL_MATCH, "[Symbol.match]"),
+ (FN_SYMBOL_MATCH_ALL, "[Symbol.matchAll]"),
+ (FN_SYMBOL_REPLACE, "[Symbol.replace]"),
+ (FN_SYMBOL_SEARCH, "[Symbol.search]"),
+ (FN_SYMBOL_SPECIES, "[Symbol.species]"),
+ (FN_SYMBOL_SPLIT, "[Symbol.split]"),
+ (FN_SYMBOL_TO_PRIMITIVE, "[Symbol.toPrimitive]"),
+ (FN_SYMBOL_TO_STRING_TAG, "[Symbol.toStringTag]"),
+ (FN_SYMBOL_UNSCOPABLES, "[Symbol.unscopables]"),
+ // Builtins
+ (ARRAY, "Array"),
+ (ARRAY_BUFFER, "ArrayBuffer"),
+ (SHARED_ARRAY_BUFFER, "SharedArrayBuffer"),
+ (ASYNC_FUNCTION, "AsyncFunction"),
+ (ASYNC_GENERATOR, "AsyncGenerator"),
+ (ASYNC_GENERATOR_FUNCTION, "AsyncGeneratorFunction"),
+ (ATOMICS, "Atomics"),
+ (BIG_INT, "BigInt"),
+ (BOOLEAN, "Boolean"),
+ (DATA_VIEW, "DataView"),
+ (DATE, "Date"),
+ (ERROR, "Error"),
+ (AGGREGATE_ERROR, "AggregateError"),
+ (EVAL_ERROR, "EvalError"),
+ (RANGE_ERROR, "RangeError"),
+ (REFERENCE_ERROR, "ReferenceError"),
+ (SYNTAX_ERROR, "SyntaxError"),
+ (TYPE_ERROR, "TypeError"),
+ (URI_ERROR, "URIError"),
+ (ESCAPE, "escape"),
+ (UNESCAPE, "unescape"),
+ (EVAL, "eval"),
+ (FUNCTION, "Function"),
+ (GENERATOR, "Generator"),
+ (GENERATOR_FUNCTION, "GeneratorFunction"),
+ (INTL, "Intl"),
+ (COLLATOR, "Collator"),
+ (LIST_FORMAT, "ListFormat"),
+ (LOCALE, "Locale"),
+ (PLURAL_RULES, "PluralRules"),
+ (SEGMENTER, "Segmenter"),
+ (DATE_TIME_FORMAT, "DateTimeFormat"),
+ (JSON, "JSON"),
+ (MAP, "Map"),
+ (MATH, "Math"),
+ (NUMBER, "Number"),
+ (NUMBER_FORMAT, "NumberFormat"),
+ (IS_FINITE, "isFinite"),
+ (IS_NAN, "isNaN"),
+ (PARSE_INT, "parseInt"),
+ (PARSE_FLOAT, "parseFloat"),
+ (OBJECT, "Object"),
+ (PROMISE, "Promise"),
+ (PROXY, "Proxy"),
+ (REFLECT, "Reflect"),
+ (REG_EXP, "RegExp"),
+ (SET, "Set"),
+ (STRING, "String"),
+ (SYMBOL, "Symbol"),
+ (TYPED_ARRAY, "TypedArray"),
+ (INT8_ARRAY, "Int8Array"),
+ (UINT8_ARRAY, "Uint8Array"),
+ (UINT8_CLAMPED_ARRAY, "Uint8ClampedArray"),
+ (INT16_ARRAY, "Int16Array"),
+ (UINT16_ARRAY, "Uint16Array"),
+ (INT32_ARRAY, "Int32Array"),
+ (UINT32_ARRAY, "Uint32Array"),
+ (BIG_INT64_ARRAY, "BigInt64Array"),
+ (BIG_UINT64_ARRAY, "BigUint64Array"),
+ (FLOAT32_ARRAY, "Float32Array"),
+ (FLOAT64_ARRAY, "Float64Array"),
+ (ENCODE_URI, "encodeURI"),
+ (ENCODE_URI_COMPONENT, "encodeURIComponent"),
+ (DECODE_URI, "decodeURI"),
+ (DECODE_URI_COMPONENT, "decodeURIComponent"),
+ (WEAK_REF, "WeakRef"),
+ (WEAK_MAP, "WeakMap"),
+ (WEAK_SET, "WeakSet"),
+ (TEMPORAL, "Temporal"),
+ (NOW, "Temporal.Now"),
+ (INSTANT, "Temporal.Instant"),
+ (DURATION, "Temporal.Duration"),
+ (PLAIN_DATE, "Temporal.PlainDate"),
+ (PLAIN_DATETIME, "Temporal.PlainDateTime"),
+ (PLAIN_TIME, "Temporal.PlainTime"),
+ (PLAIN_YM, "Temporal.PlainYearMonth"),
+ (PLAIN_MD, "Temporal.PlainMonthDay"),
+ (CALENDAR, "Temporal.Calendar"),
+ (TIMEZONE, "Temporal.TimeZone"),
+ (ZONED_DT, "Temporal.ZonedDateTime"),
+ }
+}
+
+const MAX_STATIC_LENGTH: usize = {
+ let mut max = 0;
+ let mut i = 0;
+ while i < RAW_STATICS.len() {
+ // TOOD: Because `get_index` is not const, we are accessing doc hidden stuff, that may change.
+ let len = RAW_STATICS[i].len();
+ if len > max {
+ max = len;
+ }
+ i += 1;
+ }
+ max
+};
+
+thread_local! {
+ /// Map from a string inside [`RAW_STATICS`] to its corresponding static index on `RAW_STATICS`.
+ static RAW_STATICS_CACHE: FxHashMap, usize> = {
+ let mut constants = FxHashMap::with_capacity_and_hasher(
+ RAW_STATICS.len(),
+ BuildHasherDefault::::default(),
+ );
+
+ for (idx, &s) in RAW_STATICS.iter().enumerate() {
+ constants.insert(s, idx);
+ }
+
+ constants
+ };
+}
+
+/// Array of raw static strings that aren't reference counted.
+const RAW_STATICS: &[JsStr<'_>] = &[
+ JsStr::latin1("".as_bytes()),
+ // Well known symbols
+ JsStr::latin1("Symbol.asyncIterator".as_bytes()),
+ JsStr::latin1("[Symbol.asyncIterator]".as_bytes()),
+ JsStr::latin1("Symbol.hasInstance".as_bytes()),
+ JsStr::latin1("[Symbol.hasInstance]".as_bytes()),
+ JsStr::latin1("Symbol.isConcatSpreadable".as_bytes()),
+ JsStr::latin1("[Symbol.isConcatSpreadable]".as_bytes()),
+ JsStr::latin1("Symbol.iterator".as_bytes()),
+ JsStr::latin1("[Symbol.iterator]".as_bytes()),
+ JsStr::latin1("Symbol.match".as_bytes()),
+ JsStr::latin1("[Symbol.match]".as_bytes()),
+ JsStr::latin1("Symbol.matchAll".as_bytes()),
+ JsStr::latin1("[Symbol.matchAll]".as_bytes()),
+ JsStr::latin1("Symbol.replace".as_bytes()),
+ JsStr::latin1("[Symbol.replace]".as_bytes()),
+ JsStr::latin1("Symbol.search".as_bytes()),
+ JsStr::latin1("[Symbol.search]".as_bytes()),
+ JsStr::latin1("Symbol.species".as_bytes()),
+ JsStr::latin1("[Symbol.species]".as_bytes()),
+ JsStr::latin1("Symbol.split".as_bytes()),
+ JsStr::latin1("[Symbol.split]".as_bytes()),
+ JsStr::latin1("Symbol.toPrimitive".as_bytes()),
+ JsStr::latin1("[Symbol.toPrimitive]".as_bytes()),
+ JsStr::latin1("Symbol.toStringTag".as_bytes()),
+ JsStr::latin1("[Symbol.toStringTag]".as_bytes()),
+ JsStr::latin1("Symbol.unscopables".as_bytes()),
+ JsStr::latin1("[Symbol.unscopables]".as_bytes()),
+ JsStr::latin1("get [Symbol.species]".as_bytes()),
+ JsStr::latin1("get [Symbol.toStringTag]".as_bytes()),
+ // Well known builtins
+ JsStr::latin1("Array".as_bytes()),
+ JsStr::latin1("ArrayBuffer".as_bytes()),
+ JsStr::latin1("SharedArrayBuffer".as_bytes()),
+ JsStr::latin1("AsyncFunction".as_bytes()),
+ JsStr::latin1("AsyncGenerator".as_bytes()),
+ JsStr::latin1("AsyncGeneratorFunction".as_bytes()),
+ JsStr::latin1("Atomics".as_bytes()),
+ JsStr::latin1("BigInt".as_bytes()),
+ JsStr::latin1("Boolean".as_bytes()),
+ JsStr::latin1("DataView".as_bytes()),
+ JsStr::latin1("Date".as_bytes()),
+ JsStr::latin1("Error".as_bytes()),
+ JsStr::latin1("AggregateError".as_bytes()),
+ JsStr::latin1("EvalError".as_bytes()),
+ JsStr::latin1("RangeError".as_bytes()),
+ JsStr::latin1("ReferenceError".as_bytes()),
+ JsStr::latin1("SyntaxError".as_bytes()),
+ JsStr::latin1("TypeError".as_bytes()),
+ JsStr::latin1("URIError".as_bytes()),
+ JsStr::latin1("escape".as_bytes()),
+ JsStr::latin1("unescape".as_bytes()),
+ JsStr::latin1("eval".as_bytes()),
+ JsStr::latin1("Function".as_bytes()),
+ JsStr::latin1("Generator".as_bytes()),
+ JsStr::latin1("GeneratorFunction".as_bytes()),
+ JsStr::latin1("Intl".as_bytes()),
+ JsStr::latin1("Collator".as_bytes()),
+ JsStr::latin1("ListFormat".as_bytes()),
+ JsStr::latin1("Locale".as_bytes()),
+ JsStr::latin1("PluralRules".as_bytes()),
+ JsStr::latin1("Segmenter".as_bytes()),
+ JsStr::latin1("DateTimeFormat".as_bytes()),
+ JsStr::latin1("JSON".as_bytes()),
+ JsStr::latin1("Map".as_bytes()),
+ JsStr::latin1("Math".as_bytes()),
+ JsStr::latin1("Number".as_bytes()),
+ JsStr::latin1("NumberFormat".as_bytes()),
+ JsStr::latin1("isFinite".as_bytes()),
+ JsStr::latin1("isNaN".as_bytes()),
+ JsStr::latin1("parseInt".as_bytes()),
+ JsStr::latin1("parseFloat".as_bytes()),
+ JsStr::latin1("Object".as_bytes()),
+ JsStr::latin1("Promise".as_bytes()),
+ JsStr::latin1("Proxy".as_bytes()),
+ JsStr::latin1("Reflect".as_bytes()),
+ JsStr::latin1("RegExp".as_bytes()),
+ JsStr::latin1("Set".as_bytes()),
+ JsStr::latin1("String".as_bytes()),
+ JsStr::latin1("Symbol".as_bytes()),
+ JsStr::latin1("TypedArray".as_bytes()),
+ JsStr::latin1("Int8Array".as_bytes()),
+ JsStr::latin1("Uint8Array".as_bytes()),
+ JsStr::latin1("Uint8ClampedArray".as_bytes()),
+ JsStr::latin1("Int16Array".as_bytes()),
+ JsStr::latin1("Uint16Array".as_bytes()),
+ JsStr::latin1("Int32Array".as_bytes()),
+ JsStr::latin1("Uint32Array".as_bytes()),
+ JsStr::latin1("BigInt64Array".as_bytes()),
+ JsStr::latin1("BigUint64Array".as_bytes()),
+ JsStr::latin1("Float32Array".as_bytes()),
+ JsStr::latin1("Float64Array".as_bytes()),
+ JsStr::latin1("encodeURI".as_bytes()),
+ JsStr::latin1("encodeURIComponent".as_bytes()),
+ JsStr::latin1("decodeURI".as_bytes()),
+ JsStr::latin1("decodeURIComponent".as_bytes()),
+ JsStr::latin1("WeakRef".as_bytes()),
+ JsStr::latin1("WeakMap".as_bytes()),
+ JsStr::latin1("WeakSet".as_bytes()),
+ JsStr::latin1("Temporal".as_bytes()),
+ JsStr::latin1("Temporal.Now".as_bytes()),
+ JsStr::latin1("Temporal.Instant".as_bytes()),
+ JsStr::latin1("Temporal.Duration".as_bytes()),
+ JsStr::latin1("Temporal.Calendar".as_bytes()),
+ JsStr::latin1("Temporal.PlainDate".as_bytes()),
+ JsStr::latin1("Temporal.PlainDateTime".as_bytes()),
+ JsStr::latin1("Temporal.PlainMonthDay".as_bytes()),
+ JsStr::latin1("Temporal.PlainYearMonth".as_bytes()),
+ JsStr::latin1("Temporal.PlainTime".as_bytes()),
+ JsStr::latin1("Temporal.TimeZone".as_bytes()),
+ JsStr::latin1("Temporal.ZonedDateTime".as_bytes()),
+ // Misc
+ JsStr::latin1(",".as_bytes()),
+ JsStr::latin1(":".as_bytes()),
+ // Generic use
+ JsStr::latin1("name".as_bytes()),
+ JsStr::latin1("length".as_bytes()),
+ JsStr::latin1("arguments".as_bytes()),
+ JsStr::latin1("prototype".as_bytes()),
+ JsStr::latin1("constructor".as_bytes()),
+ JsStr::latin1("return".as_bytes()),
+ JsStr::latin1("throw".as_bytes()),
+ JsStr::latin1("global".as_bytes()),
+ JsStr::latin1("globalThis".as_bytes()),
+ // typeof
+ JsStr::latin1("null".as_bytes()),
+ JsStr::latin1("undefined".as_bytes()),
+ JsStr::latin1("number".as_bytes()),
+ JsStr::latin1("string".as_bytes()),
+ JsStr::latin1("symbol".as_bytes()),
+ JsStr::latin1("bigint".as_bytes()),
+ JsStr::latin1("object".as_bytes()),
+ JsStr::latin1("function".as_bytes()),
+ // Property descriptor
+ JsStr::latin1("value".as_bytes()),
+ JsStr::latin1("get".as_bytes()),
+ JsStr::latin1("set".as_bytes()),
+ JsStr::latin1("writable".as_bytes()),
+ JsStr::latin1("enumerable".as_bytes()),
+ JsStr::latin1("configurable".as_bytes()),
+ // Object object
+ JsStr::latin1("assign".as_bytes()),
+ JsStr::latin1("create".as_bytes()),
+ JsStr::latin1("toString".as_bytes()),
+ JsStr::latin1("valueOf".as_bytes()),
+ JsStr::latin1("is".as_bytes()),
+ JsStr::latin1("seal".as_bytes()),
+ JsStr::latin1("isSealed".as_bytes()),
+ JsStr::latin1("freeze".as_bytes()),
+ JsStr::latin1("isFrozen".as_bytes()),
+ JsStr::latin1("isExtensible".as_bytes()),
+ JsStr::latin1("hasOwnProperty".as_bytes()),
+ JsStr::latin1("isPrototypeOf".as_bytes()),
+ JsStr::latin1("setPrototypeOf".as_bytes()),
+ JsStr::latin1("getPrototypeOf".as_bytes()),
+ JsStr::latin1("defineProperty".as_bytes()),
+ JsStr::latin1("defineProperties".as_bytes()),
+ JsStr::latin1("deleteProperty".as_bytes()),
+ JsStr::latin1("construct".as_bytes()),
+ JsStr::latin1("hasOwn".as_bytes()),
+ JsStr::latin1("ownKeys".as_bytes()),
+ JsStr::latin1("keys".as_bytes()),
+ JsStr::latin1("values".as_bytes()),
+ JsStr::latin1("entries".as_bytes()),
+ JsStr::latin1("fromEntries".as_bytes()),
+ JsStr::latin1("propertyIsEnumerable".as_bytes()),
+ JsStr::latin1("preventExtensions".as_bytes()),
+ JsStr::latin1("getOwnPropertyDescriptor".as_bytes()),
+ JsStr::latin1("getOwnPropertyDescriptors".as_bytes()),
+ JsStr::latin1("getOwnPropertyNames".as_bytes()),
+ JsStr::latin1("getOwnPropertySymbols".as_bytes()),
+ JsStr::latin1("__defineGetter__".as_bytes()),
+ JsStr::latin1("__defineSetter__".as_bytes()),
+ JsStr::latin1("__lookupGetter__".as_bytes()),
+ JsStr::latin1("__lookupSetter__".as_bytes()),
+ JsStr::latin1("__proto__".as_bytes()),
+ JsStr::latin1("get __proto__".as_bytes()),
+ JsStr::latin1("set __proto__".as_bytes()),
+ // Function object
+ JsStr::latin1("apply".as_bytes()),
+ JsStr::latin1("bind".as_bytes()),
+ JsStr::latin1("call".as_bytes()),
+ JsStr::latin1("caller".as_bytes()),
+ // Arguments object
+ JsStr::latin1("callee".as_bytes()),
+ // Array object
+ JsStr::latin1("at".as_bytes()),
+ JsStr::latin1("from".as_bytes()),
+ JsStr::latin1("isArray".as_bytes()),
+ JsStr::latin1("of".as_bytes()),
+ JsStr::latin1("copyWithin".as_bytes()),
+ JsStr::latin1("every".as_bytes()),
+ JsStr::latin1("fill".as_bytes()),
+ JsStr::latin1("filter".as_bytes()),
+ JsStr::latin1("find".as_bytes()),
+ JsStr::latin1("findIndex".as_bytes()),
+ JsStr::latin1("findLast".as_bytes()),
+ JsStr::latin1("findLastIndex".as_bytes()),
+ JsStr::latin1("flat".as_bytes()),
+ JsStr::latin1("flatMap".as_bytes()),
+ JsStr::latin1("forEach".as_bytes()),
+ JsStr::latin1("includes".as_bytes()),
+ JsStr::latin1("indexOf".as_bytes()),
+ JsStr::latin1("join".as_bytes()),
+ JsStr::latin1("map".as_bytes()),
+ JsStr::latin1("next".as_bytes()),
+ JsStr::latin1("reduce".as_bytes()),
+ JsStr::latin1("reduceRight".as_bytes()),
+ JsStr::latin1("reverse".as_bytes()),
+ JsStr::latin1("shift".as_bytes()),
+ JsStr::latin1("slice".as_bytes()),
+ JsStr::latin1("splice".as_bytes()),
+ JsStr::latin1("some".as_bytes()),
+ JsStr::latin1("sort".as_bytes()),
+ JsStr::latin1("unshift".as_bytes()),
+ JsStr::latin1("push".as_bytes()),
+ JsStr::latin1("pop".as_bytes()),
+ JsStr::latin1("groupBy".as_bytes()),
+ JsStr::latin1("toReversed".as_bytes()),
+ JsStr::latin1("toSorted".as_bytes()),
+ JsStr::latin1("toSpliced".as_bytes()),
+ JsStr::latin1("with".as_bytes()),
+ // String object
+ JsStr::latin1("charAt".as_bytes()),
+ JsStr::latin1("charCodeAt".as_bytes()),
+ JsStr::latin1("codePointAt".as_bytes()),
+ JsStr::latin1("concat".as_bytes()),
+ JsStr::latin1("endsWith".as_bytes()),
+ JsStr::latin1("fromCharCode".as_bytes()),
+ JsStr::latin1("fromCodePoint".as_bytes()),
+ JsStr::latin1("lastIndexOf".as_bytes()),
+ JsStr::latin1("match".as_bytes()),
+ JsStr::latin1("matchAll".as_bytes()),
+ JsStr::latin1("normalize".as_bytes()),
+ JsStr::latin1("padEnd".as_bytes()),
+ JsStr::latin1("padStart".as_bytes()),
+ JsStr::latin1("raw".as_bytes()),
+ JsStr::latin1("repeat".as_bytes()),
+ JsStr::latin1("replace".as_bytes()),
+ JsStr::latin1("replaceAll".as_bytes()),
+ JsStr::latin1("search".as_bytes()),
+ JsStr::latin1("split".as_bytes()),
+ JsStr::latin1("startsWith".as_bytes()),
+ JsStr::latin1("substr".as_bytes()),
+ JsStr::latin1("substring".as_bytes()),
+ JsStr::latin1("toLocaleString".as_bytes()),
+ JsStr::latin1("toLowerCase".as_bytes()),
+ JsStr::latin1("toUpperCase".as_bytes()),
+ JsStr::latin1("trim".as_bytes()),
+ JsStr::latin1("trimEnd".as_bytes()),
+ JsStr::latin1("trimStart".as_bytes()),
+ JsStr::latin1("isWellFormed".as_bytes()),
+ JsStr::latin1("localeCompare".as_bytes()),
+ JsStr::latin1("toWellFormed".as_bytes()),
+ JsStr::latin1("toLocaleLowerCase".as_bytes()),
+ JsStr::latin1("toLocaleUpperCase".as_bytes()),
+ JsStr::latin1("trimLeft".as_bytes()),
+ JsStr::latin1("trimRight".as_bytes()),
+ JsStr::latin1("anchor".as_bytes()),
+ JsStr::latin1("big".as_bytes()),
+ JsStr::latin1("blink".as_bytes()),
+ JsStr::latin1("bold".as_bytes()),
+ JsStr::latin1("fixed".as_bytes()),
+ JsStr::latin1("fontcolor".as_bytes()),
+ JsStr::latin1("fontsize".as_bytes()),
+ JsStr::latin1("italics".as_bytes()),
+ JsStr::latin1("link".as_bytes()),
+ JsStr::latin1("small".as_bytes()),
+ JsStr::latin1("strike".as_bytes()),
+ JsStr::latin1("sub".as_bytes()),
+ JsStr::latin1("sup".as_bytes()),
+ // Number object
+ JsStr::latin1("Infinity".as_bytes()),
+ JsStr::latin1("NaN".as_bytes()),
+ JsStr::latin1("EPSILON".as_bytes()),
+ JsStr::latin1("MAX_SAFE_INTEGER".as_bytes()),
+ JsStr::latin1("MIN_SAFE_INTEGER".as_bytes()),
+ JsStr::latin1("MAX_VALUE".as_bytes()),
+ JsStr::latin1("MIN_VALUE".as_bytes()),
+ JsStr::latin1("NEGATIVE_INFINITY".as_bytes()),
+ JsStr::latin1("POSITIVE_INFINITY".as_bytes()),
+ JsStr::latin1("isSafeInteger".as_bytes()),
+ JsStr::latin1("isInteger".as_bytes()),
+ JsStr::latin1("toExponential".as_bytes()),
+ JsStr::latin1("toFixed".as_bytes()),
+ JsStr::latin1("toPrecision".as_bytes()),
+ // BigInt object
+ JsStr::latin1("asIntN".as_bytes()),
+ JsStr::latin1("asUintN".as_bytes()),
+ // RegExp object
+ JsStr::latin1("exec".as_bytes()),
+ JsStr::latin1("test".as_bytes()),
+ JsStr::latin1("compile".as_bytes()),
+ JsStr::latin1("flags".as_bytes()),
+ JsStr::latin1("index".as_bytes()),
+ JsStr::latin1("lastIndex".as_bytes()),
+ JsStr::latin1("hasIndices".as_bytes()),
+ JsStr::latin1("ignoreCase".as_bytes()),
+ JsStr::latin1("multiline".as_bytes()),
+ JsStr::latin1("dotAll".as_bytes()),
+ JsStr::latin1("unicode".as_bytes()),
+ JsStr::latin1("sticky".as_bytes()),
+ JsStr::latin1("source".as_bytes()),
+ JsStr::latin1("get hasIndices".as_bytes()),
+ JsStr::latin1("get global".as_bytes()),
+ JsStr::latin1("get ignoreCase".as_bytes()),
+ JsStr::latin1("get multiline".as_bytes()),
+ JsStr::latin1("get dotAll".as_bytes()),
+ JsStr::latin1("get unicode".as_bytes()),
+ JsStr::latin1("get sticky".as_bytes()),
+ JsStr::latin1("get flags".as_bytes()),
+ JsStr::latin1("get source".as_bytes()),
+ // Symbol object
+ JsStr::latin1("for".as_bytes()),
+ JsStr::latin1("keyFor".as_bytes()),
+ JsStr::latin1("description".as_bytes()),
+ JsStr::latin1("asyncIterator".as_bytes()),
+ JsStr::latin1("hasInstance".as_bytes()),
+ JsStr::latin1("species".as_bytes()),
+ JsStr::latin1("unscopables".as_bytes()),
+ JsStr::latin1("iterator".as_bytes()),
+ JsStr::latin1("toStringTag".as_bytes()),
+ JsStr::latin1("toPrimitive".as_bytes()),
+ JsStr::latin1("isConcatSpreadable".as_bytes()),
+ JsStr::latin1("get description".as_bytes()),
+ // Map object
+ JsStr::latin1("clear".as_bytes()),
+ JsStr::latin1("delete".as_bytes()),
+ JsStr::latin1("has".as_bytes()),
+ JsStr::latin1("size".as_bytes()),
+ // Set object
+ JsStr::latin1("add".as_bytes()),
+ // Reflect object
+ // Proxy object
+ JsStr::latin1("revocable".as_bytes()),
+ // Error objects
+ JsStr::latin1("message".as_bytes()),
+ // Date object
+ JsStr::latin1("toJSON".as_bytes()),
+ JsStr::latin1("getDate".as_bytes()),
+ JsStr::latin1("getDay".as_bytes()),
+ JsStr::latin1("getFullYear".as_bytes()),
+ JsStr::latin1("getHours".as_bytes()),
+ JsStr::latin1("getMilliseconds".as_bytes()),
+ JsStr::latin1("getMinutes".as_bytes()),
+ JsStr::latin1("getMonth".as_bytes()),
+ JsStr::latin1("getSeconds".as_bytes()),
+ JsStr::latin1("getTime".as_bytes()),
+ JsStr::latin1("getYear".as_bytes()),
+ JsStr::latin1("getUTCDate".as_bytes()),
+ JsStr::latin1("getUTCDay".as_bytes()),
+ JsStr::latin1("getUTCFullYear".as_bytes()),
+ JsStr::latin1("getUTCHours".as_bytes()),
+ JsStr::latin1("getUTCMinutes".as_bytes()),
+ JsStr::latin1("getUTCMonth".as_bytes()),
+ JsStr::latin1("getUTCSeconds".as_bytes()),
+ JsStr::latin1("setDate".as_bytes()),
+ JsStr::latin1("setFullYear".as_bytes()),
+ JsStr::latin1("setHours".as_bytes()),
+ JsStr::latin1("setMilliseconds".as_bytes()),
+ JsStr::latin1("setMinutes".as_bytes()),
+ JsStr::latin1("setMonth".as_bytes()),
+ JsStr::latin1("setSeconds".as_bytes()),
+ JsStr::latin1("setYear".as_bytes()),
+ JsStr::latin1("setTime".as_bytes()),
+ JsStr::latin1("setUTCDate".as_bytes()),
+ JsStr::latin1("setUTCFullYear".as_bytes()),
+ JsStr::latin1("setUTCHours".as_bytes()),
+ JsStr::latin1("setUTCMinutes".as_bytes()),
+ JsStr::latin1("setUTCMonth".as_bytes()),
+ JsStr::latin1("setUTCSeconds".as_bytes()),
+ JsStr::latin1("toDateString".as_bytes()),
+ JsStr::latin1("toGMTString".as_bytes()),
+ JsStr::latin1("toISOString".as_bytes()),
+ JsStr::latin1("toTimeString".as_bytes()),
+ JsStr::latin1("toUTCString".as_bytes()),
+ JsStr::latin1("now".as_bytes()),
+ JsStr::latin1("UTC".as_bytes()),
+ JsStr::latin1("getTimezoneOffset".as_bytes()),
+ JsStr::latin1("getUTCMilliseconds".as_bytes()),
+ JsStr::latin1("setUTCMilliseconds".as_bytes()),
+ JsStr::latin1("toLocaleDateString".as_bytes()),
+ JsStr::latin1("toLocaleTimeString".as_bytes()),
+ // JSON object
+ JsStr::latin1("parse".as_bytes()),
+ JsStr::latin1("stringify".as_bytes()),
+ // Promise object
+ JsStr::latin1("promise".as_bytes()),
+ JsStr::latin1("resolve".as_bytes()),
+ JsStr::latin1("reject".as_bytes()),
+ JsStr::latin1("all".as_bytes()),
+ JsStr::latin1("allSettled".as_bytes()),
+ JsStr::latin1("any".as_bytes()),
+ JsStr::latin1("race".as_bytes()),
+ JsStr::latin1("then".as_bytes()),
+ JsStr::latin1("catch".as_bytes()),
+ JsStr::latin1("finally".as_bytes()),
+ JsStr::latin1("withResolvers".as_bytes()),
+ // Iterator object
+ JsStr::latin1("Array Iterator".as_bytes()),
+ JsStr::latin1("Set Iterator".as_bytes()),
+ JsStr::latin1("String Iterator".as_bytes()),
+ JsStr::latin1("Map Iterator".as_bytes()),
+ JsStr::latin1("For In Iterator".as_bytes()),
+ JsStr::latin1("RegExp String Iterator".as_bytes()),
+ // Iterator result object
+ JsStr::latin1("done".as_bytes()),
+ // Math object
+ JsStr::latin1("LN10".as_bytes()),
+ JsStr::latin1("LN2".as_bytes()),
+ JsStr::latin1("LOG10E".as_bytes()),
+ JsStr::latin1("LOG2E".as_bytes()),
+ JsStr::latin1("PI".as_bytes()),
+ JsStr::latin1("SQRT1_2".as_bytes()),
+ JsStr::latin1("SQRT2".as_bytes()),
+ JsStr::latin1("abs".as_bytes()),
+ JsStr::latin1("acos".as_bytes()),
+ JsStr::latin1("acosh".as_bytes()),
+ JsStr::latin1("asin".as_bytes()),
+ JsStr::latin1("asinh".as_bytes()),
+ JsStr::latin1("atan".as_bytes()),
+ JsStr::latin1("atanh".as_bytes()),
+ JsStr::latin1("atan2".as_bytes()),
+ JsStr::latin1("cbrt".as_bytes()),
+ JsStr::latin1("ceil".as_bytes()),
+ JsStr::latin1("clz32".as_bytes()),
+ JsStr::latin1("cos".as_bytes()),
+ JsStr::latin1("cosh".as_bytes()),
+ JsStr::latin1("exp".as_bytes()),
+ JsStr::latin1("expm1".as_bytes()),
+ JsStr::latin1("floor".as_bytes()),
+ JsStr::latin1("fround".as_bytes()),
+ JsStr::latin1("hypot".as_bytes()),
+ JsStr::latin1("imul".as_bytes()),
+ JsStr::latin1("log".as_bytes()),
+ JsStr::latin1("log1p".as_bytes()),
+ JsStr::latin1("log10".as_bytes()),
+ JsStr::latin1("log2".as_bytes()),
+ JsStr::latin1("max".as_bytes()),
+ JsStr::latin1("min".as_bytes()),
+ JsStr::latin1("pow".as_bytes()),
+ JsStr::latin1("random".as_bytes()),
+ JsStr::latin1("round".as_bytes()),
+ JsStr::latin1("sign".as_bytes()),
+ JsStr::latin1("sin".as_bytes()),
+ JsStr::latin1("sinh".as_bytes()),
+ JsStr::latin1("sqrt".as_bytes()),
+ JsStr::latin1("tan".as_bytes()),
+ JsStr::latin1("tanh".as_bytes()),
+ JsStr::latin1("trunc".as_bytes()),
+ // TypedArray object
+ JsStr::latin1("BYTES_PER_ELEMENT".as_bytes()),
+ JsStr::latin1("buffer".as_bytes()),
+ JsStr::latin1("byteLength".as_bytes()),
+ JsStr::latin1("byteOffset".as_bytes()),
+ JsStr::latin1("isView".as_bytes()),
+ JsStr::latin1("subarray".as_bytes()),
+ JsStr::latin1("get byteLength".as_bytes()),
+ JsStr::latin1("get buffer".as_bytes()),
+ JsStr::latin1("get byteOffset".as_bytes()),
+ JsStr::latin1("get size".as_bytes()),
+ JsStr::latin1("get length".as_bytes()),
+ // DataView object
+ JsStr::latin1("getBigInt64".as_bytes()),
+ JsStr::latin1("getBigUint64".as_bytes()),
+ JsStr::latin1("getFloat32".as_bytes()),
+ JsStr::latin1("getFloat64".as_bytes()),
+ JsStr::latin1("getInt8".as_bytes()),
+ JsStr::latin1("getInt16".as_bytes()),
+ JsStr::latin1("getInt32".as_bytes()),
+ JsStr::latin1("getUint8".as_bytes()),
+ JsStr::latin1("getUint16".as_bytes()),
+ JsStr::latin1("getUint32".as_bytes()),
+ JsStr::latin1("setBigInt64".as_bytes()),
+ JsStr::latin1("setBigUint64".as_bytes()),
+ JsStr::latin1("setFloat32".as_bytes()),
+ JsStr::latin1("setFloat64".as_bytes()),
+ JsStr::latin1("setInt8".as_bytes()),
+ JsStr::latin1("setInt16".as_bytes()),
+ JsStr::latin1("setInt32".as_bytes()),
+ JsStr::latin1("setUint8".as_bytes()),
+ JsStr::latin1("setUint16".as_bytes()),
+ JsStr::latin1("setUint32".as_bytes()),
+ // WeakRef object
+ JsStr::latin1("deref".as_bytes()),
+ // Atomic object
+ JsStr::latin1("and".as_bytes()),
+ JsStr::latin1("compareExchange".as_bytes()),
+ JsStr::latin1("exchange".as_bytes()),
+ JsStr::latin1("isLockFree".as_bytes()),
+ JsStr::latin1("load".as_bytes()),
+ JsStr::latin1("or".as_bytes()),
+ JsStr::latin1("store".as_bytes()),
+ JsStr::latin1("wait".as_bytes()),
+ JsStr::latin1("notify".as_bytes()),
+ JsStr::latin1("xor".as_bytes()),
+ // Intl object
+ JsStr::latin1("getCanonicalLocales".as_bytes()),
+ JsStr::latin1("get compare".as_bytes()),
+ JsStr::latin1("supportedLocalesOf".as_bytes()),
+ JsStr::latin1("Intl.Collator".as_bytes()),
+ JsStr::latin1("compare".as_bytes()),
+ JsStr::latin1("resolvedOptions".as_bytes()),
+ JsStr::latin1("Intl.ListFormat".as_bytes()),
+ JsStr::latin1("format".as_bytes()),
+ JsStr::latin1("formatToParts".as_bytes()),
+ JsStr::latin1("get baseName".as_bytes()),
+ JsStr::latin1("get calendar".as_bytes()),
+ JsStr::latin1("get caseFirst".as_bytes()),
+ JsStr::latin1("get collation".as_bytes()),
+ JsStr::latin1("get hourCycle".as_bytes()),
+ JsStr::latin1("get numeric".as_bytes()),
+ JsStr::latin1("get numberingSystem".as_bytes()),
+ JsStr::latin1("get language".as_bytes()),
+ JsStr::latin1("get script".as_bytes()),
+ JsStr::latin1("get region".as_bytes()),
+ JsStr::latin1("Intl.Locale".as_bytes()),
+ JsStr::latin1("maximize".as_bytes()),
+ JsStr::latin1("minimize".as_bytes()),
+ JsStr::latin1("baseName".as_bytes()),
+ JsStr::latin1("calendar".as_bytes()),
+ JsStr::latin1("caseFirst".as_bytes()),
+ JsStr::latin1("collation".as_bytes()),
+ JsStr::latin1("hourCycle".as_bytes()),
+ JsStr::latin1("numeric".as_bytes()),
+ JsStr::latin1("numberingSystem".as_bytes()),
+ JsStr::latin1("language".as_bytes()),
+ JsStr::latin1("script".as_bytes()),
+ JsStr::latin1("region".as_bytes()),
+ JsStr::latin1("Intl.Segmenter".as_bytes()),
+ JsStr::latin1("segment".as_bytes()),
+ JsStr::latin1("containing".as_bytes()),
+ JsStr::latin1("Segmenter String Iterator".as_bytes()),
+ JsStr::latin1("Intl.PluralRules".as_bytes()),
+ JsStr::latin1("select".as_bytes()),
+ // Temporal object
+ JsStr::latin1("get Id".as_bytes()),
+ JsStr::latin1("getOffsetNanosecondsFor".as_bytes()),
+ JsStr::latin1("getOffsetStringFor".as_bytes()),
+ JsStr::latin1("getPlainDateTimeFor".as_bytes()),
+ JsStr::latin1("getInstantFor".as_bytes()),
+ JsStr::latin1("getPossibleInstantFor".as_bytes()),
+ JsStr::latin1("getNextTransition".as_bytes()),
+ JsStr::latin1("getPreviousTransition".as_bytes()),
+ JsStr::latin1("id".as_bytes()),
+ JsStr::latin1("Now".as_bytes()),
+ JsStr::latin1("Calendar".as_bytes()),
+ JsStr::latin1("Duration".as_bytes()),
+ JsStr::latin1("Instant".as_bytes()),
+ JsStr::latin1("PlainDate".as_bytes()),
+ JsStr::latin1("PlainDateTime".as_bytes()),
+ JsStr::latin1("PlainMonthDay".as_bytes()),
+ JsStr::latin1("PlainTime".as_bytes()),
+ JsStr::latin1("PlainYearMonth".as_bytes()),
+ JsStr::latin1("TimeZone".as_bytes()),
+ JsStr::latin1("ZonedDateTime".as_bytes()),
+ JsStr::latin1("timeZoneId".as_bytes()),
+ JsStr::latin1("instant".as_bytes()),
+ JsStr::latin1("plainDateTime".as_bytes()),
+ JsStr::latin1("plainDateTimeISO".as_bytes()),
+ JsStr::latin1("zonedDateTime".as_bytes()),
+ JsStr::latin1("zonedDateTimeISO".as_bytes()),
+ JsStr::latin1("plainDate".as_bytes()),
+ JsStr::latin1("plainDateISO".as_bytes()),
+ JsStr::latin1("get epochSeconds".as_bytes()),
+ JsStr::latin1("get epochMilliseconds".as_bytes()),
+ JsStr::latin1("get epochMicroseconds".as_bytes()),
+ JsStr::latin1("get epochNanoseconds".as_bytes()),
+ JsStr::latin1("epochSeconds".as_bytes()),
+ JsStr::latin1("epochMilliseconds".as_bytes()),
+ JsStr::latin1("epochMicroseconds".as_bytes()),
+ JsStr::latin1("epochNanoseconds".as_bytes()),
+ JsStr::latin1("subtract".as_bytes()),
+ JsStr::latin1("until".as_bytes()),
+ JsStr::latin1("since".as_bytes()),
+ JsStr::latin1("equals".as_bytes()),
+ JsStr::latin1("toZonedDateTime".as_bytes()),
+ JsStr::latin1("toZonedDateTimeISO".as_bytes()),
+ JsStr::latin1("get Years".as_bytes()),
+ JsStr::latin1("get Months".as_bytes()),
+ JsStr::latin1("get Weeks".as_bytes()),
+ JsStr::latin1("get Days".as_bytes()),
+ JsStr::latin1("get Hours".as_bytes()),
+ JsStr::latin1("get Minutes".as_bytes()),
+ JsStr::latin1("get Seconds".as_bytes()),
+ JsStr::latin1("get Milliseconds".as_bytes()),
+ JsStr::latin1("get Microseconds".as_bytes()),
+ JsStr::latin1("get Nanoseconds".as_bytes()),
+ JsStr::latin1("get Sign".as_bytes()),
+ JsStr::latin1("get blank".as_bytes()),
+ JsStr::latin1("years".as_bytes()),
+ JsStr::latin1("months".as_bytes()),
+ JsStr::latin1("weeks".as_bytes()),
+ JsStr::latin1("days".as_bytes()),
+ JsStr::latin1("hours".as_bytes()),
+ JsStr::latin1("minutes".as_bytes()),
+ JsStr::latin1("seconds".as_bytes()),
+ JsStr::latin1("milliseconds".as_bytes()),
+ JsStr::latin1("microseconds".as_bytes()),
+ JsStr::latin1("nanoseconds".as_bytes()),
+ JsStr::latin1("blank".as_bytes()),
+ JsStr::latin1("negated".as_bytes()),
+ JsStr::latin1("total".as_bytes()),
+ JsStr::latin1("get calendarId".as_bytes()),
+ JsStr::latin1("get year".as_bytes()),
+ JsStr::latin1("get month".as_bytes()),
+ JsStr::latin1("get monthCode".as_bytes()),
+ JsStr::latin1("get day".as_bytes()),
+ JsStr::latin1("get dayOfWeek".as_bytes()),
+ JsStr::latin1("get dayOfYear".as_bytes()),
+ JsStr::latin1("get weekOfYear".as_bytes()),
+ JsStr::latin1("get yearOfWeek".as_bytes()),
+ JsStr::latin1("get daysInWeek".as_bytes()),
+ JsStr::latin1("get daysInMonth".as_bytes()),
+ JsStr::latin1("get daysInYear".as_bytes()),
+ JsStr::latin1("get monthsInYear".as_bytes()),
+ JsStr::latin1("get inLeapYear".as_bytes()),
+ JsStr::latin1("calendarId".as_bytes()),
+ JsStr::latin1("year".as_bytes()),
+ JsStr::latin1("month".as_bytes()),
+ JsStr::latin1("monthCode".as_bytes()),
+ JsStr::latin1("day".as_bytes()),
+ JsStr::latin1("dayOfWeek".as_bytes()),
+ JsStr::latin1("dayOfYear".as_bytes()),
+ JsStr::latin1("weekOfYear".as_bytes()),
+ JsStr::latin1("yearOfWeek".as_bytes()),
+ JsStr::latin1("daysInWeek".as_bytes()),
+ JsStr::latin1("daysInMonth".as_bytes()),
+ JsStr::latin1("daysInYear".as_bytes()),
+ JsStr::latin1("monthsInYear".as_bytes()),
+ JsStr::latin1("inLeapYear".as_bytes()),
+ JsStr::latin1("toPlainYearMonth".as_bytes()),
+ JsStr::latin1("toPlainMonthDay".as_bytes()),
+ JsStr::latin1("getISOFields".as_bytes()),
+ JsStr::latin1("getCalendar".as_bytes()),
+ JsStr::latin1("withCalendar".as_bytes()),
+ JsStr::latin1("dateFromFields".as_bytes()),
+ JsStr::latin1("yearMonthFromFields".as_bytes()),
+ JsStr::latin1("monthDayFromFields".as_bytes()),
+ JsStr::latin1("dateAdd".as_bytes()),
+ JsStr::latin1("dateUntil".as_bytes()),
+ JsStr::latin1("era".as_bytes()),
+ JsStr::latin1("eraYear".as_bytes()),
+ JsStr::latin1("fields".as_bytes()),
+ JsStr::latin1("mergeFields".as_bytes()),
+ // Console object
+ JsStr::latin1("console".as_bytes()),
+ JsStr::latin1("assert".as_bytes()),
+ JsStr::latin1("debug".as_bytes()),
+ JsStr::latin1("error".as_bytes()),
+ JsStr::latin1("info".as_bytes()),
+ JsStr::latin1("trace".as_bytes()),
+ JsStr::latin1("warn".as_bytes()),
+ JsStr::latin1("exception".as_bytes()),
+ JsStr::latin1("count".as_bytes()),
+ JsStr::latin1("countReset".as_bytes()),
+ JsStr::latin1("group".as_bytes()),
+ JsStr::latin1("groupCollapsed".as_bytes()),
+ JsStr::latin1("groupEnd".as_bytes()),
+ JsStr::latin1("time".as_bytes()),
+ JsStr::latin1("timeLog".as_bytes()),
+ JsStr::latin1("timeEnd".as_bytes()),
+ JsStr::latin1("dir".as_bytes()),
+ JsStr::latin1("dirxml".as_bytes()),
+ // Minified name
+ JsStr::latin1("a".as_bytes()),
+ JsStr::latin1("c".as_bytes()),
+ JsStr::latin1("d".as_bytes()),
+ JsStr::latin1("e".as_bytes()),
+ JsStr::latin1("f".as_bytes()),
+ JsStr::latin1("g".as_bytes()),
+ JsStr::latin1("h".as_bytes()),
+ JsStr::latin1("i".as_bytes()),
+ JsStr::latin1("j".as_bytes()),
+ JsStr::latin1("k".as_bytes()),
+ JsStr::latin1("l".as_bytes()),
+ JsStr::latin1("m".as_bytes()),
+ JsStr::latin1("n".as_bytes()),
+ JsStr::latin1("o".as_bytes()),
+ JsStr::latin1("p".as_bytes()),
+ JsStr::latin1("q".as_bytes()),
+ JsStr::latin1("r".as_bytes()),
+ JsStr::latin1("s".as_bytes()),
+ JsStr::latin1("t".as_bytes()),
+ JsStr::latin1("u".as_bytes()),
+ JsStr::latin1("v".as_bytes()),
+ JsStr::latin1("w".as_bytes()),
+ JsStr::latin1("x".as_bytes()),
+ JsStr::latin1("y".as_bytes()),
+ JsStr::latin1("z".as_bytes()),
+ JsStr::latin1("A".as_bytes()),
+ JsStr::latin1("C".as_bytes()),
+ JsStr::latin1("D".as_bytes()),
+ JsStr::latin1("E".as_bytes()),
+ JsStr::latin1("F".as_bytes()),
+ JsStr::latin1("G".as_bytes()),
+ JsStr::latin1("H".as_bytes()),
+ JsStr::latin1("I".as_bytes()),
+ JsStr::latin1("J".as_bytes()),
+ JsStr::latin1("K".as_bytes()),
+ JsStr::latin1("L".as_bytes()),
+ JsStr::latin1("M".as_bytes()),
+ JsStr::latin1("N".as_bytes()),
+ JsStr::latin1("O".as_bytes()),
+ JsStr::latin1("P".as_bytes()),
+ JsStr::latin1("Q".as_bytes()),
+ JsStr::latin1("R".as_bytes()),
+ JsStr::latin1("S".as_bytes()),
+ JsStr::latin1("T".as_bytes()),
+ JsStr::latin1("U".as_bytes()),
+ JsStr::latin1("V".as_bytes()),
+ JsStr::latin1("W".as_bytes()),
+ JsStr::latin1("X".as_bytes()),
+ JsStr::latin1("Y".as_bytes()),
+ JsStr::latin1("Z".as_bytes()),
+ JsStr::latin1("_".as_bytes()),
+ JsStr::latin1("$".as_bytes()),
+];
diff --git a/core/engine/src/string/iter.rs b/core/string/src/iter.rs
similarity index 97%
rename from core/engine/src/string/iter.rs
rename to core/string/src/iter.rs
index 03dedcb6b8d..ead0e8c94c7 100644
--- a/core/engine/src/string/iter.rs
+++ b/core/string/src/iter.rs
@@ -29,6 +29,7 @@ impl<'a> Iter<'a> {
impl Iterator for Iter<'_> {
type Item = u16;
+ #[inline]
fn next(&mut self) -> Option {
match &mut self.inner {
IterInner::U8(iter) => iter.map(u16::from).next(),
@@ -40,6 +41,7 @@ impl Iterator for Iter<'_> {
impl FusedIterator for Iter<'_> {}
impl ExactSizeIterator for Iter<'_> {
+ #[inline]
fn len(&self) -> usize {
match &self.inner {
IterInner::U8(v) => v.len(),
@@ -75,6 +77,7 @@ impl<'a> Windows<'a> {
impl<'a> Iterator for Windows<'a> {
type Item = JsStr<'a>;
+ #[inline]
fn next(&mut self) -> Option {
match &mut self.inner {
WindowsInner::U8(iter) => iter.next().map(JsStr::latin1),
@@ -86,6 +89,7 @@ impl<'a> Iterator for Windows<'a> {
impl FusedIterator for Windows<'_> {}
impl ExactSizeIterator for Windows<'_> {
+ #[inline]
fn len(&self) -> usize {
match &self.inner {
WindowsInner::U8(v) => v.len(),
diff --git a/core/engine/src/string/mod.rs b/core/string/src/lib.rs
similarity index 82%
rename from core/engine/src/string/mod.rs
rename to core/string/src/lib.rs
index 16f21727310..2daae7d5c6a 100644
--- a/core/engine/src/string/mod.rs
+++ b/core/string/src/lib.rs
@@ -1,10 +1,4 @@
//! A Latin1 or UTF-16 encoded, reference counted, immutable string.
-//!
-//! This module contains the [`JsString`] type, the [`js_string`][crate::js_string] macro and the
-//! [`js_str`][crate::js_str] macro.
-//!
-//! The [`js_string`][crate::js_string] macro is used when you need to create a new [`JsString`],
-//! and the [`js_str`][crate::js_str] macro is used for const conversions of string literals to [`JsStr`].
// Required per unsafe code standards to ensure every unsafe usage is properly documented.
// - `unsafe_op_in_unsafe_fn` will be warn-by-default in edition 2024:
@@ -20,22 +14,24 @@
// Right now this allows us to use the stable polyfill from the `sptr` crate, which uses
// the same names from the unstable functions of the `std::ptr` module.
#![allow(unstable_name_collisions)]
+#![allow(clippy::module_name_repetitions)]
-pub mod common;
+mod common;
mod iter;
mod str;
+mod tagged;
-use self::{common::StaticJsStrings, iter::Windows, str::JsSliceIndex};
+#[cfg(test)]
+mod tests;
+
+use self::{iter::Windows, str::JsSliceIndex};
+use crate::tagged::{Tagged, UnwrappedTagged};
#[doc(inline)]
-pub use crate::string::{
+pub use crate::{
+ common::StaticJsStrings,
iter::Iter,
str::{JsStr, JsStrVariant},
};
-use crate::{
- builtins::string::is_trimmable_whitespace,
- tagged::{Tagged, UnwrappedTagged},
-};
-use boa_gc::{Finalize, Trace};
use std::{
alloc::{alloc, dealloc, Layout},
cell::Cell,
@@ -51,65 +47,41 @@ fn alloc_overflow() -> ! {
panic!("detected overflow during string allocation")
}
-/// Utility macro to create a [`JsString`].
-///
-/// # Examples
-///
-/// You can call the macro without arguments to create an empty `JsString`:
-///
-/// ```
-/// use boa_engine::js_string;
-///
-/// let empty_str = js_string!();
-/// assert!(empty_str.is_empty());
-/// ```
-///
-///
-/// You can create a `JsString` from a string literal, which completely skips the runtime
-/// conversion from [`&str`] to [&\[u16\]][slice]
:
-///
-/// ```
-/// # use boa_engine::js_string;
-/// let hw = js_string!("Hello, world!");
-/// assert_eq!(&hw, "Hello, world!");
-/// ```
-///
-/// Any `&[u16]` slice is a valid `JsString`, including unpaired surrogates:
-///
-/// ```
-/// # use boa_engine::js_string;
-/// let array = js_string!(&[0xD8AFu16, 0x00A0, 0xD8FF, 0x00F0]);
-/// ```
-///
-/// You can also pass it any number of `&[u16]` as arguments to create a new `JsString` with
-/// the concatenation of every slice:
-///
-/// ```
-/// # use boa_engine::{js_string, js_str, JsStr};
-/// const NAME: JsStr<'_> = js_str!("human! ");
-/// let greeting = js_string!("Hello, ");
-/// let msg = js_string!(&greeting, NAME, js_str!("Nice to meet you!"));
-///
-/// assert_eq!(&msg, "Hello, human! Nice to meet you!");
-/// ```
-#[macro_export]
-#[allow(clippy::module_name_repetitions)]
-macro_rules! js_string {
- () => {
- $crate::string::JsString::default()
- };
- ($s:literal) => {
- $crate::string::JsString::from($crate::js_str!($s))
- };
- ($s:expr) => {
- $crate::string::JsString::from($s)
- };
- ( $x:expr, $y:expr ) => {
- $crate::string::JsString::concat($crate::string::JsStr::from($x), $crate::string::JsStr::from($y))
- };
- ( $( $s:expr ),+ ) => {
- $crate::string::JsString::concat_array(&[ $( $crate::string::JsStr::from($s) ),+ ])
- };
+/// Helper function to check if a `char` is trimmable.
+pub(crate) const fn is_trimmable_whitespace(c: char) -> bool {
+ // The rust implementation of `trim` does not regard the same characters whitespace as ecma standard does
+ //
+ // Rust uses \p{White_Space} by default, which also includes:
+ // `\u{0085}' (next line)
+ // And does not include:
+ // '\u{FEFF}' (zero width non-breaking space)
+ // Explicit whitespace: https://tc39.es/ecma262/#sec-white-space
+ matches!(
+ c,
+ '\u{0009}' | '\u{000B}' | '\u{000C}' | '\u{0020}' | '\u{00A0}' | '\u{FEFF}' |
+ // Unicode Space_Separator category
+ '\u{1680}' | '\u{2000}'
+ ..='\u{200A}' | '\u{202F}' | '\u{205F}' | '\u{3000}' |
+ // Line terminators: https://tc39.es/ecma262/#sec-line-terminators
+ '\u{000A}' | '\u{000D}' | '\u{2028}' | '\u{2029}'
+ )
+}
+
+/// Helper function to check if a `u8` latin1 character is trimmable.
+pub(crate) const fn is_trimmable_whitespace_latin1(c: u8) -> bool {
+ // The rust implementation of `trim` does not regard the same characters whitespace as ecma standard does
+ //
+ // Rust uses \p{White_Space} by default, which also includes:
+ // `\u{0085}' (next line)
+ // And does not include:
+ // '\u{FEFF}' (zero width non-breaking space)
+ // Explicit whitespace: https://tc39.es/ecma262/#sec-white-space
+ matches!(
+ c,
+ 0x09 | 0x0B | 0x0C | 0x20 | 0xA0 |
+ // Line terminators: https://tc39.es/ecma262/#sec-line-terminators
+ 0x0A | 0x0D
+ )
}
/// Represents a Unicode codepoint within a [`JsString`], which could be a valid
@@ -127,6 +99,7 @@ pub enum CodePoint {
impl CodePoint {
/// Get the number of UTF-16 code units needed to encode this code point.
+ #[inline]
#[must_use]
pub const fn code_unit_count(self) -> usize {
match self {
@@ -136,6 +109,7 @@ impl CodePoint {
}
/// Convert the code point to its [`u32`] representation.
+ #[inline]
#[must_use]
pub fn as_u32(self) -> u32 {
match self {
@@ -146,6 +120,7 @@ impl CodePoint {
/// If the code point represents a valid 'Unicode scalar value', returns its [`char`]
/// representation, otherwise returns [`None`] on unpaired surrogates.
+ #[inline]
#[must_use]
pub const fn as_char(self) -> Option {
match self {
@@ -161,6 +136,8 @@ impl CodePoint {
///
/// Panics if the buffer is not large enough. A buffer of length 2 is large enough to encode any
/// code point.
+ #[inline]
+ #[must_use]
pub fn encode_utf16(self, dst: &mut [u16]) -> &mut [u16] {
match self {
Self::Unicode(c) => c.encode_utf16(dst),
@@ -218,9 +195,6 @@ const DATA_OFFSET: usize = std::mem::size_of::();
///
/// We define some commonly used string constants in an interner. For these strings, we don't allocate
/// memory on the heap to reduce the overhead of memory allocation and reference counting.
-#[derive(Trace, Finalize)]
-// Safety: `JsString` does not contain any objects which needs to be traced, so this is safe.
-#[boa_gc(unsafe_empty_trace)]
#[allow(clippy::module_name_repetitions)]
pub struct JsString {
ptr: Tagged,
@@ -230,6 +204,7 @@ pub struct JsString {
static_assertions::assert_eq_size!(JsString, *const ());
impl<'a> From<&'a JsString> for JsStr<'a> {
+ #[inline]
fn from(value: &'a JsString) -> Self {
value.as_str()
}
@@ -237,8 +212,9 @@ impl<'a> From<&'a JsString> for JsStr<'a> {
impl<'a> IntoIterator for &'a JsString {
type IntoIter = Iter<'a>;
-
type Item = u16;
+
+ #[inline]
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
@@ -260,6 +236,7 @@ impl JsString {
}
/// Obtains the underlying [`&[u16]`][slice] slice of a [`JsString`]
+ #[inline]
#[must_use]
pub fn as_str(&self) -> JsStr<'_> {
match self.ptr.unwrap() {
@@ -298,6 +275,7 @@ impl JsString {
}
/// Creates a new [`JsString`] from the concatenation of `x` and `y`.
+ #[inline]
#[must_use]
pub fn concat(x: JsStr<'_>, y: JsStr<'_>) -> Self {
Self::concat_array(&[x, y])
@@ -305,6 +283,7 @@ impl JsString {
/// Creates a new [`JsString`] from the concatenation of every element of
/// `strings`.
+ #[inline]
#[must_use]
pub fn concat_array(strings: &[JsStr<'_>]) -> Self {
let mut latin1_encoding = true;
@@ -373,6 +352,7 @@ impl JsString {
/// Decodes a [`JsString`] into a [`String`], replacing invalid data with its escaped representation
/// in 4 digit hexadecimal.
+ #[inline]
#[must_use]
pub fn to_std_string_escaped(&self) -> String {
self.to_string_escaped()
@@ -383,6 +363,7 @@ impl JsString {
/// # Errors
///
/// [`FromUtf16Error`][std::string::FromUtf16Error] if it contains any invalid data.
+ #[inline]
pub fn to_std_string(&self) -> Result {
match self.as_str().variant() {
JsStrVariant::Latin1(v) => Ok(v.iter().copied().map(char::from).collect()),
@@ -392,6 +373,7 @@ impl JsString {
/// Decodes a [`JsString`] into an iterator of [`Result`], returning surrogates as
/// errors.
+ #[inline]
pub fn to_std_string_with_surrogates(&self) -> impl Iterator- > + '_ {
struct WideStringDecoderIterator {
codepoints: Peekable,
@@ -443,6 +425,7 @@ impl JsString {
}
/// Maps the valid segments of an UTF16 string and leaves the unpaired surrogates unchanged.
+ #[inline]
#[must_use]
pub fn map_valid_segments(&self, mut f: F) -> Self
where
@@ -457,10 +440,11 @@ impl JsString {
}
}
- js_string!(&text[..])
+ Self::from(&text[..])
}
/// Gets an iterator of all the Unicode codepoints of a [`JsString`].
+ #[inline]
pub fn code_points(&self) -> impl Iterator
- + Clone + '_ {
char::decode_utf16(self.iter()).map(|res| match res {
Ok(c) => CodePoint::Unicode(c),
@@ -477,6 +461,7 @@ impl JsString {
/// - [ECMAScript reference][spec]
///
/// [spec]: https://tc39.es/ecma262/#sec-stringindexof
+ #[inline]
#[must_use]
pub fn index_of(&self, search_value: JsStr<'_>, from_index: usize) -> Option {
// 1. Assert: Type(string) is String.
@@ -522,6 +507,7 @@ impl JsString {
/// # Panics
///
/// If `position` is smaller than size of string.
+ #[inline]
#[must_use]
pub fn code_point_at(&self, position: usize) -> CodePoint {
// 1. Let size be the length of string.
@@ -569,6 +555,8 @@ impl JsString {
/// - [ECMAScript reference][spec]
///
/// [spec]: https://tc39.es/ecma262/#sec-stringtonumber
+ #[inline]
+ #[must_use]
pub fn to_number(&self) -> f64 {
// 1. Let text be ! StringToCodePoints(str).
// 2. Let literal be ParseText(text, StringNumericLiteral).
@@ -781,12 +769,14 @@ impl JsString {
}
/// Convert the [`JsString`] into a [`Vec`].
+ #[inline]
#[must_use]
pub fn to_vec(&self) -> Vec {
self.as_str().to_vec()
}
/// Check if the [`JsString`] contains a byte.
+ #[inline]
#[must_use]
pub fn contains(&self, element: u8) -> bool {
match self.as_str().variant() {
@@ -796,30 +786,35 @@ impl JsString {
}
/// Trim whitespace from the start and end of the [`JsString`].
+ #[inline]
#[must_use]
pub fn trim(&self) -> JsStr<'_> {
self.as_str().trim()
}
/// Trim whitespace from the start of the [`JsString`].
+ #[inline]
#[must_use]
pub fn trim_start(&self) -> JsStr<'_> {
self.as_str().trim_start()
}
/// Trim whitespace from the end of the [`JsString`].
+ #[inline]
#[must_use]
pub fn trim_end(&self) -> JsStr<'_> {
self.as_str().trim_end()
}
/// Check if the [`JsString`] is static.
+ #[inline]
#[must_use]
pub fn is_static(&self) -> bool {
self.ptr.is_tagged()
}
/// Get the element a the given index, [`None`] otherwise.
+ #[inline]
#[must_use]
pub fn get<'a, I>(&'a self, index: I) -> Option
where
@@ -833,6 +828,7 @@ impl JsString {
/// # Panics
///
/// If the index is out of bounds.
+ #[inline]
#[must_use]
pub fn get_expect<'a, I>(&'a self, index: I) -> I::Value
where
@@ -840,6 +836,20 @@ impl JsString {
{
self.get(index).expect("Index out of bounds")
}
+
+ /// Gets the number of `JsString`s which point to this allocation.
+ #[inline]
+ #[must_use]
+ pub fn refcount(&self) -> Option {
+ match self.ptr.unwrap() {
+ UnwrappedTagged::Ptr(inner) => {
+ // SAFETY: The reference count of `JsString` guarantees that `inner` is always valid.
+ let inner = unsafe { inner.as_ref() };
+ Some(inner.refcount.get())
+ }
+ UnwrappedTagged::Tag(_inner) => None,
+ }
+ }
}
impl Clone for JsString {
@@ -866,6 +876,7 @@ impl Default for JsString {
}
impl Drop for JsString {
+ #[inline]
fn drop(&mut self) {
if let UnwrappedTagged::Ptr(raw) = self.ptr.unwrap() {
// See https://doc.rust-lang.org/src/alloc/sync.rs.html#1672 for details.
@@ -947,6 +958,7 @@ impl From<&str> for JsString {
}
impl From> for JsString {
+ #[inline]
fn from(value: JsStr<'_>) -> Self {
StaticJsStrings::get_string(&value)
.unwrap_or_else(|| JsString::from_slice_skip_interning(value))
@@ -954,6 +966,7 @@ impl From> for JsString {
}
impl From<&[JsString]> for JsString {
+ #[inline]
fn from(value: &[JsString]) -> Self {
Self::concat_array(
&value
@@ -972,36 +985,42 @@ impl From for JsString {
}
impl From<&[u16; N]> for JsString {
+ #[inline]
fn from(s: &[u16; N]) -> Self {
Self::from(&s[..])
}
}
impl Hash for JsString {
+ #[inline]
fn hash(&self, state: &mut H) {
self.as_str().hash(state);
}
}
impl PartialOrd for JsStr<'_> {
+ #[inline]
fn partial_cmp(&self, other: &Self) -> Option {
Some(self.cmp(other))
}
}
impl Ord for JsString {
+ #[inline]
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.as_str().cmp(&other.as_str())
}
}
impl PartialEq for JsString {
+ #[inline]
fn eq(&self, other: &Self) -> bool {
self.as_str() == other.as_str()
}
}
impl PartialEq for [u16] {
+ #[inline]
fn eq(&self, other: &JsString) -> bool {
if self.len() != other.len() {
return false;
@@ -1016,24 +1035,28 @@ impl PartialEq for [u16] {
}
impl PartialEq for [u16; N] {
+ #[inline]
fn eq(&self, other: &JsString) -> bool {
self[..] == *other
}
}
impl PartialEq<[u16]> for JsString {
+ #[inline]
fn eq(&self, other: &[u16]) -> bool {
other == self
}
}
impl PartialEq<[u16; N]> for JsString {
+ #[inline]
fn eq(&self, other: &[u16; N]) -> bool {
*self == other[..]
}
}
impl PartialEq for JsString {
+ #[inline]
fn eq(&self, other: &str) -> bool {
let utf16 = self.code_points();
let mut utf8 = other.chars();
@@ -1052,24 +1075,28 @@ impl PartialEq for JsString {
}
impl PartialEq for str {
+ #[inline]
fn eq(&self, other: &JsString) -> bool {
other == self
}
}
impl PartialEq> for JsString {
+ #[inline]
fn eq(&self, other: &JsStr<'_>) -> bool {
self.as_str() == *other
}
}
impl PartialEq for JsStr<'_> {
+ #[inline]
fn eq(&self, other: &JsString) -> bool {
other == self
}
}
impl PartialOrd for JsString {
+ #[inline]
fn partial_cmp(&self, other: &Self) -> Option {
Some(self.cmp(other))
}
@@ -1078,6 +1105,7 @@ impl PartialOrd for JsString {
impl FromStr for JsString {
type Err = Infallible;
+ #[inline]
fn from_str(s: &str) -> Result {
Ok(Self::from(s))
}
@@ -1091,6 +1119,7 @@ pub(crate) trait ToStringEscaped {
}
impl ToStringEscaped for [u16] {
+ #[inline]
fn to_string_escaped(&self) -> String {
char::decode_utf16(self.iter().copied())
.map(|r| match r {
@@ -1100,187 +1129,3 @@ impl ToStringEscaped for [u16] {
.collect()
}
}
-
-#[allow(clippy::redundant_clone)]
-#[cfg(test)]
-mod tests {
- use std::hash::{BuildHasher, BuildHasherDefault, Hash};
-
- use crate::{string::common::StaticJsStrings, tagged::UnwrappedTagged, JsStr};
-
- use super::JsString;
- use boa_macros::{js_str, utf16};
- use rustc_hash::FxHasher;
-
- impl JsString {
- /// Gets the number of `JsString`s which point to this allocation.
- fn refcount(&self) -> Option {
- match self.ptr.unwrap() {
- UnwrappedTagged::Ptr(inner) => {
- // SAFETY: The reference count of `JsString` guarantees that `inner` is always valid.
- let inner = unsafe { inner.as_ref() };
- Some(inner.refcount.get())
- }
- UnwrappedTagged::Tag(_inner) => None,
- }
- }
- }
-
- fn hash_value(value: &T) -> u64 {
- BuildHasherDefault::::default().hash_one(value)
- }
-
- #[test]
- fn empty() {
- let s = js_string!();
- assert_eq!(&s, utf16!(""));
- }
-
- #[test]
- fn refcount() {
- let x = js_string!("Hello world");
- assert_eq!(x.refcount(), Some(1));
-
- {
- let y = x.clone();
- assert_eq!(x.refcount(), Some(2));
- assert_eq!(y.refcount(), Some(2));
-
- {
- let z = y.clone();
- assert_eq!(x.refcount(), Some(3));
- assert_eq!(y.refcount(), Some(3));
- assert_eq!(z.refcount(), Some(3));
- }
-
- assert_eq!(x.refcount(), Some(2));
- assert_eq!(y.refcount(), Some(2));
- }
-
- assert_eq!(x.refcount(), Some(1));
- }
-
- #[test]
- fn static_refcount() {
- let x = js_string!();
- assert_eq!(x.refcount(), None);
-
- {
- let y = x.clone();
- assert_eq!(x.refcount(), None);
- assert_eq!(y.refcount(), None);
- };
-
- assert_eq!(x.refcount(), None);
- }
-
- #[test]
- fn ptr_eq() {
- let x = js_string!("Hello");
- let y = x.clone();
-
- assert!(!x.ptr.is_tagged());
-
- assert_eq!(x.ptr.addr(), y.ptr.addr());
-
- let z = js_string!("Hello");
- assert_ne!(x.ptr.addr(), z.ptr.addr());
- assert_ne!(y.ptr.addr(), z.ptr.addr());
- }
-
- #[test]
- fn static_ptr_eq() {
- let x = js_string!();
- let y = x.clone();
-
- assert!(x.ptr.is_tagged());
-
- assert_eq!(x.ptr.addr(), y.ptr.addr());
-
- let z = js_string!();
- assert_eq!(x.ptr.addr(), z.ptr.addr());
- assert_eq!(y.ptr.addr(), z.ptr.addr());
- }
-
- #[test]
- fn as_str() {
- const HELLO: &[u16] = utf16!("Hello");
- let x = js_string!(HELLO);
-
- assert_eq!(&x, HELLO);
- }
-
- #[test]
- fn hash() {
- const HELLOWORLD: JsStr<'_> = js_str!("Hello World!");
- let x = js_string!(HELLOWORLD);
-
- assert_eq!(x.as_str(), HELLOWORLD);
-
- assert!(HELLOWORLD.is_latin1());
- assert!(x.as_str().is_latin1());
-
- let s_hash = hash_value(&HELLOWORLD);
- let x_hash = hash_value(&x);
-
- assert_eq!(s_hash, x_hash);
- }
-
- #[test]
- fn concat() {
- const Y: &[u16] = utf16!(", ");
- const W: &[u16] = utf16!("!");
-
- let x = js_string!("hello");
- let z = js_string!("world");
-
- let xy = js_string!(&x, &JsString::from(Y));
- assert_eq!(&xy, utf16!("hello, "));
- assert_eq!(xy.refcount(), Some(1));
-
- let xyz = js_string!(&xy, &z);
- assert_eq!(&xyz, utf16!("hello, world"));
- assert_eq!(xyz.refcount(), Some(1));
-
- let xyzw = js_string!(&xyz, &JsString::from(W));
- assert_eq!(&xyzw, utf16!("hello, world!"));
- assert_eq!(xyzw.refcount(), Some(1));
- }
-
- #[test]
- fn trim_start_non_ascii_to_ascii() {
- let s = "\u{2029}abc";
- let x = js_string!(s);
-
- let y = js_string!(x.trim_start());
-
- assert_eq!(&y, s.trim_start());
- }
-
- #[test]
- fn conversion_to_known_static_js_string() {
- const JS_STR_U8: &JsStr<'_> = &js_str!("length");
- const JS_STR_U16: &JsStr<'_> = &JsStr::utf16(utf16!("length"));
-
- assert!(JS_STR_U8.is_latin1());
- assert!(!JS_STR_U16.is_latin1());
-
- assert_eq!(JS_STR_U8, JS_STR_U8);
- assert_eq!(JS_STR_U16, JS_STR_U16);
-
- assert_eq!(JS_STR_U8, JS_STR_U16);
- assert_eq!(JS_STR_U16, JS_STR_U8);
-
- assert_eq!(hash_value(JS_STR_U8), hash_value(JS_STR_U16));
-
- let string = StaticJsStrings::get_string(JS_STR_U8);
-
- assert!(string.is_some());
- assert!(string.unwrap().as_str().is_latin1());
-
- let string = StaticJsStrings::get_string(JS_STR_U16);
-
- assert!(string.is_some());
- assert!(string.unwrap().as_str().is_latin1());
- }
-}
diff --git a/core/engine/src/string/str.rs b/core/string/src/str.rs
similarity index 96%
rename from core/engine/src/string/str.rs
rename to core/string/src/str.rs
index 6ce3c4cabd4..ef31ac7a276 100644
--- a/core/engine/src/string/str.rs
+++ b/core/string/src/str.rs
@@ -1,7 +1,4 @@
-use crate::{
- builtins::string::{is_trimmable_whitespace, is_trimmable_whitespace_latin1},
- string::Iter,
-};
+use crate::{is_trimmable_whitespace, is_trimmable_whitespace_latin1, Iter};
use std::{
hash::{Hash, Hasher},
slice::SliceIndex,
@@ -195,6 +192,8 @@ impl<'a> JsStr<'a> {
}
/// Convert the [`JsStr`] into a [`Vec`].
+ #[inline]
+ #[must_use]
pub fn to_vec(&self) -> Vec {
match self.variant() {
JsStrVariant::Latin1(v) => v.iter().copied().map(u16::from).collect(),
@@ -203,13 +202,19 @@ impl<'a> JsStr<'a> {
}
/// Returns true if needle is a prefix of the [`JsStr`].
+ #[inline]
#[must_use]
+ // We check the size, so this should never panic.
+ #[allow(clippy::missing_panics_doc)]
pub fn starts_with(&self, needle: JsStr<'_>) -> bool {
let n = needle.len();
self.len() >= n && needle == self.get(..n).expect("already checked size")
}
/// Returns `true` if `needle` is a suffix of the [`JsStr`].
+ #[inline]
#[must_use]
+ // We check the size, so this should never panic.
+ #[allow(clippy::missing_panics_doc)]
pub fn ends_with(&self, needle: JsStr<'_>) -> bool {
let (m, n) = (self.len(), needle.len());
m >= n && needle == self.get(m - n..).expect("already checked size")
@@ -217,6 +222,7 @@ impl<'a> JsStr<'a> {
}
impl Hash for JsStr<'_> {
+ #[inline]
fn hash(&self, state: &mut H) {
// NOTE: The hash function has been inlined to ensure that a hash of latin1 and U16
// encoded strings remains the same if they have the same characters
@@ -238,6 +244,7 @@ impl Hash for JsStr<'_> {
}
impl Ord for JsStr<'_> {
+ #[inline]
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
match (self.variant(), other.variant()) {
(JsStrVariant::Latin1(x), JsStrVariant::Latin1(y)) => x.cmp(y),
@@ -250,6 +257,7 @@ impl Ord for JsStr<'_> {
impl Eq for JsStr<'_> {}
impl PartialEq for JsStr<'_> {
+ #[inline]
fn eq(&self, other: &Self) -> bool {
match (self.variant(), other.variant()) {
(JsStrVariant::Latin1(lhs), JsStrVariant::Latin1(rhs)) => return lhs == rhs,
@@ -269,6 +277,7 @@ impl PartialEq for JsStr<'_> {
}
impl<'a> PartialEq> for [u16] {
+ #[inline]
fn eq(&self, other: &JsStr<'a>) -> bool {
if self.len() != other.len() {
return false;
diff --git a/core/string/src/tagged.rs b/core/string/src/tagged.rs
new file mode 100644
index 00000000000..83ec17c24b4
--- /dev/null
+++ b/core/string/src/tagged.rs
@@ -0,0 +1,109 @@
+// Remove when/if https://github.com/rust-lang/rust/issues/95228 stabilizes.
+// Right now this allows us to use the stable polyfill from the `sptr` crate, which uses
+// the same names from the unstable functions of the `std::ptr` module.
+#![allow(unstable_name_collisions)]
+
+use sptr::Strict;
+use std::ptr::NonNull;
+
+/// A pointer that can be tagged with an `usize`.
+///
+/// Only pointers with a minimum alignment of 2-bytes are valid, and the tag must have its most
+/// significant bit (MSB) unset. In other words, the tag must fit inside `usize::BITS - 1` bits.
+///
+/// # Representation
+///
+/// If the least significant bit (LSB) of the internal [`NonNull`] is set (1), then the pointer
+/// address represents a tag where the remaining bits store the tag. Otherwise, the whole pointer
+/// represents the pointer itself.
+///
+/// It uses [`NonNull`], which guarantees that [`Tagged`] can use the "null pointer optimization"
+/// to optimize the size of [`Option`].
+///
+/// # Provenance
+///
+/// This struct stores a [`NonNull`] instead of a [`NonZeroUsize`][std::num::NonZeroUsize]
+/// in order to preserve the provenance of our valid heap pointers.
+/// On the other hand, all index values are just casted to invalid pointers, because we don't need to
+/// preserve the provenance of [`usize`] indices.
+///
+/// [tagged_wp]: https://en.wikipedia.org/wiki/Tagged_pointer
+#[derive(Debug)]
+pub(crate) struct Tagged(NonNull);
+
+impl Clone for Tagged {
+ fn clone(&self) -> Self {
+ *self
+ }
+}
+
+impl Copy for Tagged {}
+
+impl Tagged {
+ /// Creates a new, tagged `Tagged` pointer from an integer.
+ ///
+ /// # Requirements
+ ///
+ /// - `T` must have an alignment of at least 2.
+ /// - `tag` must fit inside `usize::BITS - 1` bits
+ pub(crate) const fn from_tag(tag: usize) -> Self {
+ debug_assert!(std::mem::align_of::() >= 2);
+ let addr = (tag << 1) | 1;
+ // SAFETY: `addr` is never zero, since we always set its LSB to 1
+ unsafe { Self(NonNull::new_unchecked(sptr::invalid_mut(addr))) }
+ }
+
+ /// Creates a new `Tagged` pointer from a raw pointer.
+ ///
+ /// # Requirements
+ ///
+ /// - `T` must have an alignment of at least 2.
+ ///
+ /// # Safety
+ ///
+ /// - `T` must be non null.
+ pub(crate) const unsafe fn from_ptr(ptr: *mut T) -> Self {
+ debug_assert!(std::mem::align_of::() >= 2);
+ // SAFETY: the caller must ensure the invariants hold.
+ unsafe { Self(NonNull::new_unchecked(ptr)) }
+ }
+
+ /// Creates a new `Tagged` pointer from a `NonNull` pointer.
+ ///
+ /// # Requirements
+ ///
+ /// - `T` must have an alignment of at least 2.
+ pub(crate) const fn from_non_null(ptr: NonNull) -> Self {
+ debug_assert!(std::mem::align_of::() >= 2);
+ Self(ptr)
+ }
+
+ /// Unwraps the `Tagged` pointer.
+ pub(crate) fn unwrap(self) -> UnwrappedTagged {
+ let addr = self.0.as_ptr().addr();
+ if addr & 1 == 0 {
+ UnwrappedTagged::Ptr(self.0)
+ } else {
+ UnwrappedTagged::Tag(addr >> 1)
+ }
+ }
+
+ /// Gets the address of the inner pointer.
+ #[allow(unused)]
+ pub(crate) fn addr(self) -> usize {
+ self.0.as_ptr().addr()
+ }
+
+ /// Returns `true` if `self ` is a tagged pointer.
+ #[allow(unused)]
+ pub(crate) fn is_tagged(self) -> bool {
+ self.0.as_ptr().addr() & 1 > 0
+ }
+}
+
+/// The unwrapped value of a [`Tagged`] pointer.
+#[derive(Debug, Clone, Copy)]
+pub(crate) enum UnwrappedTagged {
+ Ptr(NonNull),
+ Tag(usize),
+}
diff --git a/core/string/src/tests.rs b/core/string/src/tests.rs
new file mode 100644
index 00000000000..2022994b483
--- /dev/null
+++ b/core/string/src/tests.rs
@@ -0,0 +1,166 @@
+#![allow(clippy::redundant_clone)]
+
+use std::hash::{BuildHasher, BuildHasherDefault, Hash};
+
+use crate::{JsStr, JsString, StaticJsStrings};
+
+use boa_macros::utf16;
+use rustc_hash::FxHasher;
+
+fn hash_value(value: &T) -> u64 {
+ BuildHasherDefault::::default().hash_one(value)
+}
+
+#[test]
+fn empty() {
+ let s = StaticJsStrings::EMPTY_STRING;
+ assert_eq!(&s, utf16!(""));
+}
+
+#[test]
+fn refcount() {
+ let x = JsString::from("Hello world");
+ assert_eq!(x.refcount(), Some(1));
+
+ {
+ let y = x.clone();
+ assert_eq!(x.refcount(), Some(2));
+ assert_eq!(y.refcount(), Some(2));
+
+ {
+ let z = y.clone();
+ assert_eq!(x.refcount(), Some(3));
+ assert_eq!(y.refcount(), Some(3));
+ assert_eq!(z.refcount(), Some(3));
+ }
+
+ assert_eq!(x.refcount(), Some(2));
+ assert_eq!(y.refcount(), Some(2));
+ }
+
+ assert_eq!(x.refcount(), Some(1));
+}
+
+#[test]
+fn static_refcount() {
+ let x = StaticJsStrings::EMPTY_STRING;
+ assert_eq!(x.refcount(), None);
+
+ {
+ let y = x.clone();
+ assert_eq!(x.refcount(), None);
+ assert_eq!(y.refcount(), None);
+ };
+
+ assert_eq!(x.refcount(), None);
+}
+
+#[test]
+fn ptr_eq() {
+ let x = JsString::from("Hello");
+ let y = x.clone();
+
+ assert!(!x.ptr.is_tagged());
+
+ assert_eq!(x.ptr.addr(), y.ptr.addr());
+
+ let z = JsString::from("Hello");
+ assert_ne!(x.ptr.addr(), z.ptr.addr());
+ assert_ne!(y.ptr.addr(), z.ptr.addr());
+}
+
+#[test]
+fn static_ptr_eq() {
+ let x = StaticJsStrings::EMPTY_STRING;
+ let y = x.clone();
+
+ assert!(x.ptr.is_tagged());
+
+ assert_eq!(x.ptr.addr(), y.ptr.addr());
+
+ let z = StaticJsStrings::EMPTY_STRING;
+ assert_eq!(x.ptr.addr(), z.ptr.addr());
+ assert_eq!(y.ptr.addr(), z.ptr.addr());
+}
+
+#[test]
+fn as_str() {
+ const HELLO: &[u16] = utf16!("Hello");
+ let x = JsString::from(HELLO);
+
+ assert_eq!(&x, HELLO);
+}
+
+#[test]
+fn hash() {
+ const HELLOWORLD: JsStr<'_> = JsStr::latin1("Hello World!".as_bytes());
+ let x = JsString::from(HELLOWORLD);
+
+ assert_eq!(x.as_str(), HELLOWORLD);
+
+ assert!(HELLOWORLD.is_latin1());
+ assert!(x.as_str().is_latin1());
+
+ let s_hash = hash_value(&HELLOWORLD);
+ let x_hash = hash_value(&x);
+
+ assert_eq!(s_hash, x_hash);
+}
+
+#[test]
+fn concat() {
+ const Y: &[u16] = utf16!(", ");
+ const W: &[u16] = utf16!("!");
+
+ let x = JsString::from("hello");
+ let z = JsString::from("world");
+
+ let xy = JsString::concat(x.as_str(), JsString::from(Y).as_str());
+ assert_eq!(&xy, utf16!("hello, "));
+ assert_eq!(xy.refcount(), Some(1));
+
+ let xyz = JsString::concat(xy.as_str(), z.as_str());
+ assert_eq!(&xyz, utf16!("hello, world"));
+ assert_eq!(xyz.refcount(), Some(1));
+
+ let xyzw = JsString::concat(xyz.as_str(), JsString::from(W).as_str());
+ assert_eq!(&xyzw, utf16!("hello, world!"));
+ assert_eq!(xyzw.refcount(), Some(1));
+}
+
+#[test]
+fn trim_start_non_ascii_to_ascii() {
+ let s = "\u{2029}abc";
+ let x = JsString::from(s);
+
+ let y = JsString::from(x.trim_start());
+
+ assert_eq!(&y, s.trim_start());
+}
+
+#[test]
+fn conversion_to_known_static_js_string() {
+ const JS_STR_U8: &JsStr<'_> = &JsStr::latin1("length".as_bytes());
+ const JS_STR_U16: &JsStr<'_> = &JsStr::utf16(utf16!("length"));
+
+ assert!(JS_STR_U8.is_latin1());
+ assert!(!JS_STR_U16.is_latin1());
+
+ assert_eq!(JS_STR_U8, JS_STR_U8);
+ assert_eq!(JS_STR_U16, JS_STR_U16);
+
+ assert_eq!(JS_STR_U8, JS_STR_U16);
+ assert_eq!(JS_STR_U16, JS_STR_U8);
+
+ assert_eq!(hash_value(JS_STR_U8), hash_value(JS_STR_U16));
+
+ let string = StaticJsStrings::get_string(JS_STR_U8);
+
+ assert!(string.is_some());
+ assert!(string.unwrap().as_str().is_latin1());
+
+ let string = StaticJsStrings::get_string(JS_STR_U16);
+
+ assert!(string.is_some());
+ assert!(string.unwrap().as_str().is_latin1());
+}