use std::convert::TryInto;
use std::time::Duration;
use std::{io, mem, sync::Arc};
use crate::buffer::{Metadata, Type};
use crate::device::{Device, Handle};
use crate::io::traits::{CaptureStream, Stream as StreamTrait};
use crate::io::userptr::arena::Arena;
use crate::memory::Memory;
use crate::v4l2;
use crate::v4l_sys::*;
pub struct Stream {
handle: Arc<Handle>,
arena: Arena,
arena_index: usize,
buf_type: Type,
buf_meta: Vec<Metadata>,
timeout: Option<i32>,
active: bool,
}
impl Stream {
pub fn new(dev: &Device, buf_type: Type) -> io::Result<Self> {
Stream::with_buffers(dev, buf_type, 4)
}
pub fn with_buffers(dev: &Device, buf_type: Type, buf_count: u32) -> io::Result<Self> {
let mut arena = Arena::new(dev.handle(), buf_type);
let count = arena.allocate(buf_count)?;
let mut buf_meta = Vec::new();
buf_meta.resize(count as usize, Metadata::default());
Ok(Stream {
handle: dev.handle(),
arena,
arena_index: 0,
buf_type,
buf_meta,
active: false,
timeout: None,
})
}
pub fn handle(&self) -> Arc<Handle> {
self.handle.clone()
}
pub fn set_timeout(&mut self, duration: Duration) {
self.timeout = Some(duration.as_millis().try_into().unwrap());
}
pub fn clear_timeout(&mut self) {
self.timeout = None;
}
fn buffer_desc(&self) -> v4l2_buffer {
v4l2_buffer {
type_: self.buf_type as u32,
memory: Memory::UserPtr as u32,
..unsafe { mem::zeroed() }
}
}
}
impl Drop for Stream {
fn drop(&mut self) {
if let Err(e) = self.stop() {
if let Some(code) = e.raw_os_error() {
if code == 19 {
return;
}
}
panic!("{:?}", e)
}
}
}
impl StreamTrait for Stream {
type Item = [u8];
fn start(&mut self) -> io::Result<()> {
unsafe {
let mut typ = self.buf_type as u32;
v4l2::ioctl(
self.handle.fd(),
v4l2::vidioc::VIDIOC_STREAMON,
&mut typ as *mut _ as *mut std::os::raw::c_void,
)?;
}
self.active = true;
Ok(())
}
fn stop(&mut self) -> io::Result<()> {
unsafe {
let mut typ = self.buf_type as u32;
v4l2::ioctl(
self.handle.fd(),
v4l2::vidioc::VIDIOC_STREAMOFF,
&mut typ as *mut _ as *mut std::os::raw::c_void,
)?;
}
self.active = false;
Ok(())
}
}
impl<'a> CaptureStream<'a> for Stream {
fn queue(&mut self, index: usize) -> io::Result<()> {
let buf = &mut self.arena.bufs[index];
let mut v4l2_buf = v4l2_buffer {
index: index as u32,
m: v4l2_buffer__bindgen_ty_1 {
userptr: buf.as_ptr() as std::os::raw::c_ulong,
},
length: buf.len() as u32,
..self.buffer_desc()
};
unsafe {
v4l2::ioctl(
self.handle.fd(),
v4l2::vidioc::VIDIOC_QBUF,
&mut v4l2_buf as *mut _ as *mut std::os::raw::c_void,
)?;
}
Ok(())
}
fn dequeue(&mut self) -> io::Result<usize> {
let mut v4l2_buf = self.buffer_desc();
if self.handle.poll(libc::POLLIN, self.timeout.unwrap_or(-1))? == 0 {
return Err(io::Error::new(io::ErrorKind::TimedOut, "VIDIOC_DQBUF"));
}
unsafe {
v4l2::ioctl(
self.handle.fd(),
v4l2::vidioc::VIDIOC_DQBUF,
&mut v4l2_buf as *mut _ as *mut std::os::raw::c_void,
)?;
}
self.arena_index = v4l2_buf.index as usize;
self.buf_meta[self.arena_index] = Metadata {
bytesused: v4l2_buf.bytesused,
flags: v4l2_buf.flags.into(),
field: v4l2_buf.field,
timestamp: v4l2_buf.timestamp.into(),
sequence: v4l2_buf.sequence,
};
Ok(self.arena_index)
}
fn next(&'a mut self) -> io::Result<(&Self::Item, &Metadata, bool)> {
if !self.active {
for index in 0..self.arena.bufs.len() {
self.queue(index)?;
}
self.start()?;
} else {
self.queue(self.arena_index)?;
}
self.arena_index = self.dequeue()?;
let bytes = &mut self.arena.bufs[self.arena_index];
let meta = &self.buf_meta[self.arena_index];
Ok((bytes, meta, false))
}
}