use std::sync::{LazyLock, Mutex};
use gst::glib;
use gst::prelude::*;
use gst::subclass::prelude::*;
use atomic_refcell::AtomicRefCell;
use crate::st2038anc_utils::convert_to_st2038_buffer;
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
enum Format {
Cea608,
Cea708,
}
#[derive(Default)]
struct State {
format: Option<Format>,
}
#[derive(Clone)]
struct Settings {
c_not_y_channel: bool,
line_number: u16,
horizontal_offset: u16,
}
impl Default for Settings {
fn default() -> Self {
Settings {
c_not_y_channel: false,
line_number: 9,
horizontal_offset: 0,
}
}
}
pub struct CcToSt2038Anc {
sinkpad: gst::Pad,
srcpad: gst::Pad,
state: AtomicRefCell<State>,
settings: Mutex<Settings>,
}
static CAT: LazyLock<gst::DebugCategory> = LazyLock::new(|| {
gst::DebugCategory::new(
"cctost2038anc",
gst::DebugColorFlags::empty(),
Some("Closed Caption to ST-2038 ANC Element"),
)
});
impl CcToSt2038Anc {
fn sink_chain(
&self,
pad: &gst::Pad,
buffer: gst::Buffer,
) -> Result<gst::FlowSuccess, gst::FlowError> {
gst::trace!(CAT, obj = pad, "Handling buffer {:?}", buffer);
let state = self.state.borrow_mut();
let settings = self.settings.lock().unwrap().clone();
let map = buffer.map_readable().map_err(|_| {
gst::error!(CAT, obj = pad, "Can't map buffer readable");
gst::FlowError::Error
})?;
let (did, sdid) = match state.format {
Some(Format::Cea608) => (0x61, 0x02),
Some(Format::Cea708) => (0x61, 0x01),
None => {
gst::error!(CAT, imp = self, "No caps set");
return Err(gst::FlowError::NotNegotiated);
}
};
let mut outbuf = match convert_to_st2038_buffer(
settings.c_not_y_channel,
settings.line_number,
settings.horizontal_offset,
did,
sdid,
&map,
) {
Ok(outbuf) => outbuf,
Err(err) => {
gst::error!(
CAT,
imp = self,
"Can't convert Closed Caption buffer: {err}"
);
return Err(gst::FlowError::Error);
}
};
drop(map);
{
let outbuf = outbuf.get_mut().unwrap();
let _ = buffer.copy_into(outbuf, gst::BUFFER_COPY_METADATA, ..);
}
drop(state);
self.srcpad.push(outbuf)
}
#[allow(clippy::single_match)]
fn sink_event(&self, pad: &gst::Pad, event: gst::Event) -> bool {
use gst::EventView;
gst::log!(CAT, obj = pad, "Handling event {:?}", event);
match event.view() {
EventView::Caps(ev) => {
let caps = ev.caps();
let s = caps.structure(0).unwrap();
let format = match s.name().as_str() {
"closedcaption/x-cea-608" => Format::Cea608,
"closedcaption/x-cea-708" => Format::Cea708,
_ => {
gst::error!(CAT, imp = self, "Unsupported caps {caps:?}");
return false;
}
};
gst::debug!(CAT, imp = self, "Configuring format {format:?}");
let mut state = self.state.borrow_mut();
state.format = Some(format);
drop(state);
let templ_caps = self.srcpad.pad_template_caps();
let mut peer_caps = self.srcpad.peer_query_caps(Some(&templ_caps));
gst::debug!(CAT, obj = pad, "Downstream caps {peer_caps:?}");
let alignment = if peer_caps.is_empty() {
"frame"
} else {
peer_caps.fixate();
peer_caps
.structure(0)
.and_then(|s| s.get::<&str>("alignment").ok())
.unwrap_or("frame")
};
let src_caps = {
let mut caps = self.srcpad.pad_template_caps();
let caps_ref = caps.make_mut();
if let Ok(framerate) = s.get::<gst::Fraction>("framerate") {
caps_ref.set("framerate", framerate);
}
caps_ref.set("alignment", alignment);
caps
};
return self.srcpad.push_event(
gst::event::Caps::builder(&src_caps)
.seqnum(ev.seqnum())
.build(),
);
}
_ => (),
}
gst::Pad::event_default(pad, Some(&*self.obj()), event)
}
}
#[glib::object_subclass]
impl ObjectSubclass for CcToSt2038Anc {
const NAME: &'static str = "GstCcToSt2038Anc";
type Type = super::CcToSt2038Anc;
type ParentType = gst::Element;
fn with_class(klass: &Self::Class) -> Self {
let templ = klass.pad_template("sink").unwrap();
let sinkpad = gst::Pad::builder_from_template(&templ)
.chain_function(|pad, parent, buffer| {
CcToSt2038Anc::catch_panic_pad_function(
parent,
|| Err(gst::FlowError::Error),
|this| this.sink_chain(pad, buffer),
)
})
.event_function(|pad, parent, event| {
CcToSt2038Anc::catch_panic_pad_function(
parent,
|| false,
|this| this.sink_event(pad, event),
)
})
.flags(gst::PadFlags::FIXED_CAPS)
.build();
let templ = klass.pad_template("src").unwrap();
let srcpad = gst::Pad::builder_from_template(&templ)
.flags(gst::PadFlags::FIXED_CAPS)
.build();
Self {
sinkpad,
srcpad,
state: AtomicRefCell::new(State::default()),
settings: Mutex::new(Settings::default()),
}
}
}
impl ObjectImpl for CcToSt2038Anc {
fn properties() -> &'static [glib::ParamSpec] {
static PROPERTIES: LazyLock<Vec<glib::ParamSpec>> = LazyLock::new(|| {
vec![
glib::ParamSpecBoolean::builder("c-not-y-channel")
.nick("C Not Y Channel")
.blurb("Set the c_not_y_channel flag in the output")
.default_value(Settings::default().c_not_y_channel)
.mutable_playing()
.build(),
glib::ParamSpecUInt::builder("line-number")
.nick("Line Number")
.blurb("Line Number of the output")
.default_value(Settings::default().line_number as u32)
.maximum(2047)
.mutable_playing()
.build(),
glib::ParamSpecUInt::builder("horizontal-offset")
.nick("Horizontal Offset")
.blurb("Horizontal offset of the output")
.default_value(Settings::default().horizontal_offset as u32)
.maximum(4095)
.mutable_playing()
.build(),
]
});
PROPERTIES.as_ref()
}
fn set_property(&self, _id: usize, value: &glib::Value, pspec: &glib::ParamSpec) {
match pspec.name() {
"c-not-y-channel" => {
let mut settings = self.settings.lock().unwrap();
settings.c_not_y_channel = value.get().expect("type checked upstream");
}
"line-number" => {
let mut settings = self.settings.lock().unwrap();
settings.line_number = value.get::<u32>().expect("type checked upstream") as u16;
}
"horizontal-offset" => {
let mut settings = self.settings.lock().unwrap();
settings.horizontal_offset =
value.get::<u32>().expect("type checked upstream") as u16;
}
_ => unimplemented!(),
}
}
fn property(&self, _id: usize, pspec: &glib::ParamSpec) -> glib::Value {
match pspec.name() {
"c-not-y-channel" => {
let settings = self.settings.lock().unwrap();
settings.c_not_y_channel.to_value()
}
"line-number" => {
let settings = self.settings.lock().unwrap();
(settings.line_number as u32).to_value()
}
"horizontal-offset" => {
let settings = self.settings.lock().unwrap();
(settings.horizontal_offset as u32).to_value()
}
_ => unimplemented!(),
}
}
fn constructed(&self) {
self.parent_constructed();
let obj = self.obj();
obj.add_pad(&self.sinkpad).unwrap();
obj.add_pad(&self.srcpad).unwrap();
}
}
impl GstObjectImpl for CcToSt2038Anc {}
impl ElementImpl for CcToSt2038Anc {
fn metadata() -> Option<&'static gst::subclass::ElementMetadata> {
static ELEMENT_METADATA: LazyLock<gst::subclass::ElementMetadata> = LazyLock::new(|| {
gst::subclass::ElementMetadata::new(
"CC to ST-2038 ANC",
"Generic",
"Converts Closed Captions to ST-2038 ANC",
"Sebastian Dröge <sebastian@centricular.com>",
)
});
Some(&*ELEMENT_METADATA)
}
fn pad_templates() -> &'static [gst::PadTemplate] {
static PAD_TEMPLATES: LazyLock<Vec<gst::PadTemplate>> = LazyLock::new(|| {
let caps_aligned = gst::Caps::builder("meta/x-st-2038")
.field("alignment", gst::List::new(["frame", "line", "packet"]))
.build();
let src_pad_template = gst::PadTemplate::new(
"src",
gst::PadDirection::Src,
gst::PadPresence::Sometimes,
&caps_aligned,
)
.unwrap();
let sink_pad_template = gst::PadTemplate::new(
"sink",
gst::PadDirection::Sink,
gst::PadPresence::Always,
&[
gst::Structure::builder("closedcaption/x-cea-608")
.field("format", "s334-1a")
.build(),
gst::Structure::builder("closedcaption/x-cea-708")
.field("format", "cdp")
.build(),
]
.into_iter()
.collect::<gst::Caps>(),
)
.unwrap();
vec![src_pad_template, sink_pad_template]
});
PAD_TEMPLATES.as_ref()
}
#[allow(clippy::single_match)]
fn change_state(
&self,
transition: gst::StateChange,
) -> Result<gst::StateChangeSuccess, gst::StateChangeError> {
gst::trace!(CAT, imp = self, "Changing state {:?}", transition);
match transition {
gst::StateChange::ReadyToPaused => {
*self.state.borrow_mut() = State::default();
}
_ => (),
}
let ret = self.parent_change_state(transition)?;
match transition {
gst::StateChange::PausedToReady => {
*self.state.borrow_mut() = State::default();
}
_ => (),
}
Ok(ret)
}
}