wasm_bindgen/cache/
intern.rs

1use cfg_if::cfg_if;
2
3cfg_if! {
4    if #[cfg(feature = "enable-interning")] {
5        use std::thread_local;
6        use std::string::String;
7        use std::borrow::ToOwned;
8        use std::cell::RefCell;
9        use std::collections::HashMap;
10        use crate::JsValue;
11
12        struct Cache {
13            entries: RefCell<HashMap<String, JsValue>>,
14        }
15
16        thread_local! {
17            static CACHE: Cache = Cache {
18                entries: RefCell::new(HashMap::new()),
19            };
20        }
21
22        /// This returns the raw index of the cached JsValue, so you must take care
23        /// so that you don't use it after it is freed.
24        pub(crate) fn unsafe_get_str(s: &str) -> Option<u32> {
25            CACHE.with(|cache| {
26                let cache = cache.entries.borrow();
27
28                cache.get(s).map(|x| x.idx)
29            })
30        }
31
32        fn intern_str(key: &str) {
33            CACHE.with(|cache| {
34                let mut cache = cache.entries.borrow_mut();
35
36                // Can't use `entry` because `entry` requires a `String`
37                if !cache.contains_key(key) {
38                    cache.insert(key.to_owned(), JsValue::from(key));
39                }
40            })
41        }
42
43        fn unintern_str(key: &str) {
44            CACHE.with(|cache| {
45                let mut cache = cache.entries.borrow_mut();
46
47                cache.remove(key);
48            })
49        }
50    }
51}
52
53/// Interns Rust strings so that it's much faster to send them to JS.
54///
55/// Sending strings from Rust to JS is slow, because it has to do a full `O(n)`
56/// copy and *also* encode from UTF-8 to UTF-16. This must be done every single
57/// time a string is sent to JS.
58///
59/// If you are sending the same string multiple times, you can call this `intern`
60/// function, which simply returns its argument unchanged:
61///
62/// ```rust
63/// # use wasm_bindgen::intern;
64/// intern("foo") // returns "foo"
65/// # ;
66/// ```
67///
68/// However, if you enable the `"enable-interning"` feature for wasm-bindgen,
69/// then it will add the string into an internal cache.
70///
71/// When you send that cached string to JS, it will look it up in the cache,
72/// which completely avoids the `O(n)` copy and encoding. This has a significant
73/// speed boost (as high as 783%)!
74///
75/// However, there is a small cost to this caching, so you shouldn't cache every
76/// string. Only cache strings which have a high likelihood of being sent
77/// to JS multiple times.
78///
79/// Also, keep in mind that this function is a *performance hint*: it's not
80/// *guaranteed* that the string will be cached, and the caching strategy
81/// might change at any time, so don't rely upon it.
82#[inline]
83pub fn intern(s: &str) -> &str {
84    #[cfg(feature = "enable-interning")]
85    intern_str(s);
86
87    s
88}
89
90/// Removes a Rust string from the intern cache.
91///
92/// This does the opposite of the [`intern`](fn.intern.html) function.
93///
94/// If the [`intern`](fn.intern.html) function is called again then it will re-intern the string.
95#[allow(unused_variables)]
96#[inline]
97pub fn unintern(s: &str) {
98    #[cfg(feature = "enable-interning")]
99    unintern_str(s);
100}