aboutsummaryrefslogtreecommitdiffstats
path: root/nihav-core/src/reorder.rs
blob: 9337b9a7f232c78962aae93066e4439ecab92f41 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
//! Output frame reordering.
//!
//! NihAV decoders output frames in the same order as they are put in.
//! In result if you want to have frames in display order you might need some frame reorderer.
//! This module provides such functionality depending on codec type: audio codecs and video codecs without B-frames do not need any reorderer and can use `NoReorderer` if the common interface is required. Codecs with B-frames should use `IPBReorderer`. For codecs with very complex reordering rules like H.264 or H.256 `PictureIDReorderer` will be added eventually.
//!
//! You can find out required reorderer by quering codec properties using `nihav_core::register` module.
use std::mem::swap;
use std::collections::VecDeque;
pub use crate::frame::{FrameType, NAFrameRef};

/// A trait for frame reorderer.
pub trait FrameReorderer {
    /// Stores a newly decoded frame.
    fn add_frame(&mut self, fref: NAFrameRef) -> bool;
    /// Gets the next frame to be displayed (or `None` if that is not possible).
    fn get_frame(&mut self) -> Option<NAFrameRef>;
    /// Clears all stored frames.
    fn flush(&mut self);
    /// Retrieves the last frames stored by the reorderer.
    fn get_last_frames(&mut self) -> Option<NAFrameRef>;
}

/// Zero reorderer.
pub struct NoReorderer {
    fref:   Option<NAFrameRef>,
}

impl NoReorderer {
    /// Constructs a new instance of `NoReorderer`.
    pub fn new() -> Self {
        Self { fref: None }
    }
}

impl Default for NoReorderer {
    fn default() -> Self {
        Self::new()
    }
}

impl FrameReorderer for NoReorderer {
    fn add_frame(&mut self, fref: NAFrameRef) -> bool {
        if self.fref.is_none() {
            self.fref = Some(fref);
            true
        } else {
            false
        }
    }
    fn get_frame(&mut self) -> Option<NAFrameRef> {
        let mut ret = None;
        swap(&mut ret, &mut self.fref);
        ret
    }
    fn flush(&mut self) { self.fref = None; }
    fn get_last_frames(&mut self) -> Option<NAFrameRef> { None }
}

/// Frame reorderer for codecs with I/P/B frames.
#[derive(Default)]
pub struct IPBReorderer {
    rframe:     Option<NAFrameRef>,
    bframe:     Option<NAFrameRef>,
}

impl IPBReorderer {
    /// Constructs a new instance of `IPBReorderer`.
    pub fn new() -> Self { Self::default() }
}

impl FrameReorderer for IPBReorderer {
    fn add_frame(&mut self, fref: NAFrameRef) -> bool {
        if self.rframe.is_some() && self.bframe.is_some() { return false; }
        let is_b = fref.get_frame_type() == FrameType::B;
        if is_b && self.bframe.is_some() { return false; }
        if is_b {
            self.bframe = Some(fref);
        } else {
            std::mem::swap(&mut self.bframe, &mut self.rframe);
            self.rframe = Some(fref);
        }
        true
    }
    fn get_frame(&mut self) -> Option<NAFrameRef> {
        let mut ret = None;
        if self.bframe.is_some() {
            std::mem::swap(&mut ret, &mut self.bframe);
        }
        ret
    }
    fn flush(&mut self) {
        self.rframe = None;
        self.bframe = None;
    }
    fn get_last_frames(&mut self) -> Option<NAFrameRef> {
        let mut ret = None;
        if self.bframe.is_some() {
            std::mem::swap(&mut ret, &mut self.bframe);
        } else if self.rframe.is_some() {
            std::mem::swap(&mut ret, &mut self.rframe);
        }
        ret
    }
}

/// Frame reorderer for codecs with complex I/P/B frame structure like ITU H.26x.
#[derive(Default)]
pub struct ComplexReorderer {
    last_ref_dts:   Option<u64>,
    ready_idx:      usize,
    frames:         Vec<NAFrameRef>,
}

impl ComplexReorderer {
    /// Constructs a new instance of `IPBReorderer`.
    pub fn new() -> Self { Self::default() }
}

impl FrameReorderer for ComplexReorderer {
    fn add_frame(&mut self, fref: NAFrameRef) -> bool {
        if self.frames.len() >= 64 {
            return false;
        }
        let is_ref = fref.frame_type == FrameType::I || fref.frame_type == FrameType::P;
        if !is_ref {
            if self.frames.is_empty() || fref.get_dts().is_none() {
                self.frames.push(fref);
            } else if let Some(new_dts) = fref.get_dts() {
                let mut idx = 0;
                for (i, frm) in self.frames.iter().enumerate() {
                    idx = i;
                    if let Some(dts) = frm.get_dts() {
                        if dts > new_dts {
                            break;
                        }
                    }
                }
                self.frames.insert(idx, fref);
            }
        } else {
            for (i, frm) in self.frames.iter().enumerate() {
                if frm.get_dts() == self.last_ref_dts {
                    self.ready_idx = i + 1;
                }
            }
            self.last_ref_dts = fref.get_dts();
            self.frames.push(fref);
        }
        true
    }
    fn get_frame(&mut self) -> Option<NAFrameRef> {
        if self.ready_idx > 0 {
            self.ready_idx -= 1;
            Some(self.frames.remove(0))
        } else {
            None
        }
    }
    fn flush(&mut self) {
        self.last_ref_dts = None;
        self.ready_idx = 0;
        self.frames.clear();
    }
    fn get_last_frames(&mut self) -> Option<NAFrameRef> {
        if !self.frames.is_empty() {
            Some(self.frames.remove(0))
        } else {
            None
        }
    }
}

/// A generic reorderer for a multi-threaded decoder.
#[derive(Default)]
pub struct MTFrameReorderer {
    ids:            Vec<u32>,
    frames:         VecDeque<(u32, NAFrameRef)>,
    cur_id:         u32,
    flush_mode:     bool,
    output_to:      Option<u32>,
    last_ts:        Option<u64>,
}

impl MTFrameReorderer {
    /// Constructs a new instance of `MTFrameReorderer`.
    pub fn new() -> Self { Self::default() }
    /// Registers the fact that a new frame is queued for decoding and returns an internal ID for it.
    pub fn register_frame(&mut self) -> u32 {
        if self.flush_mode {
            self.flush_mode = false;
        }
        let ret = self.cur_id;
        self.cur_id += 1;
        self.ids.push(ret);
        ret
    }
    /// Puts a newly decoded frame into the internal queue.
    pub fn add_frame(&mut self, frm: NAFrameRef, id: u32) {
        //let ftype = frm.get_frame_type();
        let frm_id = if let Some(ts) = frm.ts.dts { ts } else { u64::from(id) };
        let mut idx = 0;
        for (_, frm) in self.frames.iter() {
            let cur_id = if let Some(ts) = frm.ts.dts { ts } else { frm.id as u64 };
            if frm_id < cur_id {
                break;
            }
            idx += 1;
        }
        self.frames.insert(idx, (id, frm));
        /*if self.frames.len() > 48 {
            for (id, frm) in self.frames.iter() { print!(" {}{}({})", frm.get_frame_type(), frm.get_dts().unwrap_or(0), *id); } println!();
            print!("reg IDs:"); for &id in self.ids.iter() { print!(" {}", id); } println!();
            panic!("too many frames in the queue");
        }*/
    }
    /// Removes the registered frame (e.g. in case of a decoding error).
    pub fn drop_frame(&mut self, id: u32) {
        self.ids.retain(|&el| el != id);
    }
    fn get_first_frame(&mut self) -> Option<NAFrameRef> {
        let (id, frm) = self.frames.pop_front().unwrap();
        self.drop_frame(id);
        self.last_ts = frm.get_dts();
        Some(frm)
    }
    /// Gets the next frame to be displayed (or `None` if that is not possible).
    #[allow(clippy::collapsible_if)]
    #[allow(clippy::collapsible_else_if)]
    pub fn get_frame(&mut self) -> Option<NAFrameRef> {
        // check if we have consequent timestamps that we can output
        if !self.frames.is_empty() {
            if let Some(dts) = self.frames[0].1.get_dts() {
                let last_ts = self.last_ts.unwrap_or(0);
                if self.last_ts.is_none() || (dts == last_ts + 1) {
                    self.output_to = None;
                    return self.get_first_frame();
                }
            }
        }
        if !self.flush_mode {
            'out_loop: loop {
                if let Some(last_id) = self.output_to {
                    if self.frames[0].0 != last_id {
                        return self.get_first_frame();
                    } else {
                        self.output_to = None;
                    }
                }
                for (pos, (id, frm)) in self.frames.iter().enumerate() {
                    if frm.is_keyframe() || (self.frames.len() > 32 && matches!(frm.get_frame_type(), FrameType::I | FrameType::P)) {
                        let kf_id = *id;
                        self.ids.sort();
                        if pos == 0 && kf_id == self.ids[0] {
                            return self.get_first_frame();
                        }
                        let end = self.ids.iter().position(|&id| id == kf_id).unwrap();
                        for ref_id in self.ids[..end].iter() {
                            if !self.frames.iter().any(|(id, _)| id == ref_id) {
                                return None;
                            }
                        }
                        self.output_to = if pos < self.frames.len() - 1 {
                                Some(self.frames[pos + 1].0)
                            } else {
                                Some(kf_id)
                            };
                        continue 'out_loop;
                    }
                }
                return None;
            }
        } else {
            if !self.frames.is_empty() {
                Some(self.frames.pop_front().unwrap().1)
            } else {
                None
            }
        }
    }
    /// Retrieves the last frames stored by the reorderer.
    pub fn get_last_frames(&mut self) -> Option<NAFrameRef> {
        self.flush_mode = true;
        self.get_frame()
    }
    /// Clears all stored frames.
    pub fn flush(&mut self) {
        self.flush_mode = false;
        self.frames.clear();
        self.ids.clear();
        self.output_to = None;
        self.last_ts = None;
    }
}