use bitflags::bitflags;
use std::convert::{TryFrom, TryInto};
use std::{ffi, fmt, mem, str};
use crate::v4l_sys::*;
#[allow(clippy::unreadable_literal)]
#[rustfmt::skip]
#[repr(u32)]
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
pub enum Type {
Integer = 1,
Boolean = 2,
Menu = 3,
Button = 4,
Integer64 = 5,
CtrlClass = 6,
String = 7,
Bitmask = 8,
IntegerMenu = 9,
U8 = 0x0100,
U16 = 0x0101,
U32 = 0x0102,
Area = 0x0106,
}
impl TryFrom<u32> for Type {
type Error = ();
fn try_from(repr: u32) -> Result<Self, Self::Error> {
match repr {
1 => Ok(Type::Integer),
2 => Ok(Type::Boolean),
3 => Ok(Type::Menu),
4 => Ok(Type::Button),
5 => Ok(Type::Integer64),
6 => Ok(Type::CtrlClass),
7 => Ok(Type::String),
8 => Ok(Type::Bitmask),
9 => Ok(Type::IntegerMenu),
0x0100 => Ok(Type::U8),
0x0101 => Ok(Type::U16),
0x0102 => Ok(Type::U32),
0x0106 => Ok(Type::Area),
_ => Err(()),
}
}
}
impl From<Type> for u32 {
fn from(t: Type) -> Self {
t as Self
}
}
impl fmt::Display for Type {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Debug::fmt(self, f)
}
}
bitflags! {
#[allow(clippy::unreadable_literal)]
pub struct Flags: u32 {
const DISABLED = 0x0001;
const GRABBED = 0x0002;
const READ_ONLY = 0x0004;
const UPDATE = 0x0008;
const INACTIVE = 0x0010;
const SLIDER = 0x0020;
const WRITE_ONLY = 0x0040;
const VOLATILE = 0x0080;
const HAS_PAYLOAD = 0x0100;
const EXECUTE_ON_WRITE = 0x0200;
const MODIFY_LAYOUT = 0x0400;
const NEXT_CTRL = 0x80000000;
const NEXT_COMPOUND = 0x40000000;
}
}
impl From<u32> for Flags {
fn from(flags: u32) -> Self {
Self::from_bits_truncate(flags)
}
}
impl From<Flags> for u32 {
fn from(flags: Flags) -> Self {
flags.bits()
}
}
impl fmt::Display for Flags {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Debug::fmt(self, f)
}
}
#[derive(Debug)]
pub enum MenuItem {
Name(String),
Value(i64),
}
impl fmt::Display for MenuItem {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
MenuItem::Name(name) => {
write!(f, "{}", name)?;
}
MenuItem::Value(value) => {
write!(f, "{}", value)?;
}
}
Ok(())
}
}
impl TryFrom<(Type, v4l2_querymenu)> for MenuItem {
type Error = ();
fn try_from(item: (Type, v4l2_querymenu)) -> Result<Self, Self::Error> {
unsafe {
match item.0 {
Type::Menu => Ok(MenuItem::Name(
str::from_utf8(&item.1.__bindgen_anon_1.name)
.unwrap()
.trim_matches(char::from(0))
.to_string(),
)),
Type::IntegerMenu => Ok(MenuItem::Value(item.1.__bindgen_anon_1.value)),
_ => Err(()),
}
}
}
}
#[derive(Debug)]
pub struct Description {
pub id: u32,
pub typ: Type,
pub name: String,
pub minimum: i64,
pub maximum: i64,
pub step: u64,
pub default: i64,
pub flags: Flags,
pub items: Option<Vec<(u32, MenuItem)>>,
}
impl From<v4l2_query_ext_ctrl> for Description {
fn from(ctrl: v4l2_query_ext_ctrl) -> Self {
Self {
id: ctrl.id,
typ: Type::try_from(ctrl.type_).unwrap(),
name: unsafe { ffi::CStr::from_ptr(ctrl.name.as_ptr()) }
.to_str()
.unwrap()
.to_string(),
minimum: ctrl.minimum,
maximum: ctrl.maximum,
step: ctrl.step,
default: ctrl.default_value,
flags: Flags::from(ctrl.flags),
items: None,
}
}
}
impl fmt::Display for Description {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
writeln!(f, "ID : {}", self.id)?;
writeln!(f, "Type : {}", self.typ)?;
writeln!(f, "Name : {}", self.name)?;
writeln!(f, "Minimum : {}", self.minimum)?;
writeln!(f, "Maximum : {}", self.maximum)?;
writeln!(f, "Step : {}", self.step)?;
writeln!(f, "Default : {}", self.default)?;
writeln!(f, "Flags : {}", self.flags)?;
if let Some(items) = &self.items {
writeln!(f, "Menu ==>")?;
for item in items {
writeln!(f, " * {}", item.1)?;
}
}
Ok(())
}
}
#[derive(Debug)]
pub struct Control {
pub id: u32,
pub value: Value,
}
#[derive(Debug, PartialEq, Eq)]
pub enum Value {
None,
Integer(i64),
Boolean(bool),
String(String),
CompoundU8(Vec<u8>),
CompoundU16(Vec<u16>),
CompoundU32(Vec<u32>),
CompoundPtr(Vec<u8>),
}
impl TryInto<v4l2_control> for Control {
type Error = ();
fn try_into(self) -> Result<v4l2_control, Self::Error> {
unsafe {
let mut ctrl = v4l2_control {
id: self.id,
..mem::zeroed()
};
match self.value {
Value::None => Ok(ctrl),
Value::Integer(val) => {
ctrl.value = val as i32;
Ok(ctrl)
}
Value::Boolean(val) => {
ctrl.value = val as i32;
Ok(ctrl)
}
_ => Err(()),
}
}
}
}