|
1 | 1 | //! Implementation for WASM based on Web and Node.js
|
| 2 | +
|
| 3 | +/// Proc-macro is not hygienic. |
| 4 | +/// See <https://github.com/rustwasm/wasm-bindgen/pull/4315>. |
| 5 | +extern crate std; |
| 6 | + |
2 | 7 | use crate::Error;
|
3 | 8 | use core::mem::MaybeUninit;
|
| 9 | +use std::thread_local; |
4 | 10 |
|
5 | 11 | pub use crate::util::{inner_u32, inner_u64};
|
6 | 12 |
|
7 | 13 | #[cfg(not(all(target_arch = "wasm32", any(target_os = "unknown", target_os = "none"))))]
|
8 | 14 | compile_error!("`wasm_js` backend can be enabled only for OS-less WASM targets!");
|
9 | 15 |
|
10 |
| -use js_sys::{global, Uint8Array}; |
11 |
| -use wasm_bindgen::{prelude::wasm_bindgen, JsCast, JsValue}; |
| 16 | +use js_sys::Uint8Array; |
| 17 | +use wasm_bindgen::{prelude::wasm_bindgen, JsValue}; |
12 | 18 |
|
13 | 19 | // Size of our temporary Uint8Array buffer used with WebCrypto methods
|
14 | 20 | // Maximum is 65536 bytes see https://developer.mozilla.org/en-US/docs/Web/API/Crypto/getRandomValues
|
15 | 21 | const CRYPTO_BUFFER_SIZE: u16 = 256;
|
16 | 22 |
|
17 | 23 | pub fn fill_inner(dest: &mut [MaybeUninit<u8>]) -> Result<(), Error> {
|
18 |
| - let global: Global = global().unchecked_into(); |
19 |
| - let crypto = global.crypto(); |
20 |
| - |
21 |
| - if !crypto.is_object() { |
22 |
| - return Err(Error::WEB_CRYPTO); |
23 |
| - } |
24 |
| - |
25 |
| - // getRandomValues does not work with all types of WASM memory, |
26 |
| - // so we initially write to browser memory to avoid exceptions. |
27 |
| - let buf = Uint8Array::new_with_length(CRYPTO_BUFFER_SIZE.into()); |
28 |
| - for chunk in dest.chunks_mut(CRYPTO_BUFFER_SIZE.into()) { |
29 |
| - let chunk_len: u32 = chunk |
30 |
| - .len() |
31 |
| - .try_into() |
32 |
| - .expect("chunk length is bounded by CRYPTO_BUFFER_SIZE"); |
33 |
| - // The chunk can be smaller than buf's length, so we call to |
34 |
| - // JS to create a smaller view of buf without allocation. |
35 |
| - let sub_buf = buf.subarray(0, chunk_len); |
36 |
| - |
37 |
| - if crypto.get_random_values(&sub_buf).is_err() { |
38 |
| - return Err(Error::WEB_GET_RANDOM_VALUES); |
| 24 | + CRYPTO.with(|crypto| { |
| 25 | + let crypto = crypto.as_ref().ok_or(Error::WEB_CRYPTO)?; |
| 26 | + |
| 27 | + // getRandomValues does not work with all types of WASM memory, |
| 28 | + // so we initially write to browser memory to avoid exceptions. |
| 29 | + let buf = Uint8Array::new_with_length(CRYPTO_BUFFER_SIZE.into()); |
| 30 | + for chunk in dest.chunks_mut(CRYPTO_BUFFER_SIZE.into()) { |
| 31 | + let chunk_len: u32 = chunk |
| 32 | + .len() |
| 33 | + .try_into() |
| 34 | + .expect("chunk length is bounded by CRYPTO_BUFFER_SIZE"); |
| 35 | + // The chunk can be smaller than buf's length, so we call to |
| 36 | + // JS to create a smaller view of buf without allocation. |
| 37 | + let sub_buf = buf.subarray(0, chunk_len); |
| 38 | + |
| 39 | + if crypto.get_random_values(&sub_buf).is_err() { |
| 40 | + return Err(Error::WEB_GET_RANDOM_VALUES); |
| 41 | + } |
| 42 | + |
| 43 | + // SAFETY: `sub_buf`'s length is the same length as `chunk` |
| 44 | + unsafe { sub_buf.raw_copy_to_ptr(chunk.as_mut_ptr().cast::<u8>()) }; |
39 | 45 | }
|
40 |
| - |
41 |
| - // SAFETY: `sub_buf`'s length is the same length as `chunk` |
42 |
| - unsafe { sub_buf.raw_copy_to_ptr(chunk.as_mut_ptr().cast::<u8>()) }; |
43 |
| - } |
44 |
| - Ok(()) |
| 46 | + Ok(()) |
| 47 | + }) |
45 | 48 | }
|
46 | 49 |
|
47 | 50 | #[wasm_bindgen]
|
48 | 51 | extern "C" {
|
49 |
| - // Return type of js_sys::global() |
50 |
| - type Global; |
51 | 52 | // Web Crypto API: Crypto interface (https://www.w3.org/TR/WebCryptoAPI/)
|
52 | 53 | type Crypto;
|
53 |
| - // Getters for the Crypto API |
54 |
| - #[wasm_bindgen(method, getter)] |
55 |
| - fn crypto(this: &Global) -> Crypto; |
| 54 | + // Holds the global `Crypto` object. |
| 55 | + #[wasm_bindgen(thread_local_v2, js_namespace = globalThis, js_name = crypto)] |
| 56 | + static CRYPTO: Option<Crypto>; |
56 | 57 | // Crypto.getRandomValues()
|
57 | 58 | #[wasm_bindgen(method, js_name = getRandomValues, catch)]
|
58 | 59 | fn get_random_values(this: &Crypto, buf: &Uint8Array) -> Result<(), JsValue>;
|
|
0 commit comments