summaryrefslogtreecommitdiffstats
path: root/js/rust/src/error.rs
diff options
context:
space:
mode:
Diffstat (limited to 'js/rust/src/error.rs')
-rw-r--r--js/rust/src/error.rs78
1 files changed, 78 insertions, 0 deletions
diff --git a/js/rust/src/error.rs b/js/rust/src/error.rs
new file mode 100644
index 0000000000..34a5fa828a
--- /dev/null
+++ b/js/rust/src/error.rs
@@ -0,0 +1,78 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+//! Functions to throw JavaScript exceptions from Rust.
+
+#![deny(missing_docs)]
+
+use jsapi::root::*;
+use libc;
+use std::ffi::CString;
+use std::{mem, os, ptr};
+
+/// Format string used to throw javascript errors.
+static ERROR_FORMAT_STRING_STRING: [libc::c_char; 4] = [
+ '{' as libc::c_char,
+ '0' as libc::c_char,
+ '}' as libc::c_char,
+ 0 as libc::c_char,
+];
+
+/// Format string struct used to throw `TypeError`s.
+static mut TYPE_ERROR_FORMAT_STRING: JSErrorFormatString = JSErrorFormatString {
+ name: b"RUSTMSG_TYPE_ERROR\0" as *const _ as *const libc::c_char,
+ format: &ERROR_FORMAT_STRING_STRING as *const libc::c_char,
+ argCount: 1,
+ exnType: JSExnType::JSEXN_TYPEERR as i16,
+};
+
+/// Format string struct used to throw `RangeError`s.
+static mut RANGE_ERROR_FORMAT_STRING: JSErrorFormatString = JSErrorFormatString {
+ name: b"RUSTMSG_RANGE_ERROR\0" as *const _ as *const libc::c_char,
+ format: &ERROR_FORMAT_STRING_STRING as *const libc::c_char,
+ argCount: 1,
+ exnType: JSExnType::JSEXN_RANGEERR as i16,
+};
+
+/// Callback used to throw javascript errors.
+/// See throw_js_error for info about error_number.
+unsafe extern "C" fn get_error_message(
+ _user_ref: *mut os::raw::c_void,
+ error_number: libc::c_uint,
+) -> *const JSErrorFormatString {
+ let num: JSExnType = mem::transmute(error_number);
+ match num {
+ JSExnType::JSEXN_TYPEERR => &TYPE_ERROR_FORMAT_STRING as *const JSErrorFormatString,
+ JSExnType::JSEXN_RANGEERR => &RANGE_ERROR_FORMAT_STRING as *const JSErrorFormatString,
+ _ => panic!(
+ "Bad js error number given to get_error_message: {}",
+ error_number
+ ),
+ }
+}
+
+/// Helper fn to throw a javascript error with the given message and number.
+/// Reuse the jsapi error codes to distinguish the error_number
+/// passed back to the get_error_message callback.
+/// c_uint is u32, so this cast is safe, as is casting to/from i32 from there.
+unsafe fn throw_js_error(cx: *mut JSContext, error: &str, error_number: u32) {
+ let error = CString::new(error).unwrap();
+ JS_ReportErrorNumberUTF8(
+ cx,
+ Some(get_error_message),
+ ptr::null_mut(),
+ error_number,
+ error.as_ptr(),
+ );
+}
+
+/// Throw a `TypeError` with the given message.
+pub unsafe fn throw_type_error(cx: *mut JSContext, error: &str) {
+ throw_js_error(cx, error, JSExnType::JSEXN_TYPEERR as u32);
+}
+
+/// Throw a `RangeError` with the given message.
+pub unsafe fn throw_range_error(cx: *mut JSContext, error: &str) {
+ throw_js_error(cx, error, JSExnType::JSEXN_RANGEERR as u32);
+}