From 9e969c4a4ee6d81c7302f75f56f833af7d7b308e Mon Sep 17 00:00:00 2001 From: kngwyu Date: Wed, 9 Sep 2020 17:26:54 +0900 Subject: [PATCH] Add null-check for function's documents --- pyo3-derive-backend/src/pyproto.rs | 2 +- src/class/methods.rs | 32 ++++++++++++++++++------------ 2 files changed, 20 insertions(+), 14 deletions(-) diff --git a/pyo3-derive-backend/src/pyproto.rs b/pyo3-derive-backend/src/pyproto.rs index c9bebbe08d4..67976f5901d 100644 --- a/pyo3-derive-backend/src/pyproto.rs +++ b/pyo3-derive-backend/src/pyproto.rs @@ -99,7 +99,7 @@ fn impl_proto_impl( ml_name: stringify!(#name), ml_meth: pyo3::class::PyMethodType::PyCFunctionWithKeywords(__wrap), ml_flags: pyo3::ffi::METH_VARARGS | pyo3::ffi::METH_KEYWORDS | #coexist, - ml_doc: "" + ml_doc: "\0" } }) }); diff --git a/src/class/methods.rs b/src/class/methods.rs index a34260d6c5c..173dfe02c18 100644 --- a/src/class/methods.rs +++ b/src/class/methods.rs @@ -2,7 +2,7 @@ use crate::{ffi, PyObject, Python}; use libc::c_int; -use std::ffi::CString; +use std::ffi::{CStr, CString}; use std::fmt; /// `PyMethodDefType` represents different types of Python callable objects. @@ -73,6 +73,18 @@ unsafe impl Sync for PySetterDef {} unsafe impl Sync for ffi::PyGetSetDef {} +fn get_name(name: &str) -> *const std::os::raw::c_char { + CString::new(name) + .expect("Method name must not contain NULL byte") + .into_raw() as _ +} + +fn get_doc(doc: &'static str) -> *const std::os::raw::c_char { + CStr::from_bytes_with_nul(doc.as_bytes()) + .expect("Document must be terminated with NULL byte") + .as_ptr() +} + impl PyMethodDef { /// Convert `PyMethodDef` to Python method definition struct `ffi::PyMethodDef` pub fn as_method_def(&self) -> ffi::PyMethodDef { @@ -84,12 +96,10 @@ impl PyMethodDef { }; ffi::PyMethodDef { - ml_name: CString::new(self.ml_name) - .expect("Method name must not contain NULL byte") - .into_raw(), + ml_name: get_name(self.ml_name), ml_meth: Some(meth), ml_flags: self.ml_flags, - ml_doc: self.ml_doc.as_ptr() as *const _, + ml_doc: get_doc(self.ml_doc), } } } @@ -108,12 +118,10 @@ impl PyGetterDef { /// Copy descriptor information to `ffi::PyGetSetDef` pub fn copy_to(&self, dst: &mut ffi::PyGetSetDef) { if dst.name.is_null() { - dst.name = CString::new(self.name) - .expect("Method name must not contain NULL byte") - .into_raw(); + dst.name = get_name(self.name) as _; } if dst.doc.is_null() { - dst.doc = self.doc.as_ptr() as *mut libc::c_char; + dst.doc = get_doc(self.doc) as _; } dst.get = Some(self.meth); } @@ -123,12 +131,10 @@ impl PySetterDef { /// Copy descriptor information to `ffi::PyGetSetDef` pub fn copy_to(&self, dst: &mut ffi::PyGetSetDef) { if dst.name.is_null() { - dst.name = CString::new(self.name) - .expect("Method name must not contain NULL byte") - .into_raw(); + dst.name = get_name(self.name) as _; } if dst.doc.is_null() { - dst.doc = self.doc.as_ptr() as *mut libc::c_char; + dst.doc = get_doc(self.doc) as _; } dst.set = Some(self.meth); }