当前位置:编程学习 > C/C++ >>

H.264参考帧列表管理分析 —— JM中相关函数解析(下)

本文解析与参考图像标记过程相关的几个函数。

[cpp] 
/*!
 ************************************************************************
 * \brief
 *    Perform Sliding window decoded reference picture marking process 参考图像的滑窗标记过程
 *
 ************************************************************************
 */ 
static void sliding_window_memory_management(StorablePicture* p) 

  unsigned i; 
 
  assert (!p->idr_flag); 
  // if this is a reference pic with sliding sliding window, unmark first ref frame 
  if (dpb.ref_frames_in_buffer==dpb.num_ref_frames - dpb.ltref_frames_in_buffer)//!< 减去dpb.ltref_frames_in_buffer因滑窗标记不能使用在长期参考帧上 
  { 
    for (i=0; i<dpb.used_size;i++) //!< 采取FIFO策略,将dpb中第一个短期参考帧移出参考帧列表,并更新参考帧列表 
    { 
      if (dpb.fs[i]->is_reference  && (!(dpb.fs[i]->is_long_term))) 
      { 
        unmark_for_reference(dpb.fs[i]); //!< 标记该帧为非参考帧 
        update_ref_list(); //!< 更新参考帧列表 
        break; 
      } 
    } 
  } 
 
  p->is_long_term = 0; 

 

[cpp]
<span style="font-size:12px;">/*!
 ************************************************************************
 * \brief
 *    Perform Adaptive memory control decoded reference picture marking process 自适应标记过程
 ************************************************************************
 */ 
static void adaptive_memory_management(StorablePicture* p) 

  DecRefPicMarking_t *tmp_drpm; 
   
  img->last_has_mmco_5 = 0;  //!< memory_management_control_operation 
 
  assert (!p->idr_flag); 
  assert (p->adaptive_ref_pic_buffering_flag); 
 
  while (p->dec_ref_pic_marking_buffer)  //!< stores the memory management control operations 
  { 
    tmp_drpm = p->dec_ref_pic_marking_buffer; //!< 临时保存操作,可用于在执行完该操作后释放掉对应的内存 
    switch (tmp_drpm->memory_management_control_operation) 
    { 
      case 0:   //!< 结束循环,退出标记操作 
        if (tmp_drpm->Next != NULL) 
        { 
          error ("memory_management_control_operation = 0 not last operation in buffer", 500); 
        } 
        break; 
      case 1:   //!< 将一个短期参考图像标记为非参考图像 
        mm_unmark_short_term_for_reference(p, tmp_drpm->difference_of_pic_nums_minus1); 
        update_ref_list(); 
        break; 
      case 2:   //!< 将一个长期参考图像标记为非参考图像 
        mm_unmark_long_term_for_reference(p, tmp_drpm->long_term_pic_num); 
        update_ltref_list(); 
        break; 
      case 3:   //!< 将一个短期参考图像转为长期参考图像 
        mm_assign_long_term_frame_idx(p, tmp_drpm->difference_of_pic_nums_minus1, tmp_drpm->long_term_frame_idx); 
        update_ref_list(); 
        update_ltref_list(); 
        break; 
      case 4:   //!< 指明长期参考帧的最大数目 
        mm_update_max_long_term_frame_idx (tmp_drpm->max_long_term_frame_idx_plus1); 
        update_ltref_list(); 
        break; 
      case 5:   //!< 清空参考帧队列,将所有参考图像移出参考帧队列,并禁用长期参考机制 
        mm_unmark_all_short_term_for_reference(); 
        mm_unmark_all_long_term_for_reference(); 
       img->last_has_mmco_5 = 1; //!< 此处作标记用于接下来对POC进行更新 
        break; 
      case 6:   //!< 将当前图像存为一个长期参考帧 
        mm_mark_current_picture_long_term(p, tmp_drpm->long_term_frame_idx); 
        check_num_ref(); 
        break; 
      default: 
        error ("invalid memory_management_control_operation in buffer", 500); 
    } 
    p->dec_ref_pic_marking_buffer = tmp_drpm->Next; //!< 取下一个操作 
    free (tmp_drpm); //!< 释放掉执行完的操作的内存 
  } 
  if ( img->last_has_mmco_5 ) 
  { 
    p->pic_num = p->frame_num = 0; //!< 因参考帧列表清空,frame_num清零 
     
    switch (p->structure) 
    { 
    case TOP_FIELD: 
      { 
        p->poc = p->top_poc = img->toppoc =0; 
        break; 
      } 
    case BOTTOM_FIELD: 
      { 
        p->poc = p->bottom_poc = img->bottompoc = 0; 
        break; 
      } 
    case FRAME: //!< 帧模式 
      {   www.zzzyk.com
        p->top_poc    -= p->poc; 
        p->bottom_poc -= p->poc; 
 
        img->toppoc = p->top_poc; 
        img->bottompoc = p->bottom_poc; 
 
        p->poc = min (p->top_poc, p->bottom_poc); 
        img->framepoc = p->poc; 
        break; 

补充:软件开发 , C++ ,
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,