summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDivy Srivastava <dj.srivastava23@gmail.com>2022-10-09 03:08:38 -0700
committerGitHub <noreply@github.com>2022-10-09 15:38:38 +0530
commita622c5df27e908bff152ce7374c47dabfdba0bba (patch)
tree0c836839a6d359db2afa8baf640860a31decf714
parentcc3e2b9b1a7ccd6c671bc0b3813cf7e2b0d88b5c (diff)
perf(ext/crypto): optimize `getRandomValues` (#16212)
-rw-r--r--cli/bench/getrandom.js21
-rw-r--r--ext/crypto/00_crypto.js5
-rw-r--r--ext/crypto/lib.rs12
3 files changed, 32 insertions, 6 deletions
diff --git a/cli/bench/getrandom.js b/cli/bench/getrandom.js
new file mode 100644
index 000000000..94df0a802
--- /dev/null
+++ b/cli/bench/getrandom.js
@@ -0,0 +1,21 @@
+// Copyright 2018-2022 the Deno authors. All rights reserved. MIT license.
+let [total, count] = typeof Deno !== "undefined"
+ ? Deno.args
+ : [process.argv[2], process.argv[3]];
+
+total = total ? parseInt(total, 0) : 50;
+count = count ? parseInt(count, 10) : 100000;
+
+async function bench(fun) {
+ const start = Date.now();
+ for (let i = 0; i < count; i++) await fun();
+ const elapsed = Date.now() - start;
+ const rate = Math.floor(count / (elapsed / 1000));
+ console.log(`time ${elapsed} ms rate ${rate}`);
+ if (--total) await bench(fun);
+}
+
+const c = typeof crypto !== "undefined" ? crypto : require("crypto").webcrypto;
+
+const ui8 = new Uint8Array(1024);
+bench(() => c.getRandomValues(ui8));
diff --git a/ext/crypto/00_crypto.js b/ext/crypto/00_crypto.js
index 907308325..7b21c9287 100644
--- a/ext/crypto/00_crypto.js
+++ b/ext/crypto/00_crypto.js
@@ -4652,6 +4652,11 @@
webidl.assertBranded(this, CryptoPrototype);
const prefix = "Failed to execute 'getRandomValues' on 'Crypto'";
webidl.requiredArguments(arguments.length, 1, { prefix });
+ // Fast path for Uint8Array
+ if (ObjectPrototypeIsPrototypeOf(Uint8ArrayPrototype, arrayBufferView)) {
+ ops.op_crypto_get_random_values(arrayBufferView);
+ return arrayBufferView;
+ }
arrayBufferView = webidl.converters.ArrayBufferView(arrayBufferView, {
prefix,
context: "Argument 1",
diff --git a/ext/crypto/lib.rs b/ext/crypto/lib.rs
index d949135e7..f09982920 100644
--- a/ext/crypto/lib.rs
+++ b/ext/crypto/lib.rs
@@ -132,24 +132,24 @@ pub fn op_crypto_base64url_encode(data: ZeroCopyBuf) -> String {
data
}
-#[op]
+#[op(fast)]
pub fn op_crypto_get_random_values(
state: &mut OpState,
- mut zero_copy: ZeroCopyBuf,
+ out: &mut [u8],
) -> Result<(), AnyError> {
- if zero_copy.len() > 65536 {
+ if out.len() > 65536 {
return Err(
- deno_web::DomExceptionQuotaExceededError::new(&format!("The ArrayBufferView's byte length ({}) exceeds the number of bytes of entropy available via this API (65536)", zero_copy.len()))
+ deno_web::DomExceptionQuotaExceededError::new(&format!("The ArrayBufferView's byte length ({}) exceeds the number of bytes of entropy available via this API (65536)", out.len()))
.into(),
);
}
let maybe_seeded_rng = state.try_borrow_mut::<StdRng>();
if let Some(seeded_rng) = maybe_seeded_rng {
- seeded_rng.fill(&mut *zero_copy);
+ seeded_rng.fill(out);
} else {
let mut rng = thread_rng();
- rng.fill(&mut *zero_copy);
+ rng.fill(out);
}
Ok(())