| 
				
				
					
				
				
				 | 
			
			 | 
			@@ -358,13 +358,12 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, AVPac | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			{ | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    const uint8_t *buf = avpkt->data; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    int buf_size = avpkt->size; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    const uint8_t *buf_end = buf + buf_size; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    UtvideoContext *c = avctx->priv_data; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    const uint8_t *ptr; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    int i, j; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    const uint8_t *plane_start[5]; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    int plane_size, max_slice_size = 0, slice_start, slice_end, slice_size; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    int ret; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    GetByteContext gb; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			
  | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    if (c->pic.data[0]) | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        ff_thread_release_buffer(avctx, &c->pic); | 
		
		
	
	
		
			
				| 
				
				
				
					
				
				 | 
			
			 | 
			@@ -379,20 +378,21 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, AVPac | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    ff_thread_finish_setup(avctx); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			
  | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    /* parse plane structure to retrieve frame flags and validate slice offsets */ | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    ptr = buf; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    bytestream2_init(&gb, buf, buf_size); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    for (i = 0; i < c->planes; i++) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        plane_start[i] = ptr; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        if (buf_end - ptr < 256 + 4 * c->slices) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        plane_start[i] = gb.buffer; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        if (bytestream2_get_bytes_left(&gb) < 256 + 4 * c->slices) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            av_log(avctx, AV_LOG_ERROR, "Insufficient data for a plane\n"); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            return AVERROR_INVALIDDATA; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        } | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        ptr += 256; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        bytestream2_skipu(&gb, 256); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        slice_start = 0; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        slice_end   = 0; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        for (j = 0; j < c->slices; j++) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            slice_end   = bytestream_get_le32(&ptr); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            slice_end   = bytestream2_get_le32u(&gb); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            slice_size  = slice_end - slice_start; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            if (slice_size < 0) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            if (slice_end <= 0 || slice_size <= 0 || | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			                bytestream2_get_bytes_left(&gb) < slice_end) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			                av_log(avctx, AV_LOG_ERROR, "Incorrect slice size\n"); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			                return AVERROR_INVALIDDATA; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            } | 
		
		
	
	
		
			
				| 
				
				
				
					
				
				 | 
			
			 | 
			@@ -400,18 +400,14 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, AVPac | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            max_slice_size = FFMAX(max_slice_size, slice_size); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        } | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        plane_size = slice_end; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        if (buf_end - ptr < plane_size) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            av_log(avctx, AV_LOG_ERROR, "Plane size is bigger than available data\n"); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			            return AVERROR_INVALIDDATA; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        } | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        ptr += plane_size; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        bytestream2_skipu(&gb, plane_size); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    } | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    plane_start[c->planes] = ptr; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    if (buf_end - ptr < c->frame_info_size) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    plane_start[c->planes] = gb.buffer; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    if (bytestream2_get_bytes_left(&gb) < c->frame_info_size) { | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        av_log(avctx, AV_LOG_ERROR, "Not enough data for frame information\n"); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			        return AVERROR_INVALIDDATA; | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    } | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    c->frame_info = AV_RL32(ptr); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    c->frame_info = bytestream2_get_le32u(&gb); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    av_log(avctx, AV_LOG_DEBUG, "frame information flags %X\n", c->frame_info); | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			
  | 
		
		
	
		
			
			 | 
			 | 
			
			 | 
			    c->frame_pred = (c->frame_info >> 8) & 3; | 
		
		
	
	
		
			
				| 
				
					
				
				
				
				 | 
			
			 | 
			
  |