pyo3/impl_/
pyclass_init.rs1use crate::ffi_ptr_ext::FfiPtrExt;
3use crate::internal::get_slot::TP_ALLOC;
4use crate::types::PyType;
5use crate::{ffi, Borrowed, PyErr, PyResult, Python};
6use crate::{ffi::PyTypeObject, sealed::Sealed, type_object::PyTypeInfo};
7use std::marker::PhantomData;
8
9pub trait PyObjectInit<T>: Sized + Sealed {
14 unsafe fn into_new_object(
17 self,
18 py: Python<'_>,
19 subtype: *mut PyTypeObject,
20 ) -> PyResult<*mut ffi::PyObject>;
21
22 #[doc(hidden)]
23 fn can_be_subclassed(&self) -> bool;
24}
25
26pub struct PyNativeTypeInitializer<T: PyTypeInfo>(pub PhantomData<T>);
28
29impl<T: PyTypeInfo> PyObjectInit<T> for PyNativeTypeInitializer<T> {
30 unsafe fn into_new_object(
31 self,
32 py: Python<'_>,
33 subtype: *mut PyTypeObject,
34 ) -> PyResult<*mut ffi::PyObject> {
35 unsafe fn inner(
36 py: Python<'_>,
37 type_object: *mut PyTypeObject,
38 subtype: *mut PyTypeObject,
39 ) -> PyResult<*mut ffi::PyObject> {
40 let is_base_object = type_object == std::ptr::addr_of_mut!(ffi::PyBaseObject_Type);
42 let subtype_borrowed: Borrowed<'_, '_, PyType> = unsafe {
43 subtype
44 .cast::<ffi::PyObject>()
45 .assume_borrowed_unchecked(py)
46 .downcast_unchecked()
47 };
48
49 if is_base_object {
50 let alloc = subtype_borrowed
51 .get_slot(TP_ALLOC)
52 .unwrap_or(ffi::PyType_GenericAlloc);
53
54 let obj = unsafe { alloc(subtype, 0) };
55 return if obj.is_null() {
56 Err(PyErr::fetch(py))
57 } else {
58 Ok(obj)
59 };
60 }
61
62 #[cfg(Py_LIMITED_API)]
63 unreachable!("subclassing native types is not possible with the `abi3` feature");
64
65 #[cfg(not(Py_LIMITED_API))]
66 {
67 match unsafe { (*type_object).tp_new } {
68 Some(newfunc) => {
70 let obj =
71 unsafe { newfunc(subtype, std::ptr::null_mut(), std::ptr::null_mut()) };
72 if obj.is_null() {
73 Err(PyErr::fetch(py))
74 } else {
75 Ok(obj)
76 }
77 }
78 None => Err(crate::exceptions::PyTypeError::new_err(
79 "base type without tp_new",
80 )),
81 }
82 }
83 }
84 let type_object = T::type_object_raw(py);
85 unsafe { inner(py, type_object, subtype) }
86 }
87
88 #[inline]
89 fn can_be_subclassed(&self) -> bool {
90 true
91 }
92}