Home | History | Annotate | Download | only in vda
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 // Note: ported from Chromium commit head: 2de6929
      5 
      6 #include <string.h>
      7 
      8 #include <algorithm>
      9 
     10 #include "base/logging.h"
     11 #include "base/stl_util.h"
     12 #include "h264_dpb.h"
     13 
     14 namespace media {
     15 
     16 H264Picture::H264Picture()
     17     : pic_order_cnt_type(0),
     18       top_field_order_cnt(0),
     19       bottom_field_order_cnt(0),
     20       pic_order_cnt(0),
     21       pic_order_cnt_msb(0),
     22       pic_order_cnt_lsb(0),
     23       delta_pic_order_cnt_bottom(0),
     24       delta_pic_order_cnt0(0),
     25       delta_pic_order_cnt1(0),
     26       pic_num(0),
     27       long_term_pic_num(0),
     28       frame_num(0),
     29       frame_num_offset(0),
     30       frame_num_wrap(0),
     31       long_term_frame_idx(0),
     32       type(H264SliceHeader::kPSlice),
     33       nal_ref_idc(0),
     34       idr(false),
     35       idr_pic_id(0),
     36       ref(false),
     37       long_term(false),
     38       outputted(false),
     39       mem_mgmt_5(false),
     40       nonexisting(false),
     41       field(FIELD_NONE),
     42       long_term_reference_flag(false),
     43       adaptive_ref_pic_marking_mode_flag(false),
     44       dpb_position(0) {
     45   memset(&ref_pic_marking, 0, sizeof(ref_pic_marking));
     46 }
     47 
     48 H264Picture::~H264Picture() = default;
     49 
     50 V4L2H264Picture* H264Picture::AsV4L2H264Picture() {
     51   return nullptr;
     52 }
     53 
     54 H264DPB::H264DPB() : max_num_pics_(0) {}
     55 H264DPB::~H264DPB() = default;
     56 
     57 void H264DPB::Clear() {
     58   pics_.clear();
     59 }
     60 
     61 void H264DPB::set_max_num_pics(size_t max_num_pics) {
     62   DCHECK_LE(max_num_pics, static_cast<size_t>(kDPBMaxSize));
     63   max_num_pics_ = max_num_pics;
     64   if (pics_.size() > max_num_pics_)
     65     pics_.resize(max_num_pics_);
     66 }
     67 
     68 void H264DPB::UpdatePicPositions() {
     69   size_t i = 0;
     70   for (auto& pic : pics_) {
     71     pic->dpb_position = i;
     72     ++i;
     73   }
     74 }
     75 
     76 void H264DPB::DeleteByPOC(int poc) {
     77   for (H264Picture::Vector::iterator it = pics_.begin(); it != pics_.end();
     78        ++it) {
     79     if ((*it)->pic_order_cnt == poc) {
     80       pics_.erase(it);
     81       UpdatePicPositions();
     82       return;
     83     }
     84   }
     85   NOTREACHED() << "Missing POC: " << poc;
     86 }
     87 
     88 void H264DPB::DeleteUnused() {
     89   for (H264Picture::Vector::iterator it = pics_.begin(); it != pics_.end();) {
     90     if ((*it)->outputted && !(*it)->ref)
     91       it = pics_.erase(it);
     92     else
     93       ++it;
     94   }
     95   UpdatePicPositions();
     96 }
     97 
     98 void H264DPB::StorePic(const scoped_refptr<H264Picture>& pic) {
     99   DCHECK_LT(pics_.size(), max_num_pics_);
    100   DVLOG(3) << "Adding PicNum: " << pic->pic_num << " ref: " << (int)pic->ref
    101            << " longterm: " << (int)pic->long_term << " to DPB";
    102   pic->dpb_position = pics_.size();
    103   pics_.push_back(pic);
    104 }
    105 
    106 int H264DPB::CountRefPics() {
    107   int ret = 0;
    108   for (size_t i = 0; i < pics_.size(); ++i) {
    109     if (pics_[i]->ref)
    110       ++ret;
    111   }
    112   return ret;
    113 }
    114 
    115 void H264DPB::MarkAllUnusedForRef() {
    116   for (size_t i = 0; i < pics_.size(); ++i)
    117     pics_[i]->ref = false;
    118 }
    119 
    120 scoped_refptr<H264Picture> H264DPB::GetShortRefPicByPicNum(int pic_num) {
    121   for (const auto& pic : pics_) {
    122     if (pic->ref && !pic->long_term && pic->pic_num == pic_num)
    123       return pic;
    124   }
    125 
    126   DVLOG(1) << "Missing short ref pic num: " << pic_num;
    127   return nullptr;
    128 }
    129 
    130 scoped_refptr<H264Picture> H264DPB::GetLongRefPicByLongTermPicNum(int pic_num) {
    131   for (const auto& pic : pics_) {
    132     if (pic->ref && pic->long_term && pic->long_term_pic_num == pic_num)
    133       return pic;
    134   }
    135 
    136   DVLOG(1) << "Missing long term pic num: " << pic_num;
    137   return nullptr;
    138 }
    139 
    140 scoped_refptr<H264Picture> H264DPB::GetLowestFrameNumWrapShortRefPic() {
    141   scoped_refptr<H264Picture> ret;
    142   for (const auto& pic : pics_) {
    143     if (pic->ref && !pic->long_term &&
    144         (!ret || pic->frame_num_wrap < ret->frame_num_wrap))
    145       ret = pic;
    146   }
    147   return ret;
    148 }
    149 
    150 void H264DPB::GetNotOutputtedPicsAppending(H264Picture::Vector* out) {
    151   for (const auto& pic : pics_) {
    152     if (!pic->outputted)
    153       out->push_back(pic);
    154   }
    155 }
    156 
    157 void H264DPB::GetShortTermRefPicsAppending(H264Picture::Vector* out) {
    158   for (const auto& pic : pics_) {
    159     if (pic->ref && !pic->long_term)
    160       out->push_back(pic);
    161   }
    162 }
    163 
    164 void H264DPB::GetLongTermRefPicsAppending(H264Picture::Vector* out) {
    165   for (const auto& pic : pics_) {
    166     if (pic->ref && pic->long_term)
    167       out->push_back(pic);
    168   }
    169 }
    170 
    171 }  // namespace media
    172