1use crate::exceptions::PyStopAsyncIteration;
2use crate::gil::LockGIL;
3use crate::impl_::callback::IntoPyCallbackOutput;
4use crate::impl_::panic::PanicTrap;
5use crate::impl_::pycell::{PyClassObject, PyClassObjectLayout};
6use crate::internal::get_slot::{get_slot, TP_BASE, TP_CLEAR, TP_TRAVERSE};
7use crate::pycell::impl_::PyClassBorrowChecker as _;
8use crate::pycell::{PyBorrowError, PyBorrowMutError};
9use crate::pyclass::boolean_struct::False;
10use crate::types::any::PyAnyMethods;
11use crate::types::PyType;
12use crate::{
13 ffi, Bound, DowncastError, Py, PyAny, PyClass, PyClassInitializer, PyErr, PyObject, PyRef,
14 PyRefMut, PyResult, PyTraverseError, PyTypeCheck, PyVisit, Python,
15};
16use std::ffi::CStr;
17use std::fmt;
18use std::marker::PhantomData;
19use std::os::raw::{c_int, c_void};
20use std::panic::{catch_unwind, AssertUnwindSafe};
21use std::ptr::null_mut;
22
23use super::trampoline;
24use crate::internal_tricks::{clear_eq, traverse_eq};
25
26#[cfg(Py_3_8)]
28#[repr(transparent)]
29pub struct IPowModulo(*mut ffi::PyObject);
30
31#[cfg(not(Py_3_8))]
33#[repr(transparent)]
34pub struct IPowModulo(#[allow(dead_code)] std::mem::MaybeUninit<*mut ffi::PyObject>);
35
36#[allow(non_camel_case_types)]
38pub type ipowfunc = unsafe extern "C" fn(
39 arg1: *mut ffi::PyObject,
40 arg2: *mut ffi::PyObject,
41 arg3: IPowModulo,
42) -> *mut ffi::PyObject;
43
44impl IPowModulo {
45 #[cfg(Py_3_8)]
46 #[inline]
47 pub fn as_ptr(self) -> *mut ffi::PyObject {
48 self.0
49 }
50
51 #[cfg(not(Py_3_8))]
52 #[inline]
53 pub fn as_ptr(self) -> *mut ffi::PyObject {
54 unsafe { ffi::Py_None() }
56 }
57}
58
59#[cfg_attr(test, derive(Clone))]
62pub enum PyMethodDefType {
63 Class(PyMethodDef),
65 Static(PyMethodDef),
67 Method(PyMethodDef),
69 ClassAttribute(PyClassAttributeDef),
71 Getter(PyGetterDef),
73 Setter(PySetterDef),
75 StructMember(ffi::PyMemberDef),
77}
78
79#[derive(Copy, Clone, Debug)]
80pub enum PyMethodType {
81 PyCFunction(ffi::PyCFunction),
82 PyCFunctionWithKeywords(ffi::PyCFunctionWithKeywords),
83 #[cfg(any(Py_3_10, not(Py_LIMITED_API)))]
84 PyCFunctionFastWithKeywords(ffi::PyCFunctionFastWithKeywords),
85}
86
87pub type PyClassAttributeFactory = for<'p> fn(Python<'p>) -> PyResult<PyObject>;
88
89#[derive(Clone, Debug)]
93pub struct PyMethodDef {
94 pub(crate) ml_name: &'static CStr,
95 pub(crate) ml_meth: PyMethodType,
96 pub(crate) ml_flags: c_int,
97 pub(crate) ml_doc: &'static CStr,
98}
99
100#[derive(Copy, Clone)]
101pub struct PyClassAttributeDef {
102 pub(crate) name: &'static CStr,
103 pub(crate) meth: PyClassAttributeFactory,
104}
105
106#[derive(Clone)]
107pub struct PyGetterDef {
108 pub(crate) name: &'static CStr,
109 pub(crate) meth: Getter,
110 pub(crate) doc: &'static CStr,
111}
112
113#[derive(Clone)]
114pub struct PySetterDef {
115 pub(crate) name: &'static CStr,
116 pub(crate) meth: Setter,
117 pub(crate) doc: &'static CStr,
118}
119
120unsafe impl Sync for PyMethodDef {}
121
122unsafe impl Sync for PyGetterDef {}
123
124unsafe impl Sync for PySetterDef {}
125
126impl PyMethodDef {
127 pub const fn noargs(
129 ml_name: &'static CStr,
130 cfunction: ffi::PyCFunction,
131 ml_doc: &'static CStr,
132 ) -> Self {
133 Self {
134 ml_name,
135 ml_meth: PyMethodType::PyCFunction(cfunction),
136 ml_flags: ffi::METH_NOARGS,
137 ml_doc,
138 }
139 }
140
141 pub const fn cfunction_with_keywords(
143 ml_name: &'static CStr,
144 cfunction: ffi::PyCFunctionWithKeywords,
145 ml_doc: &'static CStr,
146 ) -> Self {
147 Self {
148 ml_name,
149 ml_meth: PyMethodType::PyCFunctionWithKeywords(cfunction),
150 ml_flags: ffi::METH_VARARGS | ffi::METH_KEYWORDS,
151 ml_doc,
152 }
153 }
154
155 #[cfg(any(Py_3_10, not(Py_LIMITED_API)))]
157 pub const fn fastcall_cfunction_with_keywords(
158 ml_name: &'static CStr,
159 cfunction: ffi::PyCFunctionFastWithKeywords,
160 ml_doc: &'static CStr,
161 ) -> Self {
162 Self {
163 ml_name,
164 ml_meth: PyMethodType::PyCFunctionFastWithKeywords(cfunction),
165 ml_flags: ffi::METH_FASTCALL | ffi::METH_KEYWORDS,
166 ml_doc,
167 }
168 }
169
170 pub const fn flags(mut self, flags: c_int) -> Self {
171 self.ml_flags |= flags;
172 self
173 }
174
175 pub(crate) fn as_method_def(&self) -> ffi::PyMethodDef {
177 let meth = match self.ml_meth {
178 PyMethodType::PyCFunction(meth) => ffi::PyMethodDefPointer { PyCFunction: meth },
179 PyMethodType::PyCFunctionWithKeywords(meth) => ffi::PyMethodDefPointer {
180 PyCFunctionWithKeywords: meth,
181 },
182 #[cfg(any(Py_3_10, not(Py_LIMITED_API)))]
183 PyMethodType::PyCFunctionFastWithKeywords(meth) => ffi::PyMethodDefPointer {
184 PyCFunctionFastWithKeywords: meth,
185 },
186 };
187
188 ffi::PyMethodDef {
189 ml_name: self.ml_name.as_ptr(),
190 ml_meth: meth,
191 ml_flags: self.ml_flags,
192 ml_doc: self.ml_doc.as_ptr(),
193 }
194 }
195}
196
197impl PyClassAttributeDef {
198 pub const fn new(name: &'static CStr, meth: PyClassAttributeFactory) -> Self {
200 Self { name, meth }
201 }
202}
203
204impl fmt::Debug for PyClassAttributeDef {
207 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
208 f.debug_struct("PyClassAttributeDef")
209 .field("name", &self.name)
210 .finish()
211 }
212}
213
214pub(crate) type Getter =
216 for<'py> unsafe fn(Python<'py>, *mut ffi::PyObject) -> PyResult<*mut ffi::PyObject>;
217pub(crate) type Setter =
218 for<'py> unsafe fn(Python<'py>, *mut ffi::PyObject, *mut ffi::PyObject) -> PyResult<c_int>;
219
220impl PyGetterDef {
221 pub const fn new(name: &'static CStr, getter: Getter, doc: &'static CStr) -> Self {
223 Self {
224 name,
225 meth: getter,
226 doc,
227 }
228 }
229}
230
231impl PySetterDef {
232 pub const fn new(name: &'static CStr, setter: Setter, doc: &'static CStr) -> Self {
234 Self {
235 name,
236 meth: setter,
237 doc,
238 }
239 }
240}
241
242#[doc(hidden)]
278pub unsafe fn _call_traverse<T>(
279 slf: *mut ffi::PyObject,
280 impl_: fn(&T, PyVisit<'_>) -> Result<(), PyTraverseError>,
281 visit: ffi::visitproc,
282 arg: *mut c_void,
283 current_traverse: ffi::traverseproc,
284) -> c_int
285where
286 T: PyClass,
287{
288 let trap = PanicTrap::new("uncaught panic inside __traverse__ handler");
296 let lock = LockGIL::during_traverse();
297
298 let super_retval = unsafe { call_super_traverse(slf, visit, arg, current_traverse) };
299 if super_retval != 0 {
300 return super_retval;
301 }
302
303 let class_object: &PyClassObject<T> = unsafe { &*slf.cast() };
306
307 let retval =
308 if class_object.check_threadsafe().is_ok()
311 && class_object.borrow_checker().try_borrow().is_ok() {
313 struct TraverseGuard<'a, T: PyClass>(&'a PyClassObject<T>);
314 impl<T: PyClass> Drop for TraverseGuard<'_, T> {
315 fn drop(&mut self) {
316 self.0.borrow_checker().release_borrow()
317 }
318 }
319
320 let _guard = TraverseGuard(class_object);
323 let instance = unsafe {&*class_object.contents.value.get()};
324
325 let visit = PyVisit { visit, arg, _guard: PhantomData };
326
327 match catch_unwind(AssertUnwindSafe(move || impl_(instance, visit))) {
328 Ok(Ok(())) => 0,
329 Ok(Err(traverse_error)) => traverse_error.into_inner(),
330 Err(_err) => -1,
331 }
332 } else {
333 0
334 };
335
336 drop(lock);
338 trap.disarm();
339 retval
340}
341
342unsafe fn call_super_traverse(
353 obj: *mut ffi::PyObject,
354 visit: ffi::visitproc,
355 arg: *mut c_void,
356 current_traverse: ffi::traverseproc,
357) -> c_int {
358 let mut ty = unsafe { ffi::Py_TYPE(obj) };
363 let mut traverse: Option<ffi::traverseproc>;
364
365 loop {
367 traverse = unsafe { get_slot(ty, TP_TRAVERSE) };
368 if traverse_eq(traverse, current_traverse) {
369 break;
370 }
371 ty = unsafe { get_slot(ty, TP_BASE) };
372 if ty.is_null() {
373 return 0;
375 }
376 }
377
378 while traverse_eq(traverse, current_traverse) {
380 ty = unsafe { get_slot(ty, TP_BASE) };
381 if ty.is_null() {
382 break;
383 }
384 traverse = unsafe { get_slot(ty, TP_TRAVERSE) };
385 }
386
387 if let Some(traverse) = traverse {
389 return unsafe { traverse(obj, visit, arg) };
390 }
391
392 0
394}
395
396pub unsafe fn _call_clear(
398 slf: *mut ffi::PyObject,
399 impl_: for<'py> unsafe fn(Python<'py>, *mut ffi::PyObject) -> PyResult<()>,
400 current_clear: ffi::inquiry,
401) -> c_int {
402 unsafe {
403 trampoline::trampoline(move |py| {
404 let super_retval = call_super_clear(py, slf, current_clear);
405 if super_retval != 0 {
406 return Err(PyErr::fetch(py));
407 }
408 impl_(py, slf)?;
409 Ok(0)
410 })
411 }
412}
413
414unsafe fn call_super_clear(
425 py: Python<'_>,
426 obj: *mut ffi::PyObject,
427 current_clear: ffi::inquiry,
428) -> c_int {
429 let mut ty = unsafe { PyType::from_borrowed_type_ptr(py, ffi::Py_TYPE(obj)) };
430 let mut clear: Option<ffi::inquiry>;
431
432 loop {
434 clear = ty.get_slot(TP_CLEAR);
435 if clear_eq(clear, current_clear) {
436 break;
437 }
438 let base = ty.get_slot(TP_BASE);
439 if base.is_null() {
440 return 0;
442 }
443 ty = unsafe { PyType::from_borrowed_type_ptr(py, base) };
444 }
445
446 while clear_eq(clear, current_clear) {
448 let base = ty.get_slot(TP_BASE);
449 if base.is_null() {
450 break;
451 }
452 ty = unsafe { PyType::from_borrowed_type_ptr(py, base) };
453 clear = ty.get_slot(TP_CLEAR);
454 }
455
456 if let Some(clear) = clear {
458 return unsafe { clear(obj) };
459 }
460
461 0
463}
464
465pub struct IterBaseTag;
468
469impl IterBaseTag {
470 #[inline]
471 pub fn convert<'py, Value, Target>(self, py: Python<'py>, value: Value) -> PyResult<Target>
472 where
473 Value: IntoPyCallbackOutput<'py, Target>,
474 {
475 value.convert(py)
476 }
477}
478
479pub trait IterBaseKind {
480 #[inline]
481 fn iter_tag(&self) -> IterBaseTag {
482 IterBaseTag
483 }
484}
485
486impl<Value> IterBaseKind for &Value {}
487
488pub struct IterOptionTag;
489
490impl IterOptionTag {
491 #[inline]
492 pub fn convert<'py, Value>(
493 self,
494 py: Python<'py>,
495 value: Option<Value>,
496 ) -> PyResult<*mut ffi::PyObject>
497 where
498 Value: IntoPyCallbackOutput<'py, *mut ffi::PyObject>,
499 {
500 match value {
501 Some(value) => value.convert(py),
502 None => Ok(null_mut()),
503 }
504 }
505}
506
507pub trait IterOptionKind {
508 #[inline]
509 fn iter_tag(&self) -> IterOptionTag {
510 IterOptionTag
511 }
512}
513
514impl<Value> IterOptionKind for Option<Value> {}
515
516pub struct IterResultOptionTag;
517
518impl IterResultOptionTag {
519 #[inline]
520 pub fn convert<'py, Value, Error>(
521 self,
522 py: Python<'py>,
523 value: Result<Option<Value>, Error>,
524 ) -> PyResult<*mut ffi::PyObject>
525 where
526 Value: IntoPyCallbackOutput<'py, *mut ffi::PyObject>,
527 Error: Into<PyErr>,
528 {
529 match value {
530 Ok(Some(value)) => value.convert(py),
531 Ok(None) => Ok(null_mut()),
532 Err(err) => Err(err.into()),
533 }
534 }
535}
536
537pub trait IterResultOptionKind {
538 #[inline]
539 fn iter_tag(&self) -> IterResultOptionTag {
540 IterResultOptionTag
541 }
542}
543
544impl<Value, Error> IterResultOptionKind for Result<Option<Value>, Error> {}
545
546pub struct AsyncIterBaseTag;
549
550impl AsyncIterBaseTag {
551 #[inline]
552 pub fn convert<'py, Value, Target>(self, py: Python<'py>, value: Value) -> PyResult<Target>
553 where
554 Value: IntoPyCallbackOutput<'py, Target>,
555 {
556 value.convert(py)
557 }
558}
559
560pub trait AsyncIterBaseKind {
561 #[inline]
562 fn async_iter_tag(&self) -> AsyncIterBaseTag {
563 AsyncIterBaseTag
564 }
565}
566
567impl<Value> AsyncIterBaseKind for &Value {}
568
569pub struct AsyncIterOptionTag;
570
571impl AsyncIterOptionTag {
572 #[inline]
573 pub fn convert<'py, Value>(
574 self,
575 py: Python<'py>,
576 value: Option<Value>,
577 ) -> PyResult<*mut ffi::PyObject>
578 where
579 Value: IntoPyCallbackOutput<'py, *mut ffi::PyObject>,
580 {
581 match value {
582 Some(value) => value.convert(py),
583 None => Err(PyStopAsyncIteration::new_err(())),
584 }
585 }
586}
587
588pub trait AsyncIterOptionKind {
589 #[inline]
590 fn async_iter_tag(&self) -> AsyncIterOptionTag {
591 AsyncIterOptionTag
592 }
593}
594
595impl<Value> AsyncIterOptionKind for Option<Value> {}
596
597pub struct AsyncIterResultOptionTag;
598
599impl AsyncIterResultOptionTag {
600 #[inline]
601 pub fn convert<'py, Value, Error>(
602 self,
603 py: Python<'py>,
604 value: Result<Option<Value>, Error>,
605 ) -> PyResult<*mut ffi::PyObject>
606 where
607 Value: IntoPyCallbackOutput<'py, *mut ffi::PyObject>,
608 Error: Into<PyErr>,
609 {
610 match value {
611 Ok(Some(value)) => value.convert(py),
612 Ok(None) => Err(PyStopAsyncIteration::new_err(())),
613 Err(err) => Err(err.into()),
614 }
615 }
616}
617
618pub trait AsyncIterResultOptionKind {
619 #[inline]
620 fn async_iter_tag(&self) -> AsyncIterResultOptionTag {
621 AsyncIterResultOptionTag
622 }
623}
624
625impl<Value, Error> AsyncIterResultOptionKind for Result<Option<Value>, Error> {}
626
627pub struct BoundRef<'a, 'py, T>(pub &'a Bound<'py, T>);
635
636impl<'a, 'py> BoundRef<'a, 'py, PyAny> {
637 pub unsafe fn ref_from_ptr(py: Python<'py>, ptr: &'a *mut ffi::PyObject) -> Self {
638 unsafe { BoundRef(Bound::ref_from_ptr(py, ptr)) }
639 }
640
641 pub unsafe fn ref_from_ptr_or_opt(
642 py: Python<'py>,
643 ptr: &'a *mut ffi::PyObject,
644 ) -> Option<Self> {
645 unsafe { Bound::ref_from_ptr_or_opt(py, ptr).as_ref().map(BoundRef) }
646 }
647
648 pub fn downcast<T: PyTypeCheck>(self) -> Result<BoundRef<'a, 'py, T>, DowncastError<'a, 'py>> {
649 self.0.downcast::<T>().map(BoundRef)
650 }
651
652 pub unsafe fn downcast_unchecked<T>(self) -> BoundRef<'a, 'py, T> {
653 unsafe { BoundRef(self.0.downcast_unchecked::<T>()) }
654 }
655}
656
657impl<'a, 'py, T: PyClass> TryFrom<BoundRef<'a, 'py, T>> for PyRef<'py, T> {
658 type Error = PyBorrowError;
659 #[inline]
660 fn try_from(value: BoundRef<'a, 'py, T>) -> Result<Self, Self::Error> {
661 value.0.try_borrow()
662 }
663}
664
665impl<'a, 'py, T: PyClass<Frozen = False>> TryFrom<BoundRef<'a, 'py, T>> for PyRefMut<'py, T> {
666 type Error = PyBorrowMutError;
667 #[inline]
668 fn try_from(value: BoundRef<'a, 'py, T>) -> Result<Self, Self::Error> {
669 value.0.try_borrow_mut()
670 }
671}
672
673impl<'a, 'py, T> From<BoundRef<'a, 'py, T>> for Bound<'py, T> {
674 #[inline]
675 fn from(bound: BoundRef<'a, 'py, T>) -> Self {
676 bound.0.clone()
677 }
678}
679
680impl<'a, 'py, T> From<BoundRef<'a, 'py, T>> for &'a Bound<'py, T> {
681 #[inline]
682 fn from(bound: BoundRef<'a, 'py, T>) -> Self {
683 bound.0
684 }
685}
686
687impl<T> From<BoundRef<'_, '_, T>> for Py<T> {
688 #[inline]
689 fn from(bound: BoundRef<'_, '_, T>) -> Self {
690 bound.0.clone().unbind()
691 }
692}
693
694impl<'py, T> std::ops::Deref for BoundRef<'_, 'py, T> {
695 type Target = Bound<'py, T>;
696 #[inline]
697 fn deref(&self) -> &Self::Target {
698 self.0
699 }
700}
701
702pub unsafe fn tp_new_impl<T: PyClass>(
703 py: Python<'_>,
704 initializer: PyClassInitializer<T>,
705 target_type: *mut ffi::PyTypeObject,
706) -> PyResult<*mut ffi::PyObject> {
707 unsafe {
708 initializer
709 .create_class_object_of_type(py, target_type)
710 .map(Bound::into_ptr)
711 }
712}
713
714#[cfg(test)]
715mod tests {
716 #[test]
717 #[cfg(any(Py_3_10, not(Py_LIMITED_API)))]
718 fn test_fastcall_function_with_keywords() {
719 use super::PyMethodDef;
720 use crate::types::{PyAnyMethods, PyCFunction};
721 use crate::{ffi, Python};
722
723 Python::with_gil(|py| {
724 unsafe extern "C" fn accepts_no_arguments(
725 _slf: *mut ffi::PyObject,
726 _args: *const *mut ffi::PyObject,
727 nargs: ffi::Py_ssize_t,
728 kwargs: *mut ffi::PyObject,
729 ) -> *mut ffi::PyObject {
730 assert_eq!(nargs, 0);
731 assert!(kwargs.is_null());
732 unsafe { Python::assume_gil_acquired().None().into_ptr() }
733 }
734
735 let f = PyCFunction::internal_new(
736 py,
737 &PyMethodDef::fastcall_cfunction_with_keywords(
738 ffi::c_str!("test"),
739 accepts_no_arguments,
740 ffi::c_str!("doc"),
741 ),
742 None,
743 )
744 .unwrap();
745
746 f.call0().unwrap();
747 });
748 }
749}