gstreamer-rs/gstreamer-sdp/src/sdp_time.rs
Sebastian Dröge 2ba5105b80 Implement Sync/Send for more types and don't implement Send for TypeFind
They can actually be shared with multiple threads at the same time
safely as all functions requiring an immutable reference are
thread-safe.

OTOH TypeFind can't be shared safely between different threads as not
all implementations of the TypeFind struct are thread-safe.
2019-12-18 18:37:44 +02:00

115 lines
3.1 KiB
Rust

// Copyright (C) 2018 Sebastian Dröge <sebastian@centricular.com>
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
use std::ffi::CStr;
use std::fmt;
use std::mem;
use std::os::raw::c_char;
use std::ptr;
use glib::translate::*;
use gst_sdp_sys;
#[repr(C)]
pub struct SDPTime(pub(crate) gst_sdp_sys::GstSDPTime);
unsafe impl Send for SDPTime {}
unsafe impl Sync for SDPTime {}
impl SDPTime {
pub fn new(start: &str, stop: &str, repeat: &[&str]) -> Self {
assert_initialized_main_thread!();
unsafe {
let mut time = mem::MaybeUninit::zeroed();
gst_sdp_sys::gst_sdp_time_set(
time.as_mut_ptr(),
start.to_glib_none().0,
stop.to_glib_none().0,
repeat.to_glib_none().0,
);
SDPTime(time.assume_init())
}
}
pub fn start(&self) -> Option<&str> {
unsafe {
if self.0.start.is_null() {
None
} else {
Some(CStr::from_ptr(self.0.start).to_str().unwrap())
}
}
}
pub fn stop(&self) -> Option<&str> {
unsafe {
if self.0.stop.is_null() {
None
} else {
Some(CStr::from_ptr(self.0.stop).to_str().unwrap())
}
}
}
pub fn repeat(&self) -> Vec<&str> {
#[allow(clippy::cast_ptr_alignment)]
unsafe {
if self.0.repeat.is_null() || (*self.0.repeat).data.is_null() {
return vec![];
}
let arr = (*self.0.repeat).data as *const *const c_char;
let len = (*self.0.repeat).len as usize;
let mut vec = Vec::with_capacity(len);
for i in 0..len {
vec.push(CStr::from_ptr(*arr.add(i)).to_str().unwrap());
}
vec
}
}
}
impl Clone for SDPTime {
fn clone(&self) -> Self {
assert_initialized_main_thread!();
#[allow(clippy::cast_ptr_alignment)]
unsafe {
let mut time = mem::MaybeUninit::zeroed();
gst_sdp_sys::gst_sdp_time_set(
time.as_mut_ptr(),
self.0.start,
self.0.stop,
if self.0.repeat.is_null() {
ptr::null_mut()
} else {
(*self.0.repeat).data as *mut *const c_char
},
);
SDPTime(time.assume_init())
}
}
}
impl Drop for SDPTime {
fn drop(&mut self) {
unsafe {
gst_sdp_sys::gst_sdp_time_clear(&mut self.0);
}
}
}
impl fmt::Debug for SDPTime {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("SDPTime")
.field("start", &self.start())
.field("stop", &self.stop())
.field("repeat", &self.repeat())
.finish()
}
}