2018-12-28 22:00:28 +00:00
|
|
|
// Copyright (C) 2018 Sebastian Dröge <sebastian@centricular.com>
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Library General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Library General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Library General Public
|
|
|
|
// License along with this library; if not, write to the
|
|
|
|
// Free Software Foundation, Inc., 51 Franklin Street, Suite 500,
|
|
|
|
// Boston, MA 02110-1335, USA.
|
|
|
|
|
2018-12-31 09:35:13 +00:00
|
|
|
use glib::prelude::*;
|
|
|
|
|
2020-11-22 17:21:45 +00:00
|
|
|
use pretty_assertions::assert_eq;
|
|
|
|
|
2018-12-28 22:00:28 +00:00
|
|
|
fn init() {
|
2019-07-11 13:41:42 +00:00
|
|
|
use std::sync::Once;
|
|
|
|
static INIT: Once = Once::new();
|
2018-12-28 22:00:28 +00:00
|
|
|
|
|
|
|
INIT.call_once(|| {
|
|
|
|
gst::init().unwrap();
|
2019-01-21 20:13:07 +00:00
|
|
|
gstrsclosedcaption::plugin_register_static().expect("mccenc test");
|
2018-12-28 22:00:28 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Encode a single CDP packet and compare the output
|
|
|
|
#[test]
|
|
|
|
fn test_encode() {
|
|
|
|
init();
|
|
|
|
|
|
|
|
let input = [
|
|
|
|
0x96, 0x69, 0x52, 0x4f, 0x67, 0x00, 0x00, 0x72, 0xf4, 0xfc, 0x80, 0x80, 0xfd, 0x80, 0x80,
|
|
|
|
0xff, 0x02, 0x22, 0xfe, 0x8c, 0xff, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00,
|
|
|
|
0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00,
|
|
|
|
0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00,
|
|
|
|
0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0xfa, 0x00, 0x00, 0x73, 0x91, 0x81, 0x65, 0x6e, 0x67,
|
|
|
|
0x81, 0x7f, 0xff, 0x74, 0x00, 0x00, 0x1c,
|
|
|
|
];
|
|
|
|
|
|
|
|
let expected_output = b"File Format=MacCaption_MCC V1.0\r\n\
|
|
|
|
\r\n\
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////\r\n\
|
|
|
|
// Computer Prompting and Captioning Company\r\n\
|
|
|
|
// Ancillary Data Packet Transfer File\r\n\
|
|
|
|
//\r\n\
|
|
|
|
// Permission to generate this format is granted provided that\r\n\
|
|
|
|
// 1. This ANC Transfer file format is used on an as-is basis and no warranty is given, and\r\n\
|
|
|
|
// 2. This entire descriptive information text is included in a generated .mcc file.\r\n\
|
|
|
|
//\r\n\
|
|
|
|
// General file format:\r\n\
|
|
|
|
// HH:MM:SS:FF(tab)[Hexadecimal ANC data in groups of 2 characters]\r\n\
|
|
|
|
// Hexadecimal data starts with the Ancillary Data Packet DID (Data ID defined in S291M)\r\n\
|
|
|
|
// and concludes with the Check Sum following the User Data Words.\r\n\
|
|
|
|
// Each time code line must contain at most one complete ancillary data packet.\r\n\
|
|
|
|
// To transfer additional ANC Data successive lines may contain identical time code.\r\n\
|
|
|
|
// Time Code Rate=[24, 25, 30, 30DF, 50, 60]\r\n\
|
|
|
|
//\r\n\
|
|
|
|
// ANC data bytes may be represented by one ASCII character according to the following schema:\r\n\
|
|
|
|
// G FAh 00h 00h\r\n\
|
|
|
|
// H 2 x (FAh 00h 00h)\r\n\
|
|
|
|
// I 3 x (FAh 00h 00h)\r\n\
|
|
|
|
// J 4 x (FAh 00h 00h)\r\n\
|
|
|
|
// K 5 x (FAh 00h 00h)\r\n\
|
|
|
|
// L 6 x (FAh 00h 00h)\r\n\
|
|
|
|
// M 7 x (FAh 00h 00h)\r\n\
|
|
|
|
// N 8 x (FAh 00h 00h)\r\n\
|
|
|
|
// O 9 x (FAh 00h 00h)\r\n\
|
|
|
|
// P FBh 80h 80h\r\n\
|
|
|
|
// Q FCh 80h 80h\r\n\
|
|
|
|
// R FDh 80h 80h\r\n\
|
|
|
|
// S 96h 69h\r\n\
|
|
|
|
// T 61h 01h\r\n\
|
|
|
|
// U E1h 00h 00h 00h\r\n\
|
|
|
|
// Z 00h\r\n\
|
|
|
|
//\r\n\
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////\r\n\
|
|
|
|
\r\n\
|
|
|
|
UUID=14720C04-857D-40E2-86FC-F080DE44CE74\r\n\
|
2019-07-07 10:36:25 +00:00
|
|
|
Creation Program=GStreamer MCC Encoder 0.6.0\r\n\
|
2018-12-28 22:00:28 +00:00
|
|
|
Creation Date=Thursday, December 27, 2018\r\n\
|
|
|
|
Creation Time=17:34:47\r\n\
|
|
|
|
Time Code Rate=30DF\r\n\
|
|
|
|
\r\n\
|
|
|
|
11:12:13;14 T52S524F67ZZ72F4QRFF0222FE8CFFOM739181656E67817FFF74ZZ1CZ\r\n";
|
|
|
|
|
|
|
|
let mut h = gst_check::Harness::new("mccenc");
|
|
|
|
{
|
|
|
|
let enc = h.get_element().expect("could not create encoder");
|
|
|
|
enc.set_property("uuid", &"14720C04-857D-40E2-86FC-F080DE44CE74")
|
|
|
|
.unwrap();
|
|
|
|
enc.set_property(
|
|
|
|
"creation-date",
|
|
|
|
&glib::DateTime::new_utc(2018, 12, 27, 17, 34, 47.0),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
h.set_src_caps_str(
|
|
|
|
"closedcaption/x-cea-708, format=(string)cdp, framerate=(fraction)30000/1001",
|
|
|
|
);
|
|
|
|
|
2019-02-05 16:48:50 +00:00
|
|
|
let tc = gst_video::ValidVideoTimeCode::new(
|
|
|
|
gst::Fraction::new(30000, 1001),
|
|
|
|
None,
|
|
|
|
gst_video::VideoTimeCodeFlags::DROP_FRAME,
|
|
|
|
11,
|
|
|
|
12,
|
|
|
|
13,
|
|
|
|
14,
|
|
|
|
0,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
2018-12-28 22:00:28 +00:00
|
|
|
let buf = {
|
|
|
|
let mut buf = gst::Buffer::from_mut_slice(Vec::from(&input[..]));
|
|
|
|
let buf_ref = buf.get_mut().unwrap();
|
|
|
|
gst_video::VideoTimeCodeMeta::add(buf_ref, &tc);
|
2019-02-05 16:48:50 +00:00
|
|
|
buf_ref.set_pts(gst::ClockTime::from_seconds(0));
|
2018-12-28 22:00:28 +00:00
|
|
|
buf
|
|
|
|
};
|
|
|
|
|
|
|
|
assert_eq!(h.push(buf), Ok(gst::FlowSuccess::Ok));
|
2020-06-24 22:33:57 +00:00
|
|
|
h.push_event(gst::event::Eos::new());
|
2018-12-28 22:00:28 +00:00
|
|
|
|
|
|
|
let buf = h.pull().expect("Couldn't pull buffer");
|
2019-02-05 16:48:50 +00:00
|
|
|
let timecode = buf
|
|
|
|
.get_meta::<gst_video::VideoTimeCodeMeta>()
|
|
|
|
.expect("No timecode for buffer")
|
|
|
|
.get_tc();
|
|
|
|
assert_eq!(timecode, tc);
|
|
|
|
|
|
|
|
let pts = buf.get_pts();
|
|
|
|
assert_eq!(pts, gst::ClockTime::from_seconds(0));
|
|
|
|
|
2018-12-28 22:00:28 +00:00
|
|
|
let map = buf.map_readable().expect("Couldn't map buffer readable");
|
|
|
|
assert_eq!(
|
|
|
|
std::str::from_utf8(map.as_ref()),
|
|
|
|
std::str::from_utf8(expected_output.as_ref())
|
|
|
|
);
|
|
|
|
}
|