use log::error;
use std::marker::PhantomData;
use std::ptr::NonNull;
use crate::JavaVM;
use crate::objects::{JPrimitiveArray, ReleaseMode, TypeArray};
use crate::sys::jboolean;
use crate::{env::Env, errors::*, sys};
#[cfg(doc)]
use crate::objects::JByteArray;
#[derive(Debug)]
pub struct AutoElementsCritical<'array_local, T: TypeArray, TArrayRef>
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
array: TArrayRef,
len: usize,
ptr: NonNull<T>,
mode: ReleaseMode,
is_copy: bool,
_lifetime: PhantomData<&'array_local ()>,
}
impl<'array_local, T: TypeArray, TArrayRef> AutoElementsCritical<'array_local, T, TArrayRef>
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
pub(crate) unsafe fn new_with_len(
env: &Env<'_>,
array: TArrayRef,
len: usize,
mode: ReleaseMode,
) -> Result<Self> {
let mut is_copy: jboolean = true;
let ptr = unsafe {
jni_call_with_catch_and_null_check!(
catch |env| {
crate::exceptions::JOutOfMemoryError =>
Err(Error::JniCall(JniError::NoMemory)),
else => Err(Error::NullPtr("Unexpected Exception")),
},
env,
v1_2,
GetPrimitiveArrayCritical,
array.as_ref().as_raw(),
&mut is_copy
)? as *mut T
};
Ok(AutoElementsCritical {
array,
len,
ptr: NonNull::new(ptr).ok_or(Error::NullPtr("Non-null ptr expected"))?,
mode,
is_copy: is_copy == sys::JNI_TRUE,
_lifetime: PhantomData,
})
}
pub(crate) fn new(env: &Env<'_>, array: TArrayRef, mode: ReleaseMode) -> Result<Self> {
let len = array.as_ref().len(env)?;
unsafe { Self::new_with_len(env, array, len, mode) }
}
pub const fn as_ptr(&self) -> *mut T {
self.ptr.as_ptr()
}
unsafe fn release_primitive_array_critical(&mut self, mode: i32) -> Result<()> {
JavaVM::singleton()?.with_top_local_frame(|env| unsafe {
ex_safe_jni_call_no_post_check_ex!(
env,
v1_2,
ReleasePrimitiveArrayCritical,
self.array.as_ref().as_raw(),
self.ptr.as_ptr().cast(),
mode
);
Ok(())
})
}
pub fn discard(mut self) {
self.mode = ReleaseMode::NoCopyBack;
}
pub fn is_copy(&self) -> bool {
self.is_copy
}
pub fn len(&self) -> usize {
self.len
}
pub fn is_empty(&self) -> bool {
self.len == 0
}
}
impl<'array_local, T: TypeArray, TArrayRef> AsRef<AutoElementsCritical<'array_local, T, TArrayRef>>
for AutoElementsCritical<'array_local, T, TArrayRef>
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
fn as_ref(&self) -> &AutoElementsCritical<'array_local, T, TArrayRef> {
self
}
}
impl<'array_local, T: TypeArray, TArrayRef> Drop
for AutoElementsCritical<'array_local, T, TArrayRef>
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
fn drop(&mut self) {
let res = unsafe { self.release_primitive_array_critical(self.mode as i32) };
match res {
Ok(()) => {}
Err(e) => error!("error releasing primitive array: {:#?}", e),
}
}
}
impl<'array_local, T: TypeArray, TArrayRef> From<&AutoElementsCritical<'array_local, T, TArrayRef>>
for *mut T
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
fn from(other: &AutoElementsCritical<'array_local, T, TArrayRef>) -> *mut T {
other.as_ptr()
}
}
impl<'array_local, T: TypeArray, TArrayRef> std::ops::Deref
for AutoElementsCritical<'array_local, T, TArrayRef>
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
type Target = [T];
fn deref(&self) -> &Self::Target {
unsafe { std::slice::from_raw_parts(self.ptr.as_ptr(), self.len) }
}
}
impl<'array_local, T: TypeArray, TArrayRef> std::ops::DerefMut
for AutoElementsCritical<'array_local, T, TArrayRef>
where
TArrayRef: AsRef<JPrimitiveArray<'array_local, T>>,
{
fn deref_mut(&mut self) -> &mut Self::Target {
unsafe { std::slice::from_raw_parts_mut(self.ptr.as_mut(), self.len) }
}
}