use core::fmt;
use core::hash;
use core::marker::PhantomData;
use core::ptr::NonNull;
use crate::encode::{Encoding, RefEncode};
use crate::rc::{autoreleasepool_leaking, Retained};
use crate::runtime::__nsstring::nsstring_to_str;
use crate::runtime::{AnyObject, NSObjectProtocol};
use crate::Message;
pub unsafe trait ImplementedBy<T: ?Sized + Message> {
#[doc(hidden)]
const __INNER: ();
}
#[doc(alias = "id")]
#[repr(C)]
pub struct ProtocolObject<P: ?Sized> {
inner: AnyObject,
p: PhantomData<P>,
}
unsafe impl<P: ?Sized + Send> Send for ProtocolObject<P> {}
unsafe impl<P: ?Sized + Sync> Sync for ProtocolObject<P> {}
unsafe impl<P: ?Sized> RefEncode for ProtocolObject<P> {
const ENCODING_REF: Encoding = Encoding::Object;
}
unsafe impl<P: ?Sized> Message for ProtocolObject<P> {}
impl<P: ?Sized> ProtocolObject<P> {
#[inline]
pub fn from_ref<T: ?Sized + Message>(obj: &T) -> &Self
where
P: ImplementedBy<T>,
{
let ptr: NonNull<T> = NonNull::from(obj);
let ptr: NonNull<Self> = ptr.cast();
unsafe { ptr.as_ref() }
}
#[inline]
pub fn from_mut<T: ?Sized + Message>(obj: &mut T) -> &mut Self
where
P: ImplementedBy<T>,
{
let ptr: NonNull<T> = NonNull::from(obj);
let mut ptr: NonNull<Self> = ptr.cast();
unsafe { ptr.as_mut() }
}
#[inline]
pub fn from_id<T>(obj: Retained<T>) -> Retained<Self>
where
P: ImplementedBy<T> + 'static,
T: Message + 'static,
{
Self::from_retained(obj)
}
#[inline]
pub fn from_retained<T>(obj: Retained<T>) -> Retained<Self>
where
P: ImplementedBy<T> + 'static,
T: Message + 'static,
{
unsafe { Retained::cast::<Self>(obj) }
}
}
impl<P: ?Sized + NSObjectProtocol> PartialEq for ProtocolObject<P> {
#[inline]
#[doc(alias = "isEqual:")]
fn eq(&self, other: &Self) -> bool {
self.isEqual(&other.inner)
}
}
impl<P: ?Sized + NSObjectProtocol> Eq for ProtocolObject<P> {}
impl<P: ?Sized + NSObjectProtocol> hash::Hash for ProtocolObject<P> {
#[inline]
fn hash<H: hash::Hasher>(&self, state: &mut H) {
<Self as NSObjectProtocol>::hash(self).hash(state);
}
}
impl<P: ?Sized + NSObjectProtocol> fmt::Debug for ProtocolObject<P> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let description = self.description();
autoreleasepool_leaking(|pool| {
let s = unsafe { nsstring_to_str(&description, pool) };
fmt::Display::fmt(s, f)
})
}
}
impl<P: ?Sized, T> AsRef<ProtocolObject<T>> for ProtocolObject<P>
where
T: ?Sized + ImplementedBy<ProtocolObject<P>>,
{
#[inline]
fn as_ref(&self) -> &ProtocolObject<T> {
ProtocolObject::from_ref(self)
}
}
impl<P: ?Sized, T> AsMut<ProtocolObject<T>> for ProtocolObject<P>
where
T: ?Sized + ImplementedBy<ProtocolObject<P>>,
{
#[inline]
fn as_mut(&mut self) -> &mut ProtocolObject<T> {
ProtocolObject::from_mut(self)
}
}
#[cfg(test)]
#[allow(clippy::missing_safety_doc)]
#[allow(dead_code)]
mod tests {
use alloc::format;
use core::mem::ManuallyDrop;
use static_assertions::{assert_impl_all, assert_not_impl_any};
use super::*;
use crate::mutability::Mutable;
use crate::runtime::NSObject;
use crate::{
declare_class, extern_methods, extern_protocol, ClassType, DeclaredClass, ProtocolType,
};
extern_protocol!(
unsafe trait Foo {
#[method(foo)]
fn foo_class();
#[method(foo)]
fn foo_instance(&self);
}
unsafe impl ProtocolType for dyn Foo {}
);
extern_protocol!(
unsafe trait Bar: NSObjectProtocol {
#[method(bar)]
fn bar_class();
#[method(bar)]
fn bar_instance(&self);
}
unsafe impl ProtocolType for dyn Bar {}
);
extern_protocol!(
unsafe trait FooBar: Foo + Bar {
#[method(foobar)]
fn foobar_class();
#[method(foobar)]
fn foobar_instance(&self);
}
unsafe impl ProtocolType for dyn FooBar {}
);
extern_protocol!(
unsafe trait FooFooBar: Foo + FooBar {
#[method(foofoobar)]
fn foofoobar_class();
#[method(foofoobar)]
fn foofoobar_instance(&self);
}
unsafe impl ProtocolType for dyn FooFooBar {}
);
declare_class!(
#[derive(Debug, PartialEq, Eq, Hash)]
struct DummyClass;
unsafe impl ClassType for DummyClass {
type Super = NSObject;
type Mutability = Mutable;
const NAME: &'static str = "ProtocolTestsDummyClass";
}
impl DeclaredClass for DummyClass {}
unsafe impl NSObjectProtocol for DummyClass {}
);
unsafe impl Foo for DummyClass {}
unsafe impl Bar for DummyClass {}
unsafe impl FooBar for DummyClass {}
unsafe impl Send for DummyClass {}
unsafe impl Sync for DummyClass {}
extern_methods!(
unsafe impl DummyClass {
#[method_id(new)]
fn new() -> Retained<Self>;
}
);
#[test]
fn impl_traits() {
assert_impl_all!(NSObject: NSObjectProtocol);
assert_impl_all!(ProtocolObject<dyn NSObjectProtocol>: NSObjectProtocol);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol>: Send, Sync);
assert_impl_all!(ProtocolObject<dyn NSObjectProtocol + Send>: NSObjectProtocol, Send);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol + Send>: Sync);
assert_impl_all!(ProtocolObject<dyn NSObjectProtocol + Sync>: NSObjectProtocol, Sync);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol + Sync>: Send);
assert_impl_all!(ProtocolObject<dyn NSObjectProtocol + Send + Sync>: NSObjectProtocol, Send, Sync);
assert_not_impl_any!(ProtocolObject<dyn Foo>: NSObjectProtocol);
assert_impl_all!(ProtocolObject<dyn Bar>: NSObjectProtocol);
assert_impl_all!(ProtocolObject<dyn FooBar>: NSObjectProtocol);
assert_impl_all!(ProtocolObject<dyn FooFooBar>: NSObjectProtocol);
assert_impl_all!(DummyClass: NSObjectProtocol);
assert_not_impl_any!(NSObject: Foo);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol>: Foo);
assert_impl_all!(ProtocolObject<dyn Foo>: Foo);
assert_not_impl_any!(ProtocolObject<dyn Bar>: Foo);
assert_impl_all!(ProtocolObject<dyn FooBar>: Foo);
assert_impl_all!(ProtocolObject<dyn FooFooBar>: Foo);
assert_impl_all!(DummyClass: Foo);
assert_not_impl_any!(NSObject: Bar);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol>: Bar);
assert_not_impl_any!(ProtocolObject<dyn Foo>: Bar);
assert_impl_all!(ProtocolObject<dyn Bar>: Bar);
assert_impl_all!(ProtocolObject<dyn FooBar>: Bar);
assert_impl_all!(ProtocolObject<dyn FooFooBar>: Bar);
assert_impl_all!(DummyClass: Bar);
assert_not_impl_any!(NSObject: FooBar);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol>: FooBar);
assert_not_impl_any!(ProtocolObject<dyn Foo>: FooBar);
assert_not_impl_any!(ProtocolObject<dyn Bar>: FooBar);
assert_impl_all!(ProtocolObject<dyn FooBar>: FooBar);
assert_impl_all!(ProtocolObject<dyn FooFooBar>: FooBar);
assert_impl_all!(DummyClass: FooBar);
assert_not_impl_any!(NSObject: FooFooBar);
assert_not_impl_any!(ProtocolObject<dyn NSObjectProtocol>: FooFooBar);
assert_not_impl_any!(ProtocolObject<dyn Foo>: FooFooBar);
assert_not_impl_any!(ProtocolObject<dyn Bar>: FooFooBar);
assert_not_impl_any!(ProtocolObject<dyn FooBar>: FooFooBar);
assert_impl_all!(ProtocolObject<dyn FooFooBar>: FooFooBar);
assert_not_impl_any!(DummyClass: FooFooBar);
}
#[test]
fn convertible() {
let mut obj = DummyClass::new();
let foobar: &ProtocolObject<dyn FooBar> = ProtocolObject::from_ref(&*obj);
let foobar: &ProtocolObject<dyn FooBar> = ProtocolObject::from_ref(foobar);
let _bar: &ProtocolObject<dyn Bar> = ProtocolObject::from_ref(foobar);
let bar: &ProtocolObject<dyn Bar> = ProtocolObject::from_ref(&*obj);
let bar: &ProtocolObject<dyn Bar> = ProtocolObject::from_ref(bar);
let _foo: &ProtocolObject<dyn Foo> = ProtocolObject::from_ref(foobar);
let foo: &ProtocolObject<dyn Foo> = ProtocolObject::from_ref(&*obj);
let _foo: &ProtocolObject<dyn Foo> = ProtocolObject::from_ref(foo);
let _nsobject: &ProtocolObject<dyn NSObjectProtocol> = ProtocolObject::from_ref(foobar);
let _nsobject: &ProtocolObject<dyn NSObjectProtocol> = ProtocolObject::from_ref(bar);
let nsobject: &ProtocolObject<dyn NSObjectProtocol> = ProtocolObject::from_ref(&*obj);
let _nsobject: &ProtocolObject<dyn NSObjectProtocol> = ProtocolObject::from_ref(nsobject);
let _: &ProtocolObject<dyn NSObjectProtocol + Send> = ProtocolObject::from_ref(&*obj);
let _: &ProtocolObject<dyn NSObjectProtocol + Sync> = ProtocolObject::from_ref(&*obj);
let _: &ProtocolObject<dyn NSObjectProtocol + Send + Sync> =
ProtocolObject::from_ref(&*obj);
let _foobar: &mut ProtocolObject<dyn FooBar> = ProtocolObject::from_mut(&mut *obj);
let _foobar: Retained<ProtocolObject<dyn FooBar>> = ProtocolObject::from_retained(obj);
}
#[test]
fn test_traits() {
use core::hash::Hasher;
use std::collections::hash_map::DefaultHasher;
use std::hash::Hash;
let obj = DummyClass::new();
let obj2 = DummyClass::new();
let foobar: &ProtocolObject<dyn FooBar> = ProtocolObject::from_ref(&*obj);
let foobar2: &ProtocolObject<dyn FooBar> = ProtocolObject::from_ref(&*obj2);
assert_eq!(
format!("{obj:?}"),
format!(
"DummyClass {{ __superclass: {:?}, __ivars: PhantomData<()> }}",
ManuallyDrop::new(foobar)
),
);
assert_eq!(obj == obj2, foobar == foobar2);
let mut hashstate_a = DefaultHasher::new();
let mut hashstate_b = DefaultHasher::new();
obj.hash(&mut hashstate_a);
<_ as Hash>::hash(foobar, &mut hashstate_b);
assert_eq!(hashstate_a.finish(), hashstate_b.finish());
}
extern_protocol!(
#[cfg(debug_assertions)]
unsafe trait CfgTest {}
#[cfg(debug_assertions)]
unsafe impl ProtocolType for dyn CfgTest {}
);
#[test]
#[cfg(debug_assertions)]
fn test_meta() {
if false {
let _protocol = <dyn CfgTest>::protocol();
}
}
}