gstreamer-base/adapter: Add error checking at the bindings level

Instead of running into (inconsistent even) assertions inside the C
code.

Also don't return uninitialized data and signed offsets from the
masked_scan() functions.

Fixes https://gitlab.freedesktop.org/gstreamer/gstreamer-rs/-/issues/298
This commit is contained in:
Sebastian Dröge 2020-12-06 19:39:28 +02:00
parent 15722ec5d2
commit 1199e89dca
3 changed files with 294 additions and 166 deletions

View file

@ -42,7 +42,6 @@ name = "GstBase.Adapter"
status = "generate" status = "generate"
final_type = true final_type = true
concurrency = "send-unique" concurrency = "send-unique"
[[object.function]] [[object.function]]
name = "map" name = "map"
# Unsafe # Unsafe
@ -54,7 +53,7 @@ concurrency = "send-unique"
ignore = true ignore = true
[[object.function]] [[object.function]]
name = "copy" pattern = "copy.*"
# Unsafe # Unsafe
manual = true manual = true
@ -64,44 +63,24 @@ concurrency = "send-unique"
manual = true manual = true
[[object.function]] [[object.function]]
name = "take" pattern = "take.*"
# Useless copying of data # Unsafe
ignore = true ignore = true
[[object.function]] [[object.function]]
name = "copy_bytes" pattern = "get.*"
[object.function.return] # Unsafe
nullable_return_is_error = "Failed to copy bytes" ignore = true
[[object.function]] [[object.function]]
name = "get_buffer" pattern = "masked.*"
[object.function.return] # Unsafe
nullable_return_is_error = "Failed to get buffer" ignore = true
[[object.function]] [[object.function]]
name = "get_buffer_fast" name = "flush"
[object.function.return] # Unsafe Buffer
nullable_return_is_error = "Failed to get buffer" manual = true
[[object.function]]
name = "get_buffer_list"
[object.function.return]
nullable_return_is_error = "Failed to get buffer list"
[[object.function]]
name = "take_buffer"
[object.function.return]
nullable_return_is_error = "Failed to take buffer"
[[object.function]]
name = "take_buffer_fast"
[object.function.return]
nullable_return_is_error = "Failed to take buffer"
[[object.function]]
name = "take_buffer_list"
[object.function.return]
nullable_return_is_error = "Failed to take buffer list"
[[object]] [[object]]
name = "GstBase.FlowCombiner" name = "GstBase.FlowCombiner"

View file

@ -9,10 +9,23 @@
use crate::Adapter; use crate::Adapter;
use glib::translate::*; use glib::translate::*;
use std::io; use std::io;
use std::mem;
use std::ops; use std::ops;
impl Adapter { impl Adapter {
pub fn copy(&self, offset: usize, dest: &mut [u8]) { pub fn copy(&self, offset: usize, dest: &mut [u8]) -> Result<(), glib::BoolError> {
if offset
.checked_add(dest.len())
.map(|end| end <= self.available())
!= Some(true)
{
return Err(glib::glib_bool_error!("Not enough data available"));
}
if dest.is_empty() {
return Ok(());
}
unsafe { unsafe {
let size = dest.len(); let size = dest.len();
ffi::gst_adapter_copy( ffi::gst_adapter_copy(
@ -22,6 +35,240 @@ impl Adapter {
size, size,
); );
} }
Ok(())
}
pub fn copy_bytes(&self, offset: usize, size: usize) -> Result<glib::Bytes, glib::BoolError> {
if offset.checked_add(size).map(|end| end <= self.available()) != Some(true) {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if size == 0 {
return Ok(glib::Bytes::from_static(&[]));
}
unsafe {
Ok(from_glib_full(ffi::gst_adapter_copy_bytes(
self.to_glib_none().0,
offset,
size,
)))
}
}
pub fn flush(&self, flush: usize) -> Result<(), glib::BoolError> {
if flush > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
unsafe {
ffi::gst_adapter_flush(self.to_glib_none().0, flush);
}
Ok(())
}
pub fn get_buffer(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(gst::Buffer::new());
}
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_get_buffer(self.to_glib_none().0, nbytes))
.ok_or_else(|| glib::glib_bool_error!("Failed to get buffer"))
}
}
pub fn get_buffer_fast(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(gst::Buffer::new());
}
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_get_buffer_fast(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to get buffer"))
}
}
pub fn get_buffer_list(&self, nbytes: usize) -> Result<gst::BufferList, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(gst::BufferList::new());
}
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_get_buffer_list(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to get buffer list"))
}
}
pub fn get_list(&self, nbytes: usize) -> Result<Vec<gst::Buffer>, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(Vec::new());
}
unsafe {
Ok(FromGlibPtrContainer::from_glib_full(
ffi::gst_adapter_get_list(self.to_glib_none().0, nbytes),
))
}
}
pub fn masked_scan_uint32(
&self,
mask: u32,
pattern: u32,
offset: usize,
size: usize,
) -> Result<Option<usize>, glib::BoolError> {
if offset.checked_add(size).map(|end| end <= self.available()) != Some(true) {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if size == 0 || ((!mask) & pattern) != 0 {
return Ok(None);
}
unsafe {
let ret = ffi::gst_adapter_masked_scan_uint32(
self.to_glib_none().0,
mask,
pattern,
offset,
size,
);
if ret == -1 {
Ok(None)
} else {
assert!(ret >= 0);
Ok(Some(ret as usize))
}
}
}
pub fn masked_scan_uint32_peek(
&self,
mask: u32,
pattern: u32,
offset: usize,
size: usize,
) -> Result<Option<(usize, u32)>, glib::BoolError> {
if offset.checked_add(size).map(|end| end <= self.available()) != Some(true) {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if size == 0 || ((!mask) & pattern) != 0 {
return Ok(None);
}
unsafe {
let mut value = mem::MaybeUninit::uninit();
let ret = ffi::gst_adapter_masked_scan_uint32_peek(
self.to_glib_none().0,
mask,
pattern,
offset,
size,
value.as_mut_ptr(),
);
if ret == -1 {
Ok(None)
} else {
assert!(ret >= 0);
let value = value.assume_init();
Ok(Some((ret as usize, value)))
}
}
}
pub fn take_buffer(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(gst::Buffer::new());
}
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_take_buffer(self.to_glib_none().0, nbytes))
.ok_or_else(|| glib::glib_bool_error!("Failed to take buffer"))
}
}
pub fn take_buffer_fast(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(gst::Buffer::new());
}
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_take_buffer_fast(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to take buffer"))
}
}
pub fn take_buffer_list(&self, nbytes: usize) -> Result<gst::BufferList, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(gst::BufferList::new());
}
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_take_buffer_list(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to take buffer list"))
}
}
pub fn take_list(&self, nbytes: usize) -> Result<Vec<gst::Buffer>, glib::BoolError> {
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(Vec::new());
}
unsafe {
Ok(FromGlibPtrContainer::from_glib_full(
ffi::gst_adapter_take_list(self.to_glib_none().0, nbytes),
))
}
} }
pub fn push(&self, buf: gst::Buffer) { pub fn push(&self, buf: gst::Buffer) {
@ -50,8 +297,11 @@ impl io::Read for Adapter {
len = buf.len(); len = buf.len();
} }
self.copy(0, &mut buf[0..len]); self.copy(0, &mut buf[0..len])
self.flush(len); .map_err(|err| io::Error::new(io::ErrorKind::Other, err))?;
// Can't happen because we checked the size first
self.flush(len).expect("Failed to flush");
Ok(len) Ok(len)
} }
} }
@ -80,8 +330,7 @@ impl UniqueAdapter {
} }
pub fn copy_bytes(&self, offset: usize, size: usize) -> Result<glib::Bytes, glib::BoolError> { pub fn copy_bytes(&self, offset: usize, size: usize) -> Result<glib::Bytes, glib::BoolError> {
// TBD: https://gitlab.freedesktop.org/gstreamer/gstreamer-rs/-/issues/298 self.0.copy_bytes(offset, size)
Ok(self.0.copy_bytes(offset, size))
} }
#[cfg(any(feature = "v1_10", feature = "dox"))] #[cfg(any(feature = "v1_10", feature = "dox"))]
@ -96,8 +345,8 @@ impl UniqueAdapter {
self.0.dts_at_discont() self.0.dts_at_discont()
} }
pub fn flush(&mut self, flush: usize) { pub fn flush(&mut self, flush: usize) -> Result<(), glib::BoolError> {
self.0.flush(flush); self.0.flush(flush)
} }
pub fn get_buffer(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> { pub fn get_buffer(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
@ -112,11 +361,17 @@ impl UniqueAdapter {
self.0.get_buffer_list(nbytes) self.0.get_buffer_list(nbytes)
} }
pub fn get_list(&self, nbytes: usize) -> Vec<gst::Buffer> { pub fn get_list(&self, nbytes: usize) -> Result<Vec<gst::Buffer>, glib::BoolError> {
self.0.get_list(nbytes) self.0.get_list(nbytes)
} }
pub fn masked_scan_uint32(&self, mask: u32, pattern: u32, offset: usize, size: usize) -> isize { pub fn masked_scan_uint32(
&self,
mask: u32,
pattern: u32,
offset: usize,
size: usize,
) -> Result<Option<usize>, glib::BoolError> {
self.0.masked_scan_uint32(mask, pattern, offset, size) self.0.masked_scan_uint32(mask, pattern, offset, size)
} }
@ -126,7 +381,7 @@ impl UniqueAdapter {
pattern: u32, pattern: u32,
offset: usize, offset: usize,
size: usize, size: usize,
) -> (isize, u32) { ) -> Result<Option<(usize, u32)>, glib::BoolError> {
self.0.masked_scan_uint32_peek(mask, pattern, offset, size) self.0.masked_scan_uint32_peek(mask, pattern, offset, size)
} }
@ -176,12 +431,12 @@ impl UniqueAdapter {
self.0.take_buffer_list(nbytes) self.0.take_buffer_list(nbytes)
} }
pub fn take_list(&mut self, nbytes: usize) -> Vec<gst::Buffer> { pub fn take_list(&mut self, nbytes: usize) -> Result<Vec<gst::Buffer>, glib::BoolError> {
self.0.take_list(nbytes) self.0.take_list(nbytes)
} }
pub fn copy(&self, offset: usize, dest: &mut [u8]) { pub fn copy(&self, offset: usize, dest: &mut [u8]) -> Result<(), glib::BoolError> {
self.0.copy(offset, dest); self.0.copy(offset, dest)
} }
pub fn push(&mut self, buf: gst::Buffer) { pub fn push(&mut self, buf: gst::Buffer) {
@ -191,13 +446,21 @@ impl UniqueAdapter {
pub fn map(&mut self, nbytes: usize) -> Result<UniqueAdapterMap, glib::error::BoolError> { pub fn map(&mut self, nbytes: usize) -> Result<UniqueAdapterMap, glib::error::BoolError> {
use std::slice; use std::slice;
if nbytes > self.available() {
return Err(glib::glib_bool_error!("Not enough data available"));
}
if nbytes == 0 {
return Ok(UniqueAdapterMap(None, &[]));
}
unsafe { unsafe {
let ptr = ffi::gst_adapter_map(self.0.to_glib_none().0, nbytes); let ptr = ffi::gst_adapter_map(self.0.to_glib_none().0, nbytes);
if ptr.is_null() { if ptr.is_null() {
Err(glib::glib_bool_error!("size bytes are not available")) Err(glib::glib_bool_error!("size bytes are not available"))
} else { } else {
Ok(UniqueAdapterMap( Ok(UniqueAdapterMap(
self, Some(self),
slice::from_raw_parts(ptr as *const u8, nbytes), slice::from_raw_parts(ptr as *const u8, nbytes),
)) ))
} }
@ -206,12 +469,14 @@ impl UniqueAdapter {
} }
#[derive(Debug)] #[derive(Debug)]
pub struct UniqueAdapterMap<'a>(&'a UniqueAdapter, &'a [u8]); pub struct UniqueAdapterMap<'a>(Option<&'a UniqueAdapter>, &'a [u8]);
impl<'a> Drop for UniqueAdapterMap<'a> { impl<'a> Drop for UniqueAdapterMap<'a> {
fn drop(&mut self) { fn drop(&mut self) {
unsafe { if let Some(adapter) = self.0 {
ffi::gst_adapter_unmap((self.0).0.to_glib_none().0); unsafe {
ffi::gst_adapter_unmap(adapter.0.to_glib_none().0);
}
} }
} }
} }

View file

@ -34,16 +34,6 @@ impl Adapter {
} }
} }
pub fn copy_bytes(&self, offset: usize, size: usize) -> glib::Bytes {
unsafe {
from_glib_full(ffi::gst_adapter_copy_bytes(
self.to_glib_none().0,
offset,
size,
))
}
}
#[cfg(any(feature = "v1_10", feature = "dox"))] #[cfg(any(feature = "v1_10", feature = "dox"))]
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_10")))] #[cfg_attr(feature = "dox", doc(cfg(feature = "v1_10")))]
pub fn distance_from_discont(&self) -> u64 { pub fn distance_from_discont(&self) -> u64 {
@ -56,76 +46,6 @@ impl Adapter {
unsafe { from_glib(ffi::gst_adapter_dts_at_discont(self.to_glib_none().0)) } unsafe { from_glib(ffi::gst_adapter_dts_at_discont(self.to_glib_none().0)) }
} }
pub fn flush(&self, flush: usize) {
unsafe {
ffi::gst_adapter_flush(self.to_glib_none().0, flush);
}
}
pub fn get_buffer(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_get_buffer(self.to_glib_none().0, nbytes))
.ok_or_else(|| glib::glib_bool_error!("Failed to get buffer"))
}
}
pub fn get_buffer_fast(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_get_buffer_fast(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to get buffer"))
}
}
pub fn get_buffer_list(&self, nbytes: usize) -> Result<gst::BufferList, glib::BoolError> {
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_get_buffer_list(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to get buffer list"))
}
}
pub fn get_list(&self, nbytes: usize) -> Vec<gst::Buffer> {
unsafe {
FromGlibPtrContainer::from_glib_full(ffi::gst_adapter_get_list(
self.to_glib_none().0,
nbytes,
))
}
}
pub fn masked_scan_uint32(&self, mask: u32, pattern: u32, offset: usize, size: usize) -> isize {
unsafe {
ffi::gst_adapter_masked_scan_uint32(self.to_glib_none().0, mask, pattern, offset, size)
}
}
pub fn masked_scan_uint32_peek(
&self,
mask: u32,
pattern: u32,
offset: usize,
size: usize,
) -> (isize, u32) {
unsafe {
let mut value = mem::MaybeUninit::uninit();
let ret = ffi::gst_adapter_masked_scan_uint32_peek(
self.to_glib_none().0,
mask,
pattern,
offset,
size,
value.as_mut_ptr(),
);
let value = value.assume_init();
(ret, value)
}
}
#[cfg(any(feature = "v1_10", feature = "dox"))] #[cfg(any(feature = "v1_10", feature = "dox"))]
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_10")))] #[cfg_attr(feature = "dox", doc(cfg(feature = "v1_10")))]
pub fn offset_at_discont(&self) -> u64 { pub fn offset_at_discont(&self) -> u64 {
@ -198,42 +118,6 @@ impl Adapter {
pub fn pts_at_discont(&self) -> gst::ClockTime { pub fn pts_at_discont(&self) -> gst::ClockTime {
unsafe { from_glib(ffi::gst_adapter_pts_at_discont(self.to_glib_none().0)) } unsafe { from_glib(ffi::gst_adapter_pts_at_discont(self.to_glib_none().0)) }
} }
pub fn take_buffer(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_take_buffer(self.to_glib_none().0, nbytes))
.ok_or_else(|| glib::glib_bool_error!("Failed to take buffer"))
}
}
pub fn take_buffer_fast(&self, nbytes: usize) -> Result<gst::Buffer, glib::BoolError> {
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_take_buffer_fast(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to take buffer"))
}
}
pub fn take_buffer_list(&self, nbytes: usize) -> Result<gst::BufferList, glib::BoolError> {
unsafe {
Option::<_>::from_glib_full(ffi::gst_adapter_take_buffer_list(
self.to_glib_none().0,
nbytes,
))
.ok_or_else(|| glib::glib_bool_error!("Failed to take buffer list"))
}
}
pub fn take_list(&self, nbytes: usize) -> Vec<gst::Buffer> {
unsafe {
FromGlibPtrContainer::from_glib_full(ffi::gst_adapter_take_list(
self.to_glib_none().0,
nbytes,
))
}
}
} }
impl Default for Adapter { impl Default for Adapter {