feat: Very basic tally parsing
This commit is contained in:
parent
2ee3d71e78
commit
9d872eecc9
|
@ -93,6 +93,7 @@ dependencies = [
|
|||
"clap",
|
||||
"color-eyre",
|
||||
"env_logger",
|
||||
"log",
|
||||
"tokio",
|
||||
]
|
||||
|
||||
|
|
|
@ -7,13 +7,3 @@ pub struct AtemOptions {
|
|||
disable_multi_threaded: bool,
|
||||
child_process_timeout: Option<u64>,
|
||||
}
|
||||
|
||||
pub enum AtemEvents {
|
||||
Error(String),
|
||||
Info(String),
|
||||
Debug(String),
|
||||
Connected,
|
||||
Disconnected,
|
||||
StateChanged(Box<(AtemState, Vec<String>)>),
|
||||
ReceivedCommands(Vec<Box<dyn DeserializedCommand>>),
|
||||
}
|
||||
|
|
|
@ -1,13 +1,17 @@
|
|||
use std::{
|
||||
io,
|
||||
net::SocketAddr,
|
||||
sync::Arc,
|
||||
time::{Duration, SystemTime},
|
||||
};
|
||||
|
||||
use log::debug;
|
||||
use tokio::net::UdpSocket;
|
||||
|
||||
use crate::atem_lib::atem_util;
|
||||
use crate::{
|
||||
atem_lib::atem_util,
|
||||
commands::{command_base::DeserializedCommand, parse_commands::deserialize_commands},
|
||||
};
|
||||
|
||||
const IN_FLIGHT_TIMEOUT: u64 = 60;
|
||||
const CONNECTION_TIMEOUT: u64 = 5000;
|
||||
|
@ -21,6 +25,17 @@ const MAX_PACKET_PER_ACK: u16 = 16;
|
|||
const MAX_PACKET_RECEIVE_SIZE: usize = 65535;
|
||||
const ACK_PACKET_LENGTH: u16 = 12;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub enum AtemEvent {
|
||||
Error(String),
|
||||
Info(String),
|
||||
Debug(String),
|
||||
Connected,
|
||||
Disconnected,
|
||||
ReceivedCommand(Arc<dyn DeserializedCommand>),
|
||||
AckedCommand(AckedPacket),
|
||||
}
|
||||
|
||||
#[derive(PartialEq, Clone)]
|
||||
enum ConnectionState {
|
||||
Closed,
|
||||
|
@ -69,6 +84,7 @@ struct InFlightPacket {
|
|||
pub resent: u16,
|
||||
}
|
||||
|
||||
#[derive(Clone)]
|
||||
struct AckedPacket {
|
||||
packet_id: u16,
|
||||
tracking_id: u64,
|
||||
|
@ -97,6 +113,8 @@ pub struct AtemSocketInner {
|
|||
in_flight: Vec<InFlightPacket>,
|
||||
ack_timer: Option<SystemTime>,
|
||||
received_without_ack: u16,
|
||||
|
||||
atem_event_tx: tokio::sync::broadcast::Sender<AtemEvent>,
|
||||
}
|
||||
|
||||
enum AtemSocketReceiveError {
|
||||
|
@ -114,6 +132,8 @@ enum AtemSocketWriteError {
|
|||
|
||||
impl AtemSocketInner {
|
||||
pub fn new() -> Self {
|
||||
let (atem_event_tx, _) = tokio::sync::broadcast::channel(100);
|
||||
|
||||
AtemSocketInner {
|
||||
connection_state: ConnectionState::Closed,
|
||||
reconnect_timer: None,
|
||||
|
@ -131,6 +151,8 @@ impl AtemSocketInner {
|
|||
in_flight: vec![],
|
||||
ack_timer: None,
|
||||
received_without_ack: 0,
|
||||
|
||||
atem_event_tx,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -214,6 +236,10 @@ impl AtemSocketInner {
|
|||
})
|
||||
}
|
||||
|
||||
pub fn subscribe_to_events(&self) -> tokio::sync::broadcast::Receiver<AtemEvent> {
|
||||
self.atem_event_tx.subscribe()
|
||||
}
|
||||
|
||||
async fn restart_connection(&mut self) {
|
||||
self.disconnect();
|
||||
self.connect(self.address.clone(), self.port).await.ok();
|
||||
|
@ -320,7 +346,7 @@ impl AtemSocketInner {
|
|||
self.send_or_queue_ack().await;
|
||||
|
||||
if length > 12 {
|
||||
self.on_command_received(&packet[12..], remote_packet_id);
|
||||
self.on_command_received(&packet[12..]);
|
||||
}
|
||||
} else if self
|
||||
.is_packet_covered_by_ack(self.last_received_packed_id, remote_packet_id)
|
||||
|
@ -438,16 +464,18 @@ impl AtemSocketInner {
|
|||
}
|
||||
}
|
||||
|
||||
fn on_command_received(&mut self, payload: &[u8], packet_id: u16) {
|
||||
// TODO: Emit some event
|
||||
fn on_command_received(&mut self, payload: &[u8]) {
|
||||
let commands = deserialize_commands(payload);
|
||||
}
|
||||
|
||||
fn on_command_acknowledged(&mut self, ids: Vec<AckedPacket>) {
|
||||
// TODO: Emit some event
|
||||
fn on_command_acknowledged(&mut self, packets: Vec<AckedPacket>) {
|
||||
for ack in packets {
|
||||
let _ = self.atem_event_tx.send(AtemEvent::AckedCommand(ack));
|
||||
}
|
||||
}
|
||||
|
||||
fn on_disconnect(&mut self) {
|
||||
// TODO: Emit some event
|
||||
let _ = self.atem_event_tx.send(AtemEvent::Disconnected);
|
||||
}
|
||||
|
||||
fn start_timers(&mut self) {
|
||||
|
|
|
@ -1,8 +1,11 @@
|
|||
use std::collections::HashMap;
|
||||
use std::{collections::HashMap, sync::Arc};
|
||||
|
||||
use crate::{enums::ProtocolVersion, state::AtemState};
|
||||
|
||||
pub trait DeserializedCommand {
|
||||
pub trait DeserializedCommand: Send + Sync {
|
||||
fn deserialize(buffer: &[u8]) -> Self
|
||||
where
|
||||
Self: Sized;
|
||||
fn apply_to_state(&self, state: &mut AtemState) -> Vec<String>;
|
||||
}
|
||||
|
||||
|
|
|
@ -1,27 +1,3 @@
|
|||
use super::command_base::{BasicWritableCommand, SerializableCommand};
|
||||
|
||||
#[derive(new)]
|
||||
pub struct ProgramInput {
|
||||
mix_effect: u8,
|
||||
source: u16,
|
||||
}
|
||||
|
||||
impl SerializableCommand for ProgramInput {
|
||||
fn payload(&self, _version: crate::enums::ProtocolVersion) -> Vec<u8> {
|
||||
let mut buf = vec![0; 4];
|
||||
buf[..1].copy_from_slice(&self.mix_effect.to_be_bytes());
|
||||
buf[2..].copy_from_slice(&self.source.to_be_bytes());
|
||||
|
||||
buf
|
||||
}
|
||||
}
|
||||
|
||||
impl BasicWritableCommand for ProgramInput {
|
||||
fn get_raw_name(&self) -> &'static str {
|
||||
"CPgI"
|
||||
}
|
||||
|
||||
fn get_minimum_version(&self) -> crate::enums::ProtocolVersion {
|
||||
crate::enums::ProtocolVersion::Unknown
|
||||
}
|
||||
}
|
||||
pub mod program_input;
|
||||
|
|
|
@ -0,0 +1,27 @@
|
|||
use crate::commands::command_base::{BasicWritableCommand, SerializableCommand};
|
||||
|
||||
#[derive(new)]
|
||||
pub struct ProgramInput {
|
||||
mix_effect: u8,
|
||||
source: u16,
|
||||
}
|
||||
|
||||
impl SerializableCommand for ProgramInput {
|
||||
fn payload(&self, _version: crate::enums::ProtocolVersion) -> Vec<u8> {
|
||||
let mut buf = vec![0; 4];
|
||||
buf[..1].copy_from_slice(&self.mix_effect.to_be_bytes());
|
||||
buf[2..].copy_from_slice(&self.source.to_be_bytes());
|
||||
|
||||
buf
|
||||
}
|
||||
}
|
||||
|
||||
impl BasicWritableCommand for ProgramInput {
|
||||
fn get_raw_name(&self) -> &'static str {
|
||||
"CPgI"
|
||||
}
|
||||
|
||||
fn get_minimum_version(&self) -> crate::enums::ProtocolVersion {
|
||||
crate::enums::ProtocolVersion::Unknown
|
||||
}
|
||||
}
|
|
@ -1,2 +1,4 @@
|
|||
pub mod command_base;
|
||||
pub mod mix_effects;
|
||||
pub mod parse_commands;
|
||||
pub mod tally_by_source;
|
||||
|
|
|
@ -0,0 +1,43 @@
|
|||
use std::sync::Arc;
|
||||
|
||||
use crate::commands::tally_by_source::TallyBySource;
|
||||
|
||||
use super::command_base::DeserializedCommand;
|
||||
|
||||
pub fn deserialize_commands(payload: &[u8]) -> Vec<Arc<dyn DeserializedCommand>> {
|
||||
let mut parsed_commands = vec![];
|
||||
let mut head = 0;
|
||||
|
||||
while payload.len() > head + 8 {
|
||||
// log::debug!("Head at {} out of {}", head, payload.len());
|
||||
let length = u16::from_be_bytes([payload[head], payload[head + 1]]) as usize;
|
||||
let Ok(name) = String::from_utf8(payload[(head + 4)..(head + 8)].to_vec()) else {
|
||||
break;
|
||||
};
|
||||
|
||||
if length < 8 {
|
||||
break;
|
||||
}
|
||||
|
||||
log::debug!("Received command {} with length {}", name, length);
|
||||
|
||||
match name.as_str() {
|
||||
"TlSr" => {
|
||||
let tally = TallyBySource::deserialize(&payload[head + 8..head + length]);
|
||||
for (source, state) in tally.sources {
|
||||
log::info!(
|
||||
"Source {} Program: {}, Preview: {}",
|
||||
source,
|
||||
state.program,
|
||||
state.preview
|
||||
)
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
|
||||
head += length;
|
||||
}
|
||||
|
||||
parsed_commands
|
||||
}
|
|
@ -0,0 +1,56 @@
|
|||
use std::collections::HashMap;
|
||||
|
||||
use crate::commands::command_base::{BasicWritableCommand, SerializableCommand};
|
||||
|
||||
use super::command_base::{DeserializableCommand, DeserializedCommand};
|
||||
|
||||
pub struct TallySource {
|
||||
pub program: bool,
|
||||
pub preview: bool,
|
||||
}
|
||||
|
||||
#[derive(new)]
|
||||
pub struct TallyBySource {
|
||||
pub sources: HashMap<u16, TallySource>,
|
||||
}
|
||||
|
||||
impl DeserializableCommand for TallyBySource {
|
||||
fn get_raw_name(&self) -> &'static str {
|
||||
"TlSr"
|
||||
}
|
||||
|
||||
fn get_minimum_version(&self) -> crate::enums::ProtocolVersion {
|
||||
crate::enums::ProtocolVersion::Unknown
|
||||
}
|
||||
}
|
||||
|
||||
impl DeserializedCommand for TallyBySource {
|
||||
fn deserialize(buffer: &[u8]) -> Self {
|
||||
let source_count = u16::from_be_bytes([buffer[0], buffer[1]]) as usize;
|
||||
|
||||
log::debug!("{:?}", buffer);
|
||||
log::debug!("Source count: {}", source_count);
|
||||
|
||||
let mut sources = HashMap::new();
|
||||
for i in 0..source_count {
|
||||
let source_byte_offset = 2 + (i * 3);
|
||||
let source =
|
||||
u16::from_be_bytes([buffer[source_byte_offset], buffer[source_byte_offset + 1]]);
|
||||
let value_byte_offset = 4 + (i * 3);
|
||||
let value = u8::from_be_bytes([buffer[value_byte_offset]]);
|
||||
sources.insert(
|
||||
source,
|
||||
TallySource {
|
||||
program: (value & 0x01) > 0,
|
||||
preview: (value & 0x02) > 0,
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
Self { sources }
|
||||
}
|
||||
|
||||
fn apply_to_state(&self, state: &mut crate::state::AtemState) -> Vec<String> {
|
||||
todo!()
|
||||
}
|
||||
}
|
|
@ -10,4 +10,5 @@ atem-connection-rs = { path = "../atem-connection-rs" }
|
|||
clap = { version = "4.4.18", features = ["derive"] }
|
||||
color-eyre = "0.5.11"
|
||||
env_logger = "0.9.0"
|
||||
log = "0.4.14"
|
||||
tokio = "1.14.0"
|
||||
|
|
|
@ -4,7 +4,7 @@ use atem_connection_rs::{
|
|||
atem_lib::atem_socket::AtemSocket,
|
||||
commands::{
|
||||
command_base::{BasicWritableCommand, SerializableCommand},
|
||||
mix_effects::ProgramInput,
|
||||
mix_effects::program_input::ProgramInput,
|
||||
},
|
||||
};
|
||||
|
||||
|
@ -35,6 +35,7 @@ async fn main() {
|
|||
|
||||
let mut tracking_id = 0;
|
||||
loop {
|
||||
tracking_id += 1;
|
||||
sleep(Duration::from_millis(5000)).await;
|
||||
atem.send_command(
|
||||
&switch_to_source_1.payload(atem_connection_rs::enums::ProtocolVersion::Unknown),
|
||||
|
|
Loading…
Reference in New Issue