/* 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 https://mozilla.org/MPL/2.0/. */ use std::borrow::Cow; use std::default::Default; use std::ptr; use std::ptr::NonNull; use std::vec::Vec; use dom_struct::dom_struct; use euclid::default::{Rect, Size2D}; use ipc_channel::ipc::IpcSharedMemory; use js::jsapi::{Heap, JSObject}; use js::rust::{HandleObject, Runtime}; use js::typedarray::{CreateWith, Uint8ClampedArray}; use crate::dom::bindings::codegen::Bindings::ImageDataBinding::ImageDataMethods; use crate::dom::bindings::error::{Error, Fallible}; use crate::dom::bindings::reflector::{reflect_dom_object_with_proto, Reflector}; use crate::dom::bindings::root::DomRoot; use crate::dom::globalscope::GlobalScope; use crate::script_runtime::JSContext; #[dom_struct] pub struct ImageData { reflector_: Reflector, width: u32, height: u32, #[ignore_malloc_size_of = "mozjs"] data: Heap<*mut JSObject>, } impl ImageData { #[allow(unsafe_code)] pub fn new( global: &GlobalScope, width: u32, height: u32, mut data: Option>, ) -> Fallible> { let len = width * height * 4; unsafe { let cx = GlobalScope::get_cx(); rooted!(in (*cx) let mut js_object = ptr::null_mut::()); if let Some(ref mut d) = data { d.resize(len as usize, 0); let data = CreateWith::Slice(&d[..]); Uint8ClampedArray::create(*cx, data, js_object.handle_mut()).unwrap(); Self::new_with_jsobject(global, None, width, Some(height), js_object.get()) } else { Self::new_without_jsobject(global, None, width, height) } } } #[allow(unsafe_code)] unsafe fn new_with_jsobject( global: &GlobalScope, proto: Option, width: u32, opt_height: Option, jsobject: *mut JSObject, ) -> Fallible> { // checking jsobject type let cx = GlobalScope::get_cx(); typedarray!(in(*cx) let array_res: Uint8ClampedArray = jsobject); let array = array_res.map_err(|_| { Error::Type("Argument to Image data is not an Uint8ClampedArray".to_owned()) })?; let byte_len = array.as_slice().len() as u32; if byte_len == 0 || byte_len % 4 != 0 { return Err(Error::InvalidState); } let len = byte_len / 4; if width == 0 || len % width != 0 { return Err(Error::IndexSize); } let height = len / width; if opt_height.map_or(false, |x| height != x) { return Err(Error::IndexSize); } let imagedata = Box::new(ImageData { reflector_: Reflector::new(), width: width, height: height, data: Heap::default(), }); (*imagedata).data.set(jsobject); Ok(reflect_dom_object_with_proto(imagedata, global, proto)) } #[allow(unsafe_code)] unsafe fn new_without_jsobject( global: &GlobalScope, proto: Option, width: u32, height: u32, ) -> Fallible> { if width == 0 || height == 0 { return Err(Error::IndexSize); } let imagedata = Box::new(ImageData { reflector_: Reflector::new(), width: width, height: height, data: Heap::default(), }); let len = width * height * 4; let cx = GlobalScope::get_cx(); rooted!(in (*cx) let mut array = ptr::null_mut::()); Uint8ClampedArray::create(*cx, CreateWith::Length(len as usize), array.handle_mut()) .unwrap(); (*imagedata).data.set(array.get()); Ok(reflect_dom_object_with_proto(imagedata, global, proto)) } // https://html.spec.whatwg.org/multipage/#pixel-manipulation:dom-imagedata-3 #[allow(unsafe_code, non_snake_case)] pub fn Constructor( global: &GlobalScope, proto: Option, width: u32, height: u32, ) -> Fallible> { unsafe { Self::new_without_jsobject(global, proto, width, height) } } // https://html.spec.whatwg.org/multipage/#pixel-manipulation:dom-imagedata-4 #[allow(unsafe_code, unused_variables, non_snake_case)] pub unsafe fn Constructor_( cx: JSContext, global: &GlobalScope, proto: Option, jsobject: *mut JSObject, width: u32, opt_height: Option, ) -> Fallible> { Self::new_with_jsobject(global, proto, width, opt_height, jsobject) } /// Nothing must change the array on the JS side while the slice is live. #[allow(unsafe_code)] pub unsafe fn as_slice(&self) -> &[u8] { assert!(!self.data.get().is_null()); let cx = Runtime::get(); assert!(!cx.is_null()); typedarray!(in(cx) let array: Uint8ClampedArray = self.data.get()); let array = array.as_ref().unwrap(); // NOTE(nox): This is just as unsafe as `as_slice` itself even though we // are extending the lifetime of the slice, because the data in // this ImageData instance will never change. The method is thus unsafe // because the array may be manipulated from JS while the reference // is live. let ptr = array.as_slice() as *const _; &*ptr } #[allow(unsafe_code)] pub fn to_shared_memory(&self) -> IpcSharedMemory { IpcSharedMemory::from_bytes(unsafe { self.as_slice() }) } #[allow(unsafe_code)] pub unsafe fn get_rect(&self, rect: Rect) -> Cow<[u8]> { pixels::rgba8_get_rect(self.as_slice(), self.get_size().to_u64(), rect) } pub fn get_size(&self) -> Size2D { Size2D::new(self.Width(), self.Height()) } } impl ImageDataMethods for ImageData { // https://html.spec.whatwg.org/multipage/#dom-imagedata-width fn Width(&self) -> u32 { self.width } // https://html.spec.whatwg.org/multipage/#dom-imagedata-height fn Height(&self) -> u32 { self.height } // https://html.spec.whatwg.org/multipage/#dom-imagedata-data fn Data(&self, _: JSContext) -> NonNull { NonNull::new(self.data.get()).expect("got a null pointer") } }