use crate::event::{Action, Event, HostRequest, Notification};
use crate::resource::{
ApplicationInformation, ConditionalAccess, DateTime, Mmi, Resource, ResourceManager,
ResourceOut,
};
use crate::session::{SessionLayer, SessionOut};
use crate::transport::{Out as TransportOut, Transport};
use dvb_ci::resource::{ResourceId, DATE_TIME, RESOURCE_MANAGER};
pub struct CiStack {
transport: Transport,
session: SessionLayer,
provided: Vec<ResourceId>,
resources: Vec<Box<dyn Resource>>,
}
impl Default for CiStack {
fn default() -> Self {
Self::new()
}
}
impl CiStack {
#[must_use]
pub fn new() -> Self {
let provided = vec![RESOURCE_MANAGER, DATE_TIME];
Self {
transport: Transport::new(1),
session: SessionLayer::new(),
resources: vec![
Box::new(ResourceManager::new(provided.clone())),
Box::new(ApplicationInformation),
Box::new(ConditionalAccess),
Box::new(DateTime::new()),
Box::new(Mmi),
],
provided,
}
}
pub fn register(&mut self, resource: Box<dyn Resource>) -> &mut Self {
self.resources.push(resource);
self
}
fn handler_index(&self, resource: ResourceId) -> Option<usize> {
self.resources.iter().position(|r| r.id() == resource)
}
pub fn handle(&mut self, event: Event<'_>) -> Vec<Action> {
match event {
Event::Host(HostRequest::Init) => {
let mut actions = vec![Action::Reset, Action::QuerySlot];
let out = self.transport.init();
actions.extend(self.emit_transport(out));
actions
}
Event::Tick { elapsed } => {
let out = self.transport.tick(elapsed);
let mut actions = self.emit_transport(out);
for (session_nb, resource) in self.session.sessions() {
if let Some(i) = self.handler_index(resource) {
let out = self.resources[i].tick(elapsed);
actions.extend(self.process_resource_out(session_nb, out));
}
}
actions
}
Event::Readable(frame) => {
let out = self.transport.on_frame(frame);
self.emit_transport(out)
}
Event::Host(HostRequest::SendCaPmt(apdu)) => {
self.send_to_resource(dvb_ci::resource::CONDITIONAL_ACCESS_SUPPORT, apdu)
}
Event::Host(HostRequest::Shutdown) => Vec::new(),
}
}
fn send_to_resource(&mut self, resource: ResourceId, apdu: &[u8]) -> Vec<Action> {
let nb = (1u16..=u16::MAX).find(|&n| self.session.resource_of(n) == Some(resource));
match nb {
Some(nb) => {
let spdu = self.session.send_apdu(nb, apdu);
let out = self.transport.send_spdu(&spdu);
self.emit_transport(out)
}
None => vec![Action::Notify(Notification::Error {
detail: format!("no open session for resource {}", resource.name()),
})],
}
}
fn emit_transport(&mut self, out: TransportOut) -> Vec<Action> {
let mut actions = Vec::new();
for w in out.writes {
actions.push(Action::Write(w));
}
if let Some(after) = out.timer {
actions.push(Action::SetTimer { after });
}
if let Some(err) = out.error {
actions.push(Action::Notify(Notification::Error {
detail: err.to_string(),
}));
}
for spdu in out.spdus {
actions.extend(self.drive_session(&spdu));
}
actions
}
fn drive_session(&mut self, spdu: &[u8]) -> Vec<Action> {
let provided = self.provided.clone();
let SessionOut {
spdus,
apdus,
opened,
closed,
} = self.session.on_spdu(spdu, |r| provided.contains(&r));
let mut actions = Vec::new();
for s in spdus {
actions.extend(self.send_spdu_actions(&s));
}
for (session_nb, resource) in opened {
actions.push(Action::Notify(Notification::SessionOpened { resource }));
if let Some(i) = self.handler_index(resource) {
let out = self.resources[i].on_open();
actions.extend(self.process_resource_out(session_nb, out));
}
}
for session_nb in closed {
actions.push(Action::Notify(Notification::SessionClosed { session_nb }));
}
for (session_nb, apdu) in apdus {
if let Some(resource) = self.session.resource_of(session_nb) {
if let Some(i) = self.handler_index(resource) {
let out = self.resources[i].on_apdu(&apdu);
actions.extend(self.process_resource_out(session_nb, out));
}
}
}
actions
}
fn send_spdu_actions(&mut self, spdu: &[u8]) -> Vec<Action> {
let t = self.transport.send_spdu(spdu);
let mut actions = Vec::new();
for w in t.writes {
actions.push(Action::Write(w));
}
if let Some(after) = t.timer {
actions.push(Action::SetTimer { after });
}
actions
}
fn process_resource_out(&mut self, session_nb: u16, out: ResourceOut) -> Vec<Action> {
let mut actions = Vec::new();
for apdu in out.apdus {
let spdu = self.session.send_apdu(session_nb, &apdu);
actions.extend(self.send_spdu_actions(&spdu));
}
for note in out.notify {
actions.push(Action::Notify(note));
}
for resource in out.open {
let spdu = self.session.create_session(resource);
actions.extend(self.send_spdu_actions(&spdu));
}
actions
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::transport::DEFAULT_POLL_INTERVAL;
use dvb_ci::resource::RESOURCE_MANAGER;
use dvb_ci::spdu::{tags as spdu_tags, OpenSessionRequest};
use dvb_ci::tpdu::{tags as tpdu_tags, SbValue};
use dvb_common::Serialize;
fn ser<S: Serialize>(s: &S) -> Vec<u8> {
let mut b = vec![0u8; s.serialized_len()];
match s.serialize_into(&mut b) {
Ok(n) => b.truncate(n),
Err(_) => b.clear(),
}
b
}
fn r_data(tcid: u8, spdu: &[u8]) -> Vec<u8> {
let mut v = vec![tpdu_tags::DATA_LAST, (1 + spdu.len()) as u8, tcid];
v.extend_from_slice(spdu);
v.extend_from_slice(&[tpdu_tags::SB, 0x02, tcid, SbValue::new(false).0]);
v
}
#[test]
fn init_resets_and_opens_transport() {
let mut s = CiStack::new();
let a = s.handle(Event::Host(HostRequest::Init));
assert_eq!(a[0], Action::Reset);
assert_eq!(a[1], Action::QuerySlot);
assert!(matches!(&a[2], Action::Write(w) if w[0] == tpdu_tags::CREATE_T_C));
}
#[test]
fn full_pipeline_opens_a_session_for_a_provided_resource() {
let mut s = CiStack::new();
s.handle(Event::Host(HostRequest::Init));
s.handle(Event::Readable(&[tpdu_tags::C_T_C_REPLY, 0x01, 0x01]));
let osr = ser(&OpenSessionRequest {
resource: RESOURCE_MANAGER,
});
let actions = s.handle(Event::Readable(&r_data(1, &osr)));
assert!(actions.iter().any(|x| matches!(
x,
Action::Notify(Notification::SessionOpened {
resource
}) if *resource == RESOURCE_MANAGER
)));
let wrote_osr = actions.iter().any(|x| match x {
Action::Write(w) => w
.windows(1)
.any(|_| w.contains(&spdu_tags::OPEN_SESSION_RESPONSE)),
_ => false,
});
assert!(wrote_osr, "open_session_response must be sent down");
let nb = (1u16..16).find(|&n| s.session.resource_of(n).is_some());
assert!(nb.is_some());
}
#[test]
fn tick_drives_poll_when_active() {
let mut s = CiStack::new();
s.handle(Event::Host(HostRequest::Init));
s.handle(Event::Readable(&[tpdu_tags::C_T_C_REPLY, 0x01, 0x01]));
let a = s.handle(Event::Tick {
elapsed: DEFAULT_POLL_INTERVAL,
});
assert!(a
.iter()
.any(|x| matches!(x, Action::Write(w) if w.first() == Some(&tpdu_tags::DATA_LAST))));
}
}