mirror of
https://github.com/fluencelabs/wasm-bindgen
synced 2025-04-03 19:01:06 +00:00
Merge pull request #958 from alexcrichton/no-global-cache
js_sys: Use a thread local to cache `global()`
This commit is contained in:
commit
41d3a08028
@ -23,7 +23,6 @@ extern crate wasm_bindgen;
|
|||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::mem;
|
use std::mem;
|
||||||
|
|
||||||
use std::sync::atomic::{AtomicUsize, Ordering::SeqCst, ATOMIC_USIZE_INIT};
|
|
||||||
use wasm_bindgen::prelude::*;
|
use wasm_bindgen::prelude::*;
|
||||||
use wasm_bindgen::JsCast;
|
use wasm_bindgen::JsCast;
|
||||||
|
|
||||||
@ -4407,19 +4406,7 @@ extern "C" {
|
|||||||
/// This allows access to the global properties and global names by accessing
|
/// This allows access to the global properties and global names by accessing
|
||||||
/// the `Object` returned.
|
/// the `Object` returned.
|
||||||
pub fn global() -> Object {
|
pub fn global() -> Object {
|
||||||
// Cached `Box<JsValue>`, if we've already executed this.
|
thread_local!(static GLOBAL: Object = {
|
||||||
//
|
|
||||||
// 0 = not calculated
|
|
||||||
// n = Some(n) == Some(Box<JsValue>)
|
|
||||||
static GLOBAL: AtomicUsize = ATOMIC_USIZE_INIT;
|
|
||||||
|
|
||||||
match GLOBAL.load(SeqCst) {
|
|
||||||
0 => {}
|
|
||||||
n => return unsafe { (*(n as *const JsValue)).clone().unchecked_into() },
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ok we don't have a cached value, let's load one!
|
|
||||||
//
|
|
||||||
// According to StackOverflow you can access the global object via:
|
// According to StackOverflow you can access the global object via:
|
||||||
//
|
//
|
||||||
// const global = Function('return this')();
|
// const global = Function('return this')();
|
||||||
@ -4438,17 +4425,11 @@ pub fn global() -> Object {
|
|||||||
// Note that we avoid `unwrap()` on `call0` to avoid code size bloat, we
|
// Note that we avoid `unwrap()` on `call0` to avoid code size bloat, we
|
||||||
// just handle the `Err` case as returning a different object.
|
// just handle the `Err` case as returning a different object.
|
||||||
debug_assert!(this.is_some());
|
debug_assert!(this.is_some());
|
||||||
let this = match this {
|
match this {
|
||||||
Some(this) => this,
|
Some(this) => this.unchecked_into(),
|
||||||
None => return JsValue::undefined().unchecked_into(),
|
None => JsValue::undefined().unchecked_into(),
|
||||||
};
|
}
|
||||||
|
});
|
||||||
|
|
||||||
let ptr: *mut JsValue = Box::into_raw(Box::new(this.clone()));
|
GLOBAL.with(|g| g.clone())
|
||||||
match GLOBAL.compare_exchange(0, ptr as usize, SeqCst, SeqCst) {
|
|
||||||
// We stored out value, relinquishing ownership of `ptr`
|
|
||||||
Ok(_) => {}
|
|
||||||
// Another thread one, drop our value
|
|
||||||
Err(_) => unsafe { drop(Box::from_raw(ptr)) },
|
|
||||||
}
|
|
||||||
this.unchecked_into()
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user