@ -4,125 +4,114 @@ 
			
		
	
		
		
			
				
					
					# include  <unistd.h> # include  <unistd.h>  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					static  int  ffmpeg_lockmgr_cb ( void  * * arg ,  enum  AVLockOp  op )  { static  int  ffmpeg_lockmgr_cb ( void  * * arg ,  enum  AVLockOp  op )  {  
			
		
	
		
		
			
				
					
					  pthread_mutex_t  * mutex  =  ( pthread_mutex_t  * ) * arg ;    std : : mutex  * mutex  =  ( std : : mutex  * ) * arg ;   
			
				
				
			
		
	
		
		
			
				
					
					  int  err ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
	
		
		
			
				
					
					  switch  ( op )  {    switch  ( op )  {   
			
		
	
		
		
			
				
					
					  case  AV_LOCK_CREATE :    case  AV_LOCK_CREATE :   
			
		
	
		
		
			
				
					
					    mutex  =  ( pthread_mutex_t  * ) malloc ( sizeof ( * mutex ) ) ;      mutex  =  new  std : : mutex ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    if  ( ! mutex )      break ;   
			
				
				
			
		
	
		
		
			
				
					
					        return  AVERROR ( ENOMEM ) ;   
			
		
	
		
		
			
				
					
					    if  ( ( err  =  pthread_mutex_init ( mutex ,  NULL ) ) )  {   
			
		
	
		
		
			
				
					
					        free ( mutex ) ;   
			
		
	
		
		
			
				
					
					        return  AVERROR ( err ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					    * arg  =  mutex ;   
			
		
	
		
		
			
				
					
					    return  0 ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  case  AV_LOCK_OBTAIN :    case  AV_LOCK_OBTAIN :   
			
		
	
		
		
			
				
					
					    if  ( ( err  =  pthread_mutex_lock ( mutex ) ) )      mutex - > lock ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        return  AVERROR ( err ) ;      break ;   
			
				
				
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    return  0 ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  case  AV_LOCK_RELEASE :    case  AV_LOCK_RELEASE :   
			
		
	
		
		
			
				
					
					    if  ( ( err  =  pthread_mutex_unlock ( mutex ) ) )      mutex - > unlock ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        return  AVERROR ( err ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    return  0 ;   
			
		
	
		
		
	
		
		
			
				
					
					  case  AV_LOCK_DESTROY :    case  AV_LOCK_DESTROY :   
			
		
	
		
		
			
				
					
					    if  ( mutex )      delete  mutex ;   
			
				
				
			
		
	
		
		
			
				
					
					        pthread_mutex_destroy ( mutex ) ;      break ;   
			
				
				
			
		
	
		
		
			
				
					
					    free ( mutex ) ;   
			
		
	
		
		
			
				
					
					    * arg  =  NULL ;   
			
		
	
		
		
			
				
					
					    return  0 ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  return  1 ;    return  0 ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					FrameReader : : FrameReader ( const  char  * fn )  { FrameReader : : FrameReader ( const  std : : string  & fn )  :  url ( fn )  {  
			
				
				
			
		
	
		
		
			
				
					
					  int  ret ;    int  ret  =  av_lockmgr_register ( ffmpeg_lockmgr_cb ) ;   
			
				
				
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ret  =  av_lockmgr_register ( ffmpeg_lockmgr_cb ) ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  assert ( ret  > =  0 ) ;    assert ( ret  > =  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  avformat_network_init ( ) ;    avformat_network_init ( ) ;   
			
		
	
		
		
			
				
					
					  av_register_all ( ) ;    av_register_all ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  snprintf ( url ,  sizeof ( url ) - 1 , " %s " , fn ) ;   
			
		
	
		
		
			
				
					
					  t  =  new  std : : thread ( [ & ] ( )  {  this - > loaderThread ( ) ;  } ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  FrameReader : : loaderThread ( )  { FrameReader : : ~ FrameReader ( )  {  
			
				
				
			
		
	
		
		
			
				
					
					  int  ret ;    exit_  =  true ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  thread . join ( ) ;   
			
		
	
		
		
			
				
					
					  for  ( auto  & f  :  frames )  {   
			
		
	
		
		
			
				
					
					    delete  f - > pkt ;   
			
		
	
		
		
			
				
					
					    if  ( f - > picture )  {   
			
		
	
		
		
			
				
					
					      av_frame_free ( & f - > picture ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					    delete  f ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  avcodec_free_context ( & pCodecCtx ) ;   
			
		
	
		
		
			
				
					
					  avformat_free_context ( pFormatCtx ) ;   
			
		
	
		
		
			
				
					
					  sws_freeContext ( sws_ctx ) ;   
			
		
	
		
		
			
				
					
					  avformat_network_deinit ( ) ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  if  ( avformat_open_input ( & pFormatCtx ,  url ,  NULL ,  NULL )  ! =  0 )  {  void  FrameReader : : process ( )  {  
			
				
				
			
		
	
		
		
			
				
					
					    fprintf ( stderr ,  " error loading %s \n " ,  url ) ;    if  ( avformat_open_input ( & pFormatCtx ,  url . c_str ( ) ,  NULL ,  NULL )  ! =  0 )  {   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					    fprintf ( stderr ,  " error loading %s \n " ,  url . c_str ( ) ) ;   
			
		
	
		
		
			
				
					
					    valid  =  false ;      valid  =  false ;   
			
		
	
		
		
			
				
					
					    return ;      return ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  av_dump_format ( pFormatCtx ,  0 ,  url ,  0 ) ;    avformat_find_stream_info ( pFormatCtx ,  NULL ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  av_dump_format ( pFormatCtx ,  0 ,  url . c_str ( ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  auto  pCodecCtxOrig  =  pFormatCtx - > streams [ 0 ] - > codec ;    auto  pCodecCtxOrig  =  pFormatCtx - > streams [ 0 ] - > codec ;   
			
		
	
		
		
			
				
					
					  auto  pCodec  =  avcodec_find_decoder ( pCodecCtxOrig - > codec_id ) ;    auto  pCodec  =  avcodec_find_decoder ( pCodecCtxOrig - > codec_id ) ;   
			
		
	
		
		
			
				
					
					  assert ( pCodec  ! =  NULL ) ;    assert ( pCodec  ! =  NULL ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  pCodecCtx  =  avcodec_alloc_context3 ( pCodec ) ;    pCodecCtx  =  avcodec_alloc_context3 ( pCodec ) ;   
			
		
	
		
		
			
				
					
					  ret  =  avcodec_copy_context ( pCodecCtx ,  pCodecCtxOrig ) ;    int  ret  =  avcodec_copy_context ( pCodecCtx ,  pCodecCtxOrig ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  assert ( ret  = =  0 ) ;    assert ( ret  = =  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ret  =  avcodec_open2 ( pCodecCtx ,  pCodec ,  NULL ) ;    ret  =  avcodec_open2 ( pCodecCtx ,  pCodec ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  assert ( ret  > =  0 ) ;    assert ( ret  > =  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  width  =  pCodecCtxOrig - > width ;   
			
		
	
		
		
			
				
					
					  height  =  pCodecCtxOrig - > height ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  sws_ctx  =  sws_getContext ( width ,  height ,  AV_PIX_FMT_YUV420P ,    sws_ctx  =  sws_getContext ( width ,  height ,  AV_PIX_FMT_YUV420P ,   
			
		
	
		
		
			
				
					
					                           width ,  height ,  AV_PIX_FMT_BGR24 ,                             width ,  height ,  AV_PIX_FMT_BGR24 ,   
			
		
	
		
		
			
				
					
					                           SWS_BILINEAR ,  NULL ,  NULL ,  NULL ) ;                             SWS_BILINEAR ,  NULL ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  assert ( sws_ctx  ! =  NULL ) ;    assert ( sws_ctx  ! =  NULL ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  AVPacket  * pkt  =  ( AVPacket  * ) malloc ( sizeof ( AVPacket ) ) ;    do  {   
			
				
				
			
		
	
		
		
			
				
					
					  assert ( pkt  ! =  NULL ) ;      AVPacket  * pkt  =  new  AVPacket ;   
			
				
				
			
		
	
		
		
			
				
					
					  bool  first  =  true ;      if  ( av_read_frame ( pFormatCtx ,  pkt )  <  0 )  {   
			
				
				
			
		
	
		
		
			
				
					
					  while  ( av_read_frame ( pFormatCtx ,  pkt ) > = 0 )  {        delete  pkt ;   
			
				
				
			
		
	
		
		
			
				
					
					    //printf("%d pkt %d %d\n", pkts.size(), pkt->size, pkt->pos);
        break ;   
			
				
				
			
		
	
		
		
			
				
					
					    if  ( first )  {   
			
		
	
		
		
			
				
					
					      AVFrame  * pFrame  =  av_frame_alloc ( ) ;   
			
		
	
		
		
			
				
					
					      int  frameFinished ;   
			
		
	
		
		
			
				
					
					      avcodec_decode_video2 ( pCodecCtx ,  pFrame ,  & frameFinished ,  pkt ) ;   
			
		
	
		
		
			
				
					
					      first  =  false ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    pkts . push_back ( pkt ) ;      Frame  * frame  =  new  Frame { . pkt  =  pkt } ;   
			
				
				
			
		
	
		
		
			
				
					
					    pkt  =  ( AVPacket  * ) malloc ( sizeof ( AVPacket ) ) ;      frames . push_back ( frame ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    assert ( pkt  ! =  NULL ) ;    }  while  ( true ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  }  
 
			
				
				
			
		
	
		
		
			
				
					
					  free ( pkt ) ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  printf ( " framereader download done \n " ) ;    printf ( " framereader download done \n " ) ;   
			
		
	
		
		
			
				
					
					  joined  =  true ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // cache
    thread  =  std : : thread ( & FrameReader : : decodeThread ,  this ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  while  ( 1 )  {   
			
		
	
		
		
			
				
					
					    GOPCache ( to_cache . get ( ) ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  FrameReader : : decodeThread ( )  {  
			
		
	
		
		
			
				
					
					  while  ( ! exit_ )  {   
			
		
	
		
		
			
				
					
					    int  gop  =  0 ;   
			
		
	
		
		
			
				
					
					    {   
			
		
	
		
		
			
				
					
					      std : : unique_lock  lk ( mutex ) ;   
			
		
	
		
		
			
				
					
					      cv_decode . wait ( lk ,  [ = ]  {  return  exit_  | |  decode_idx  ! =  - 1 ;  } ) ;   
			
		
	
		
		
			
				
					
					      if  ( exit_ )  break ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  FrameReader : : GOPCache ( int  idx )  {       gop  =  std : : max ( decode_idx  -  decode_idx  %  15 ,  0 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  AVFrame  * pFrame ;        decode_idx  =  - 1  ;   
			
				
				
			
		
	
		
		
			
				
					
					  int  gop  =  idx  -  idx % 15 ;      }    
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  mcache . lock ( ) ;      for  ( int  i  =  gop ;  i  <  std : : min ( gop  +  15 ,  ( int ) frames . size ( ) ) ;  + + i )  {   
			
				
				
			
		
	
		
		
			
				
					
					  bool  has_gop  =  cache . find ( gop )  ! =  cache . end ( ) ;        if  ( frames [ i ] - > picture  ! =  nullptr )  continue ;   
			
				
				
			
		
	
		
		
			
				
					
					  mcache . unlock ( ) ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  if  ( ! has_gop )  {   
			
		
	
		
		
			
				
					
					    //printf("caching %d\n", gop);
   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  gop ;  i  <  gop + 15 ;  i + + )  {   
			
		
	
		
		
			
				
					
					      if  ( i  > =  pkts . size ( ) )  break ;   
			
		
	
		
		
			
				
					
					      //printf("decode %d\n", i);
   
			
		
	
		
		
			
				
					
					      int  frameFinished ;        int  frameFinished ;   
			
		
	
		
		
			
				
					
					      pFrame  =  av_frame_alloc ( ) ;        AVFrame  * pFrame  =  av_frame_alloc ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      avcodec_decode_video2 ( pCodecCtx ,  pFrame ,  & frameFinished ,  pkts [ i ] ) ;        avcodec_decode_video2 ( pCodecCtx ,  pFrame ,  & frameFinished ,  frames [ i ] - > pkt ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      uint8_t  * dat  =  toRGB ( pFrame ) - > data [ 0 ] ;        AVFrame  * picture  =  toRGB ( pFrame ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      mcache . lock ( ) ;        av_frame_free ( & pFrame ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      cache . insert ( std : : make_pair ( i ,  dat ) ) ;  
 
			
				
				
			
		
	
		
		
			
				
					
					      mcache . unlock ( ) ;        std : : unique_lock  lk ( mutex ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					      frames [ i ] - > picture  =  picture ;   
			
		
	
		
		
			
				
					
					      cv_frame . notify_all ( ) ;   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -139,39 +128,14 @@ AVFrame *FrameReader::toRGB(AVFrame *pFrame) { 
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					uint8_t  * FrameReader : : get ( int  idx )  { uint8_t  * FrameReader : : get ( int  idx )  {  
			
		
	
		
		
			
				
					
					  if  ( ! valid )  return  NULL ;    if  ( ! valid  | |  idx  <  0  | |  idx  > =  frames . size ( ) )  return  nullptr ;   
			
				
				
			
		
	
		
		
			
				
					
					  waitForReady ( ) ;   
			
		
	
		
		
			
				
					
					  // TODO: one line?
   
			
		
	
		
		
			
				
					
					  uint8_t  * dat  =  NULL ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // lookahead
   
			
		
	
		
		
			
				
					
					  to_cache . put ( idx ) ;   
			
		
	
		
		
			
				
					
					  to_cache . put ( idx + 15 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  mcache . lock ( ) ;   
			
		
	
		
		
			
				
					
					  auto  it  =  cache . find ( idx ) ;   
			
		
	
		
		
			
				
					
					  if  ( it  ! =  cache . end ( ) )  {   
			
		
	
		
		
			
				
					
					    dat  =  it - > second ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  mcache . unlock ( ) ;   
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  if  ( dat  = =  NULL )  {    std : : unique_lock  lk ( mutex ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    to_cache . put_front ( idx ) ;    decode_idx  =  idx ;   
			
				
				
			
		
	
		
		
			
				
					
					    // lookahead
    cv_decode . notify_one ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    while  ( dat  = =  NULL )  {    Frame  * frame  =  frames [ idx ] ;   
			
				
				
			
		
	
		
		
			
				
					
					      // wait for frame
    if  ( ! frame - > picture )  {   
			
				
				
			
		
	
		
		
			
				
					
					      usleep ( 50 * 1000 ) ;      cv_frame . wait ( lk ,  [ = ]  {  return  exit_  | |  frame - > picture  ! =  nullptr ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      // check for frame
   
			
		
	
		
		
			
				
					
					      mcache . lock ( ) ;   
			
		
	
		
		
			
				
					
					      auto  it  =  cache . find ( idx ) ;   
			
		
	
		
		
			
				
					
					      if  ( it  ! =  cache . end ( ) )  dat  =  it - > second ;   
			
		
	
		
		
			
				
					
					      mcache . unlock ( ) ;   
			
		
	
		
		
			
				
					
					      if  ( dat  = =  NULL )  {   
			
		
	
		
		
			
				
					
					        printf ( " . " ) ;   
			
		
	
		
		
			
				
					
					        fflush ( stdout ) ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  return  frame - > picture  ?  frame - > picture - > data [ 0 ]  :  nullptr ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  return  dat ;   
			
		
	
		
		
			
				
					
					}