| 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 |
#ifdef HAVE_CONFIG_H |
|---|
| 29 |
# include "config.h" |
|---|
| 30 |
#endif |
|---|
| 31 |
|
|---|
| 32 |
#include <vlc_common.h> |
|---|
| 33 |
#include <vlc_plugin.h> |
|---|
| 34 |
#include <vlc_demux.h> |
|---|
| 35 |
|
|---|
| 36 |
|
|---|
| 37 |
|
|---|
| 38 |
|
|---|
| 39 |
|
|---|
| 40 |
|
|---|
| 41 |
|
|---|
| 42 |
|
|---|
| 43 |
static int Open ( vlc_object_t * ); |
|---|
| 44 |
static void Close ( vlc_object_t * ); |
|---|
| 45 |
|
|---|
| 46 |
vlc_module_begin(); |
|---|
| 47 |
set_category( CAT_INPUT ); |
|---|
| 48 |
set_subcategory( SUBCAT_INPUT_DEMUX ); |
|---|
| 49 |
set_description( N_("Nuv demuxer") ); |
|---|
| 50 |
set_capability( "demux", 145 ); |
|---|
| 51 |
set_callbacks( Open, Close ); |
|---|
| 52 |
add_shortcut( "nuv" ); |
|---|
| 53 |
vlc_module_end(); |
|---|
| 54 |
|
|---|
| 55 |
|
|---|
| 56 |
|
|---|
| 57 |
|
|---|
| 58 |
static int Demux ( demux_t * ); |
|---|
| 59 |
static int Control( demux_t *, int, va_list ); |
|---|
| 60 |
|
|---|
| 61 |
|
|---|
| 62 |
typedef struct |
|---|
| 63 |
{ |
|---|
| 64 |
int64_t i_time; |
|---|
| 65 |
int64_t i_offset; |
|---|
| 66 |
|
|---|
| 67 |
} demux_index_entry_t; |
|---|
| 68 |
|
|---|
| 69 |
typedef struct |
|---|
| 70 |
{ |
|---|
| 71 |
int i_idx; |
|---|
| 72 |
int i_idx_max; |
|---|
| 73 |
|
|---|
| 74 |
demux_index_entry_t *idx; |
|---|
| 75 |
} demux_index_t; |
|---|
| 76 |
|
|---|
| 77 |
|
|---|
| 78 |
static void demux_IndexInit( demux_index_t * ); |
|---|
| 79 |
static void demux_IndexClean( demux_index_t * ); |
|---|
| 80 |
static void demux_IndexAppend( demux_index_t *, |
|---|
| 81 |
int64_t i_time, int64_t i_offset ); |
|---|
| 82 |
|
|---|
| 83 |
static int64_t demux_IndexConvertTime( demux_index_t *, int64_t i_time ); |
|---|
| 84 |
|
|---|
| 85 |
static int64_t demux_IndexFindOffset( demux_index_t *, int64_t i_offset ); |
|---|
| 86 |
|
|---|
| 87 |
|
|---|
| 88 |
|
|---|
| 89 |
typedef struct |
|---|
| 90 |
{ |
|---|
| 91 |
char id[12]; |
|---|
| 92 |
char version[5]; |
|---|
| 93 |
|
|---|
| 94 |
int i_width; |
|---|
| 95 |
int i_height; |
|---|
| 96 |
int i_width_desired; |
|---|
| 97 |
int i_height_desired; |
|---|
| 98 |
|
|---|
| 99 |
char i_mode; |
|---|
| 100 |
|
|---|
| 101 |
double d_aspect; |
|---|
| 102 |
double d_fps; |
|---|
| 103 |
|
|---|
| 104 |
int i_video_blocks; |
|---|
| 105 |
int i_audio_blocks; |
|---|
| 106 |
int i_text_blocks; |
|---|
| 107 |
|
|---|
| 108 |
int i_keyframe_distance; |
|---|
| 109 |
|
|---|
| 110 |
} header_t; |
|---|
| 111 |
|
|---|
| 112 |
#define NUV_FH_SIZE 12 |
|---|
| 113 |
typedef struct |
|---|
| 114 |
{ |
|---|
| 115 |
char i_type; |
|---|
| 116 |
R: Seekpoint (string:RTjjjjjjjj) |
|---|
| 117 |
D: Extra data for codec |
|---|
| 118 |
X: extended data Q: seektable */ |
|---|
| 119 |
char i_compression; |
|---|
| 120 |
1 RTJpeg |
|---|
| 121 |
2 RTJpeg+lzo |
|---|
| 122 |
N black frame |
|---|
| 123 |
L copy last |
|---|
| 124 |
A: 0 uncompressed (44100 1-bits, 2ch) |
|---|
| 125 |
1 lzo |
|---|
| 126 |
2 layer 2 |
|---|
| 127 |
3 layer 3 |
|---|
| 128 |
F flac |
|---|
| 129 |
S shorten |
|---|
| 130 |
N null frame loudless |
|---|
| 131 |
L copy last |
|---|
| 132 |
S: B audio and vdeo sync point |
|---|
| 133 |
A audio sync info (timecode == effective |
|---|
| 134 |
dsp frequency*100) |
|---|
| 135 |
V next video sync (timecode == next video |
|---|
| 136 |
frame num) |
|---|
| 137 |
S audio,video,text correlation */ |
|---|
| 138 |
char i_keyframe; |
|---|
| 139 |
uint8_t i_filters; |
|---|
| 140 |
|
|---|
| 141 |
|
|---|
| 142 |
|
|---|
| 143 |
int i_timecode; |
|---|
| 144 |
|
|---|
| 145 |
int i_length; |
|---|
| 146 |
S: length of packet correl */ |
|---|
| 147 |
} frame_header_t; |
|---|
| 148 |
|
|---|
| 149 |
typedef struct |
|---|
| 150 |
{ |
|---|
| 151 |
int i_version; |
|---|
| 152 |
vlc_fourcc_t i_video_fcc; |
|---|
| 153 |
|
|---|
| 154 |
vlc_fourcc_t i_audio_fcc; |
|---|
| 155 |
int i_audio_sample_rate; |
|---|
| 156 |
int i_audio_bits_per_sample; |
|---|
| 157 |
int i_audio_channels; |
|---|
| 158 |
int i_audio_compression_ratio; |
|---|
| 159 |
int i_audio_quality; |
|---|
| 160 |
int i_rtjpeg_quality; |
|---|
| 161 |
int i_rtjpeg_luma_filter; |
|---|
| 162 |
int i_rtjpeg_chroma_filter; |
|---|
| 163 |
int i_lavc_bitrate; |
|---|
| 164 |
int i_lavc_qmin; |
|---|
| 165 |
int i_lavc_qmax; |
|---|
| 166 |
int i_lavc_maxqdiff; |
|---|
| 167 |
int64_t i_seektable_offset; |
|---|
| 168 |
int64_t i_keyframe_adjust_offset; |
|---|
| 169 |
|
|---|
| 170 |
} extended_header_t; |
|---|
| 171 |
|
|---|
| 172 |
struct demux_sys_t |
|---|
| 173 |
{ |
|---|
| 174 |
header_t hdr; |
|---|
| 175 |
extended_header_t exh; |
|---|
| 176 |
|
|---|
| 177 |
int64_t i_pcr; |
|---|
| 178 |
es_out_id_t *p_es_video; |
|---|
| 179 |
int i_extra_f; |
|---|
| 180 |
uint8_t *p_extra_f; |
|---|
| 181 |
|
|---|
| 182 |
es_out_id_t *p_es_audio; |
|---|
| 183 |
|
|---|
| 184 |
|
|---|
| 185 |
demux_index_t idx; |
|---|
| 186 |
bool b_index; |
|---|
| 187 |
bool b_seekable; |
|---|
| 188 |
|
|---|
| 189 |
uint8_t fh_buffer[NUV_FH_SIZE]; |
|---|
| 190 |
int64_t i_total_frames; |
|---|
| 191 |
int64_t i_total_length; |
|---|
| 192 |
|
|---|
| 193 |
int i_first_frame_offset; |
|---|
| 194 |
}; |
|---|
| 195 |
|
|---|
| 196 |
static int HeaderLoad( demux_t *, header_t *h ); |
|---|
| 197 |
static int FrameHeaderLoad( demux_t *, frame_header_t *h ); |
|---|
| 198 |
static int ExtendedHeaderLoad( demux_t *, extended_header_t *h ); |
|---|
| 199 |
static int SeekTableLoad( demux_t *, demux_sys_t * ); |
|---|
| 200 |
static int ControlSetPosition( demux_t *p_demux, int64_t i_pos, bool b_guess ); |
|---|
| 201 |
|
|---|
| 202 |
|
|---|
| 203 |
|
|---|
| 204 |
|
|---|
| 205 |
static int Open( vlc_object_t * p_this ) |
|---|
| 206 |
{ |
|---|
| 207 |
demux_t *p_demux = (demux_t*)p_this; |
|---|
| 208 |
demux_sys_t *p_sys; |
|---|
| 209 |
const uint8_t *p_peek; |
|---|
| 210 |
frame_header_t fh; |
|---|
| 211 |
bool b_extended; |
|---|
| 212 |
|
|---|
| 213 |
|
|---|
| 214 |
if( stream_Peek( p_demux->s, &p_peek, 12 ) != 12 || |
|---|
| 215 |
( strncmp( (char *)p_peek, "MythTVVideo", 11 ) && |
|---|
| 216 |
strncmp( (char *)p_peek, "NuppelVideo", 11 ) ) ) |
|---|
| 217 |
return VLC_EGENERIC; |
|---|
| 218 |
|
|---|
| 219 |
p_sys = malloc( sizeof( demux_sys_t ) ); |
|---|
| 220 |
if( p_sys == NULL ) |
|---|
| 221 |
return VLC_ENOMEM; |
|---|
| 222 |
memset( p_sys, 0, sizeof( demux_sys_t ) ); |
|---|
| 223 |
p_sys->p_es_video = NULL; |
|---|
| 224 |
p_sys->p_es_audio = NULL; |
|---|
| 225 |
p_sys->p_extra_f = NULL; |
|---|
| 226 |
p_sys->i_pcr = -1; |
|---|
| 227 |
p_sys->b_index = false; |
|---|
| 228 |
p_sys->i_total_frames = -1; |
|---|
| 229 |
p_sys->i_total_length = -1; |
|---|
| 230 |
demux_IndexInit( &p_sys->idx ); |
|---|
| 231 |
|
|---|
| 232 |
p_demux->p_sys = p_sys; |
|---|
| 233 |
|
|---|
| 234 |
|
|---|
| 235 |
stream_Control( p_demux->s, STREAM_CAN_SEEK, &p_sys->b_seekable ); |
|---|
| 236 |
#if 0 |
|---|
| 237 |
if( p_sys->b_seekable ) |
|---|
| 238 |
msg_Dbg( p_demux, "stream is seekable" ); |
|---|
| 239 |
else |
|---|
| 240 |
msg_Dbg( p_demux, "stream is NOT seekable" ); |
|---|
| 241 |
#endif |
|---|
| 242 |
|
|---|
| 243 |
if( HeaderLoad( p_demux, &p_sys->hdr ) ) |
|---|
| 244 |
goto error; |
|---|
| 245 |
|
|---|
| 246 |
|
|---|
| 247 |
if( FrameHeaderLoad( p_demux, &fh ) || fh.i_type != 'D' ) |
|---|
| 248 |
goto error; |
|---|
| 249 |
if( fh.i_length > 0 ) |
|---|
| 250 |
{ |
|---|
| 251 |
if( fh.i_compression == 'F' || fh.i_compression == 'R' ) |
|---|
| 252 |
{ |
|---|
| 253 |
|
|---|
| 254 |
p_sys->i_extra_f = fh.i_length; |
|---|
| 255 |
p_sys->p_extra_f = malloc( fh.i_length ); |
|---|
| 256 |
if( p_sys->p_extra_f == NULL || stream_Read( p_demux->s, |
|---|
| 257 |
p_sys->p_extra_f, fh.i_length ) != fh.i_length ) |
|---|
| 258 |
goto error; |
|---|
| 259 |
} |
|---|
| 260 |
else |
|---|
| 261 |
{ |
|---|
| 262 |
msg_Warn( p_demux, "unsupported 'D' frame (c=%c)", fh.i_compression ); |
|---|
| 263 |
if( stream_Read( p_demux->s, NULL, fh.i_length ) != fh.i_length ) |
|---|
| 264 |
goto error; |
|---|
| 265 |
} |
|---|
| 266 |
} |
|---|
| 267 |
|
|---|
| 268 |
|
|---|
| 269 |
if( stream_Peek( p_demux->s, &p_peek, 1 ) != 1 ) |
|---|
| 270 |
goto error; |
|---|
| 271 |
if( p_peek[0] == 'X' ) |
|---|
| 272 |
{ |
|---|
| 273 |
b_extended = true; |
|---|
| 274 |
|
|---|
| 275 |
if( FrameHeaderLoad( p_demux, &fh ) ) |
|---|
| 276 |
goto error; |
|---|
| 277 |
if( fh.i_length != 512 ) |
|---|
| 278 |
goto error; |
|---|
| 279 |
|
|---|
| 280 |
if( ExtendedHeaderLoad( p_demux, &p_sys->exh ) ) |
|---|
| 281 |
goto error; |
|---|
| 282 |
|
|---|
| 283 |
if( !p_sys->b_seekable ) |
|---|
| 284 |
msg_Warn( p_demux, "stream is not seekable, skipping seektable" ); |
|---|
| 285 |
else if( SeekTableLoad( p_demux, p_sys ) ) |
|---|
| 286 |
goto error; |
|---|
| 287 |
|
|---|
| 288 |
} |
|---|
| 289 |
else |
|---|
| 290 |
{ |
|---|
| 291 |
b_extended = false; |
|---|
| 292 |
|
|---|
| 293 |
|
|---|
| 294 |
|
|---|
| 295 |
msg_Err( p_demux, "incomplete NUV support (upload samples)" ); |
|---|
| 296 |
goto error; |
|---|
| 297 |
} |
|---|
| 298 |
|
|---|
| 299 |
|
|---|
| 300 |
if( p_sys->hdr.i_video_blocks != 0 ) |
|---|
| 301 |
{ |
|---|
| 302 |
es_format_t fmt; |
|---|
| 303 |
|
|---|
| 304 |
es_format_Init( &fmt, VIDEO_ES, p_sys->exh.i_video_fcc ); |
|---|
| 305 |
fmt.video.i_width = p_sys->hdr.i_width; |
|---|
| 306 |
fmt.video.i_height = p_sys->hdr.i_height; |
|---|
| 307 |
fmt.i_extra = p_sys->i_extra_f; |
|---|
| 308 |
fmt.p_extra = p_sys->p_extra_f; |
|---|
| 309 |
fmt.video.i_aspect = VOUT_ASPECT_FACTOR * p_sys->hdr.d_aspect; |
|---|
| 310 |
|
|---|
| 311 |
p_sys->p_es_video = es_out_Add( p_demux->out, &fmt ); |
|---|
| 312 |
} |
|---|
| 313 |
if( p_sys->hdr.i_audio_blocks != 0 ) |
|---|
| 314 |
{ |
|---|
| 315 |
es_format_t fmt; |
|---|
| 316 |
|
|---|
| 317 |
es_format_Init( &fmt, AUDIO_ES, VLC_FOURCC('m','p','g','a') ); |
|---|
| 318 |
fmt.audio.i_rate = p_sys->exh.i_audio_sample_rate; |
|---|
| 319 |
fmt.audio.i_bitspersample = p_sys->exh.i_audio_bits_per_sample; |
|---|
| 320 |
|
|---|
| 321 |
p_sys->p_es_audio = es_out_Add( p_demux->out, &fmt ); |
|---|
| 322 |
} |
|---|
| 323 |
if( p_sys->hdr.i_text_blocks != 0 ) |
|---|
| 324 |
{ |
|---|
| 325 |
msg_Warn( p_demux, "text not yet supported (upload samples)" ); |
|---|
| 326 |
} |
|---|
| 327 |
|
|---|
| 328 |
p_sys->i_first_frame_offset = stream_Tell( p_demux->s ); |
|---|
| 329 |
|
|---|
| 330 |
|
|---|
| 331 |
p_demux->pf_demux = Demux; |
|---|
| 332 |
p_demux->pf_control = Control; |
|---|
| 333 |
|
|---|
| 334 |
return VLC_SUCCESS; |
|---|
| 335 |
|
|---|
| 336 |
error: |
|---|
| 337 |
msg_Warn( p_demux, "cannot load Nuv file" ); |
|---|
| 338 |
p_demux->p_sys = NULL; |
|---|
| 339 |
free( p_sys ); |
|---|
| 340 |
return VLC_EGENERIC; |
|---|
| 341 |
} |
|---|
| 342 |
|
|---|
| 343 |
|
|---|
| 344 |
|
|---|
| 345 |
|
|---|
| 346 |
static void Close( vlc_object_t * p_this ) |
|---|
| 347 |
{ |
|---|
| 348 |
demux_t *p_demux = (demux_t*)p_this; |
|---|
| 349 |
demux_sys_t *p_sys = p_demux->p_sys; |
|---|
| 350 |
|
|---|
| 351 |
free( p_sys->p_extra_f ); |
|---|
| 352 |
demux_IndexClean( &p_sys->idx ); |
|---|
| 353 |
free( p_sys ); |
|---|
| 354 |
} |
|---|
| 355 |
|
|---|
| 356 |
|
|---|
| 357 |
|
|---|
| 358 |
|
|---|
| 359 |
|
|---|
| 360 |
|
|---|
| 361 |
static int Demux( demux_t *p_demux ) |
|---|
| 362 |
{ |
|---|
| 363 |
demux_sys_t *p_sys = p_demux->p_sys; |
|---|
| 364 |
frame_header_t fh; |
|---|
| 365 |
block_t *p_data; |
|---|
| 366 |
|
|---|
| 367 |
for( ;; ) |
|---|
| 368 |
{ |
|---|
| 369 |
if( !vlc_object_alive (p_demux) ) |
|---|
| 370 |
return -1; |
|---|
| 371 |
|
|---|
| 372 |
if( FrameHeaderLoad( p_demux, &fh ) ) |
|---|
| 373 |
return 0; |
|---|
| 374 |
|
|---|
| 375 |
if( fh.i_type == 'A' || fh.i_type == 'V' ) |
|---|
| 376 |
break; |
|---|
| 377 |
|
|---|
| 378 |
|
|---|
| 379 |
|
|---|
| 380 |
if( fh.i_type != 'R' && fh.i_length > 0 ) |
|---|
| 381 |
{ |
|---|
| 382 |
if( stream_Read( p_demux->s, NULL, fh.i_length ) != fh.i_length ) |
|---|
| 383 |
return -1; |
|---|
| 384 |
} |
|---|
| 385 |
} |
|---|
| 386 |
|
|---|
| 387 |
|
|---|
| 388 |
p_data = stream_Block( p_demux->s, fh.i_length ); |
|---|
| 389 |
p_data->i_dts = (int64_t)fh.i_timecode * 1000; |
|---|
| 390 |
p_data->i_pts = (fh.i_type == 'V') ? 0 : p_data->i_dts; |
|---|
| 391 |
|
|---|
| 392 |
|
|---|
| 393 |
if( !fh.i_keyframe && !p_sys->b_index ) |
|---|
| 394 |
demux_IndexAppend( &p_sys->idx, p_data->i_dts, stream_Tell(p_demux->s) - NUV_FH_SIZE ); |
|---|
| 395 |
|
|---|
| 396 |
|
|---|
| 397 |
if( p_data->i_dts > p_sys->i_pcr ) |
|---|
| 398 |
{ |
|---|
| 399 |
p_sys->i_pcr = p_data->i_dts; |
|---|
| 400 |
es_out_Control( p_demux->out, ES_OUT_SET_PCR, p_sys->i_pcr ); |
|---|
| 401 |
} |
|---|
| 402 |
|
|---|
| 403 |
if( fh.i_type == 'A' && p_sys->p_es_audio ) |
|---|
| 404 |
{ |
|---|
| 405 |
if( fh.i_compression == '3' ) |
|---|
| 406 |
es_out_Send( p_demux->out, p_sys->p_es_audio, p_data ); |
|---|
| 407 |
else |
|---|
| 408 |
{ |
|---|
| 409 |
msg_Dbg( p_demux, "unsupported compression %c for audio (upload samples)", fh.i_compression ); |
|---|
| 410 |
block_Release( p_data ); |
|---|
| 411 |
} |
|---|
| 412 |
} |
|---|
| 413 |
else if( fh.i_type == 'V' && p_sys->p_es_video ) |
|---|
| 414 |
{ |
|---|
| 415 |
if( fh.i_compression >='0' && fh.i_compression <='3' ) |
|---|
| 416 |
{ |
|---|
| 417 |
|
|---|
| 418 |
p_data = block_Realloc( p_data, NUV_FH_SIZE, fh.i_length ); |
|---|
| 419 |
memcpy( p_data->p_buffer, p_sys->fh_buffer, NUV_FH_SIZE ); |
|---|
| 420 |
} |
|---|
| 421 |
|
|---|
| 422 |
if( fh.i_compression >= '0' ) |
|---|
| 423 |
es_out_Send( p_demux->out, p_sys->p_es_video, p_data ); |
|---|
| 424 |
else |
|---|
| 425 |
{ |
|---|
| 426 |
msg_Dbg( p_demux, "unsupported compression %c for video (upload samples)", fh.i_compression ); |
|---|
| 427 |
block_Release( p_data ); |
|---|
| 428 |
} |
|---|
| 429 |
} |
|---|
| 430 |
else |
|---|
| 431 |
{ |
|---|
| 432 |
block_Release( p_data ); |
|---|
| 433 |
} |
|---|
| 434 |
|
|---|
| 435 |
return 1; |
|---|
| 436 |
} |
|---|
| 437 |
|
|---|
| 438 |
|
|---|
| 439 |
|
|---|
| 440 |
|
|---|
| 441 |
static int Control( demux_t *p_demux, int i_query, va_list args ) |
|---|
| 442 |
{ |
|---|
| 443 |
demux_sys_t *p_sys = p_demux->p_sys; |
|---|
| 444 |
|
|---|
| 445 |
double f, *pf; |
|---|
| 446 |
int64_t i64, *pi64; |
|---|
| 447 |
|
|---|
| 448 |
switch( i_query ) |
|---|
| 449 |
{ |
|---|
| 450 |
|
|---|
| 451 |
case DEMUX_GET_POSITION: |
|---|
| 452 |
pf = (double*)va_arg( args, double * ); |
|---|
| 453 |
|
|---|
| 454 |
if( p_sys->i_total_length > 0 && p_sys->i_pcr >= 0 ) |
|---|
| 455 |
{ |
|---|
| 456 |
*pf = (double)p_sys->i_pcr / (double)p_sys->i_total_length; |
|---|
| 457 |
} |
|---|
| 458 |
else |
|---|
| 459 |
{ |
|---|
| 460 |
i64 = stream_Size( p_demux->s ); |
|---|
| 461 |
if( i64 > 0 ) |
|---|
| 462 |
*pf = (double)stream_Tell( p_demux->s ) / (double)i64; |
|---|
| 463 |
else |
|---|
| 464 |
*pf = 0.0; |
|---|
| 465 |
} |
|---|
| 466 |
return VLC_SUCCESS; |
|---|
| 467 |
|
|---|
| 468 |
case DEMUX_SET_POSITION: |
|---|
| 469 |
{ |
|---|
| 470 |
int64_t i_pos; |
|---|
| 471 |
|
|---|
| 472 |
f = (double)va_arg( args, double ); |
|---|
| 473 |
|
|---|
| 474 |
p_sys->i_pcr = -1; |
|---|
| 475 |
|
|---|
| 476 |
|
|---|
| 477 |
if( p_sys->i_total_length > 0 && ( i_pos = demux_IndexConvertTime( &p_sys->idx, p_sys->i_total_length * f ) ) > 0 ) |
|---|
| 478 |
return ControlSetPosition( p_demux, i_pos, false ); |
|---|
| 479 |
|
|---|
| 480 |
|
|---|
| 481 |
else if( ( i_pos = demux_IndexFindOffset( &p_sys->idx, stream_Size( p_demux->s ) * f ) ) >= 0 ) |
|---|
| 482 |
return ControlSetPosition( p_demux, i_pos, false ); |
|---|
| 483 |
|
|---|
| 484 |
else if( ( i_pos = p_sys->i_first_frame_offset + ( stream_Size( p_demux->s ) - p_sys->i_first_frame_offset ) * f ) >= 0 ) |
|---|
| 485 |
return ControlSetPosition( p_demux, i_pos, true ); |
|---|
| 486 |
|
|---|
| 487 |
else |
|---|
| 488 |
return VLC_EGENERIC; |
|---|
| 489 |
} |
|---|
| 490 |
|
|---|
| 491 |
case DEMUX_GET_TIME: |
|---|
| 492 |
pi64 = (int64_t*)va_arg( args, int64_t * ); |
|---|
| 493 |
*pi64 = p_sys->i_pcr >= 0 ? p_sys->i_pcr : 0; |
|---|
| 494 |
return VLC_SUCCESS; |
|---|
| 495 |
|
|---|
| 496 |
case DEMUX_SET_TIME: |
|---|
| 497 |
{ |
|---|
| 498 |
int64_t i_pos; |
|---|
| 499 |
i64 = (int64_t)va_arg( args, int64_t ); |
|---|
| 500 |
|
|---|
| 501 |
p_sys->i_pcr = -1; |
|---|
| 502 |
|
|---|
| 503 |
i_pos = demux_IndexConvertTime( &p_sys->idx, i64 ); |
|---|
| 504 |
if( i_pos < 0 ) |
|---|
| 505 |
return VLC_EGENERIC; |
|---|
| 506 |
else |
|---|
| 507 |
return ControlSetPosition( p_demux, i_pos, false ); |
|---|
| 508 |
} |
|---|
| 509 |
|
|---|
| 510 |
case DEMUX_GET_LENGTH: |
|---|
| 511 |
pi64 = (int64_t*)va_arg( args, int64_t * ); |
|---|
| 512 |
if( p_sys->i_total_length >= 0 ) |
|---|
| 513 |
{ |
|---|
| 514 |
*pi64 = p_sys->i_total_length; |
|---|
| 515 |
return VLC_SUCCESS; |
|---|
| 516 |
} |
|---|
| 517 |
else if( stream_Tell( p_demux->s ) > p_sys->i_first_frame_offset ) |
|---|
| 518 |
{ |
|---|
| 519 |
|
|---|
| 520 |
*pi64 = (double)( stream_Size( p_demux->s ) - p_sys->i_first_frame_offset ) / |
|---|
| 521 |
(double)( stream_Tell( p_demux->s ) - p_sys->i_first_frame_offset ) |
|---|
| 522 |
* (double)( p_sys->i_pcr >= 0 ? p_sys->i_pcr : 0 ); |
|---|
| 523 |
return VLC_SUCCESS; |
|---|
| 524 |
} |
|---|
| 525 |
else |
|---|
| 526 |
return VLC_EGENERIC; |
|---|
| 527 |
|
|---|
| 528 |
case DEMUX_GET_FPS: |
|---|
| 529 |
pf = (double*)va_arg( args, double * ); |
|---|
| 530 |
*pf = p_sys->hdr.d_fps; |
|---|
| 531 |
return VLC_SUCCESS; |
|---|
| 532 |
|
|---|
| 533 |
case DEMUX_GET_META: |
|---|
| 534 |
default: |
|---|
| 535 |
return VLC_EGENERIC; |
|---|
| 536 |
|
|---|
| 537 |
} |
|---|
| 538 |
} |
|---|
| 539 |
static int ControlSetPosition( demux_t *p_demux, int64_t i_pos, bool b_guess ) |
|---|
| 540 |
{ |
|---|
| 541 |
demux_sys_t *p_sys = p_demux->p_sys; |
|---|
| 542 |
|
|---|
| 543 |
if( i_pos < 0 ) |
|---|
| 544 |
return VLC_EGENERIC; |
|---|
| 545 |
|
|---|
| 546 |
|
|---|
| 547 |
if( p_sys->b_seekable && !b_guess ) |
|---|
| 548 |
{ |
|---|
| 549 |
if( stream_Seek( p_demux->s, i_pos ) ) |
|---|
| 550 |
return VLC_EGENERIC; |
|---|
| 551 |
} |
|---|
| 552 |
else |
|---|
| 553 |
{ |
|---|
| 554 |
|
|---|
| 555 |
if( i_pos > stream_Tell( p_demux->s ) ) |
|---|
| 556 |
{ |
|---|
| 557 |
msg_Dbg( p_demux, "unable to seek, skipping frames (slow)" ); |
|---|
| 558 |
} |
|---|
| 559 |
else |
|---|
| 560 |
{ |
|---|
| 561 |
msg_Warn( p_demux, "unable to seek, only forward seeking is possible" ); |
|---|
| 562 |
|
|---|
| 563 |
return VLC_EGENERIC; |
|---|
| 564 |
} |
|---|
| 565 |
} |
|---|
| 566 |
|
|---|
| 567 |
while( vlc_object_alive (p_demux) ) |
|---|
| 568 |
{ |
|---|
| 569 |
frame_header_t fh; |
|---|
| 570 |
int64_t i_tell; |
|---|
| 571 |
|
|---|
| 572 |
if( ( i_tell = stream_Tell( p_demux->s ) ) >= i_pos ) |
|---|
| 573 |
break; |
|---|
| 574 |
|
|---|
| 575 |
if( FrameHeaderLoad( p_demux, &fh ) ) |
|---|
| 576 |
return VLC_EGENERIC; |
|---|
| 577 |
|
|---|
| 578 |
if( fh.i_type == 'A' || fh.i_type == 'V' ) |
|---|
| 579 |
{ |
|---|
| 580 |
if( !fh.i_keyframe && !p_sys->b_index ) |
|---|
| 581 |
demux_IndexAppend( &p_sys->idx,(int64_t)fh.i_timecode*1000, i_tell ); |
|---|
| 582 |
} |
|---|
| 583 |
|
|---|
| 584 |
if( fh.i_type != 'R' && fh.i_length > 0 ) |
|---|
| 585 |
{ |
|---|
| 586 |
if( stream_Read( p_demux->s, NULL, fh.i_length ) != fh.i_length ) |
|---|
| 587 |
return VLC_EGENERIC; |
|---|
| 588 |
} |
|---|
| 589 |
} |
|---|
| 590 |
|
|---|
| 591 |
return VLC_SUCCESS; |
|---|
| 592 |
} |
|---|
| 593 |
|
|---|
| 594 |
|
|---|
| 595 |
|
|---|
| 596 |
|
|---|
| 597 |
static inline void GetDoubleLE( double *pd, void *src ) |
|---|
| 598 |
{ |
|---|
| 599 |
|
|---|
| 600 |
#ifdef WORDS_BIGENDIAN |
|---|
| 601 |
uint8_t *p = (uint8_t*)pd, *q = (uint8_t*)src; |
|---|
| 602 |
int i; |
|---|
| 603 |
for( i = 0; i < 8; i++ ) |
|---|
| 604 |
p[i] = q[7-i]; |
|---|
| 605 |
#else |
|---|
| 606 |
memcpy( pd, src, 8 ); |
|---|
| 607 |
#endif |
|---|
| 608 |
} |
|---|
| 609 |
|
|---|
| 610 |
|
|---|
| 611 |
|
|---|
| 612 |
static int HeaderLoad( demux_t *p_demux, header_t *h ) |
|---|
| 613 |
{ |
|---|
| 614 |
uint8_t buffer[72]; |
|---|
| 615 |
|
|---|
| 616 |
if( stream_Read( p_demux->s, buffer, 72 ) != 72 ) |
|---|
| 617 |
return VLC_EGENERIC; |
|---|
| 618 |
|
|---|
| 619 |
|
|---|
| 620 |
memcpy( h->id, &buffer[ 0], 12 ); |
|---|
| 621 |
memcpy( h->version, &buffer[12], 5 ); |
|---|
| 622 |
h->i_width = GetDWLE( &buffer[20] ); |
|---|
| 623 |
h->i_height = GetDWLE( &buffer[24] ); |
|---|
| 624 |
h->i_width_desired = GetDWLE( &buffer[28] ); |
|---|
| 625 |
h->i_height_desired = GetDWLE( &buffer[32] ); |
|---|
| 626 |
h->i_mode = buffer[36]; |
|---|
| 627 |
GetDoubleLE( &h->d_aspect, &buffer[40] ); |
|---|
| 628 |
GetDoubleLE( &h->d_fps, &buffer[48] ); |
|---|
| 629 |
h->i_video_blocks = GetDWLE( &buffer[56] ); |
|---|
| 630 |
h->i_audio_blocks = GetDWLE( &buffer[60] ); |
|---|
| 631 |
h->i_text_blocks = GetDWLE( &buffer[64] ); |
|---|
| 632 |
h->i_keyframe_distance = GetDWLE( &buffer[68] ); |
|---|
| 633 |
#if 0 |
|---|
| 634 |
msg_Dbg( p_demux, "nuv: h=%s v=%s %dx%d a=%f fps=%f v=%d a=%d t=%d kfd=%d", |
|---|
| 635 |
h->id, h->version, h->i_width, h->i_height, h->d_aspect, |
|---|
| 636 |
h->d_fps, h->i_video_blocks, h->i_audio_blocks, h->i_text_blocks, |
|---|
| 637 |
h->i_keyframe_distance ); |
|---|
| 638 |
#endif |
|---|
| 639 |
return VLC_SUCCESS; |
|---|
| 640 |
} |
|---|
| 641 |
|
|---|
| 642 |
|
|---|
| 643 |
|
|---|
| 644 |
static int FrameHeaderLoad( demux_t *p_demux, frame_header_t *h ) |
|---|
| 645 |
{ |
|---|
| 646 |
uint8_t* buffer = p_demux->p_sys->fh_buffer; |
|---|
| 647 |
|
|---|
| 648 |
if( stream_Read( p_demux->s, buffer, 12 ) != 12 ) |
|---|
| 649 |
return VLC_EGENERIC; |
|---|
| 650 |
|
|---|
| 651 |
h->i_type = buffer[0]; |
|---|
| 652 |
h->i_compression = buffer[1]; |
|---|
| 653 |
h->i_keyframe = buffer[2]; |
|---|
| 654 |
h->i_filters = buffer[3]; |
|---|
| 655 |
|
|---|
| 656 |
h->i_timecode = GetDWLE( &buffer[4] ); |
|---|
| 657 |
h->i_length = GetDWLE( &buffer[8] ); |
|---|
| 658 |
#if 0 |
|---|
| 659 |
msg_Dbg( p_demux, "frame hdr: t=%c c=%c k=%d f=0x%x timecode=%d l=%d", |
|---|
| 660 |
h->i_type, |
|---|
| 661 |
h->i_compression ? h->i_compression : ' ', |
|---|
| 662 |
h->i_keyframe ? h->i_keyframe : ' ', |
|---|
| 663 |
h->i_filters, |
|---|
| 664 |
h->i_timecode, h->i_length ); |
|---|
| 665 |
#endif |
|---|
| 666 |
return VLC_SUCCESS; |
|---|
| 667 |
} |
|---|
| 668 |
|
|---|
| 669 |
static int ExtendedHeaderLoad( demux_t *p_demux, extended_header_t *h ) |
|---|
| 670 |
{ |
|---|
| 671 |
uint8_t buffer[512]; |
|---|
| 672 |
|
|---|
| 673 |
if( stream_Read( p_demux->s, buffer, 512 ) != 512 ) |
|---|
| 674 |
return VLC_EGENERIC; |
|---|
| 675 |
|
|---|
| 676 |
h->i_version = GetDWLE( &buffer[0] ); |
|---|
| 677 |
h->i_video_fcc = VLC_FOURCC( buffer[4], buffer[5], buffer[6], buffer[7] ); |
|---|
| 678 |
h->i_audio_fcc = VLC_FOURCC( buffer[8], buffer[9], buffer[10], buffer[11] ); |
|---|
| 679 |
h->i_audio_sample_rate = GetDWLE( &buffer[12] ); |
|---|
| 680 |
h->i_audio_bits_per_sample = GetDWLE( &buffer[16] ); |
|---|
| 681 |
h->i_audio_channels = GetDWLE( &buffer[20] ); |
|---|
| 682 |
h->i_audio_compression_ratio = GetDWLE( &buffer[24] ); |
|---|
| 683 |
h->i_audio_quality = GetDWLE( &buffer[28] ); |
|---|
| 684 |
h->i_rtjpeg_quality = GetDWLE( &buffer[32] ); |
|---|
| 685 |
h->i_rtjpeg_luma_filter = GetDWLE( &buffer[36] ); |
|---|
| 686 |
h->i_rtjpeg_chroma_filter = GetDWLE( &buffer[40] ); |
|---|
| 687 |
h->i_lavc_bitrate = GetDWLE( &buffer[44] ); |
|---|
| 688 |
h->i_lavc_qmin = GetDWLE( &buffer[48] ); |
|---|
| 689 |
h->i_lavc_qmin = GetDWLE( &buffer[52] ); |
|---|
| 690 |
h->i_lavc_maxqdiff = GetDWLE( &buffer[56] ); |
|---|
| 691 |
h->i_seektable_offset = GetQWLE( &buffer[60] ); |
|---|
| 692 |
h->i_keyframe_adjust_offset= GetQWLE( &buffer[68] ); |
|---|
| 693 |
#if 0 |
|---|
| 694 |
msg_Dbg( p_demux, "ex hdr: v=%d vffc=%4.4s afcc=%4.4s %dHz %dbits ach=%d acr=%d aq=%d" |
|---|
| 695 |
"rtjpeg q=%d lf=%d lc=%d lavc br=%d qmin=%d qmax=%d maxqdiff=%d seekableoff=%"PRIi64" keyfao=%"PRIi64, |
|---|
| 696 |
h->i_version, |
|---|
| 697 |
(char*)&h->i_video_fcc, |
|---|
| 698 |
(char*)&h->i_audio_fcc, h->i_audio_sample_rate, h->i_audio_bits_per_sample, h->i_audio_channels, |
|---|
| 699 |
h->i_audio_compression_ratio, h->i_audio_quality, |
|---|
| 700 |
h->i_rtjpeg_quality, h->i_rtjpeg_luma_filter, h->i_rtjpeg_chroma_filter, |
|---|
| 701 |
h->i_lavc_bitrate, h->i_lavc_qmin, h->i_lavc_qmax, h->i_lavc_maxqdiff, |
|---|
| 702 |
h->i_seektable_offset, h->i_keyframe_adjust_offset ); |
|---|
| 703 |
#endif |
|---|
| 704 |
return VLC_SUCCESS; |
|---|
| 705 |
} |
|---|
| 706 |
|
|---|
| 707 |
|
|---|
| 708 |
|
|---|
| 709 |
|
|---|
| 710 |
|
|---|
| 711 |
|
|---|
| 712 |
|
|---|
| 713 |
|
|---|
| 714 |
|
|---|
| 715 |
|
|---|
| 716 |
|
|---|
| 717 |
|
|---|
| 718 |
|
|---|
| 719 |
|
|---|
| 720 |
static int SeekTableLoad( demux_t *p_demux, demux_sys_t *p_sys ) |
|---|
| 721 |
{ |
|---|
| 722 |
frame_header_t fh; |
|---|
| 723 |
int64_t i_original_pos; |
|---|
| 724 |
uint8_t* p_seek_table; |
|---|
| 725 |
uint8_t* p_kfa_table; |
|---|
| 726 |
int32_t i_seek_elements = 0, i_kfa_elements = 0, j; |
|---|
| 727 |
int64_t i_time, i_offset; |
|---|
| 728 |
int keyframe, last_keyframe = 0, frame = 0, kfa_entry_id = 0; |
|---|
| 729 |
|
|---|
| 730 |
if( p_sys->exh.i_seektable_offset <= 0 ) |
|---|
| 731 |
return VLC_SUCCESS; |
|---|
| 732 |
|
|---|
| 733 |
|
|---|
| 734 |
i_original_pos = stream_Tell( p_demux->s ); |
|---|
| 735 |
#if 0 |
|---|
| 736 |
msg_Dbg( p_demux, "current offset %"PRIi64, i_original_pos ); |
|---|
| 737 |
|
|---|
| 738 |
msg_Dbg( p_demux, "seeking in stream to %"PRIi64, p_sys->exh.i_seektable_offset ); |
|---|
| 739 |
#endif |
|---|
| 740 |
if( stream_Seek( p_demux->s, p_sys->exh.i_seektable_offset ) ) |
|---|
| 741 |
return VLC_EGENERIC; |
|---|
| 742 |
|
|---|
| 743 |
if( FrameHeaderLoad( p_demux, &fh ) ) |
|---|
| 744 |
return VLC_EGENERIC; |
|---|
| 745 |
|
|---|
| 746 |
if( fh.i_type == 'Q' ) |
|---|
| 747 |
{ |
|---|
| 748 |
p_seek_table = malloc( fh.i_length ); |
|---|
| 749 |
if( p_seek_table == NULL ) |
|---|
| 750 |
return VLC_ENOMEM; |
|---|
| 751 |
|
|---|
| 752 |
if( stream_Read( p_demux->s, p_seek_table, fh.i_length ) != fh.i_length ) |
|---|
| 753 |
{ |
|---|
| 754 |
free( p_seek_table ); |
|---|
| 755 |
return VLC_EGENERIC; |
|---|
| 756 |
} |
|---|
| 757 |
|
|---|
| 758 |
i_seek_elements = fh.i_length / 12; |
|---|
| 759 |
} |
|---|
| 760 |
else |
|---|
| 761 |
{ |
|---|
| 762 |
msg_Warn( p_demux, "invalid seektable, frame type=%c", fh.i_type ); |
|---|
| 763 |
stream_Seek( p_demux->s, i_original_pos ); |
|---|
| 764 |
return VLC_EGENERIC; |
|---|
| 765 |
} |
|---|
| 766 |
|
|---|
| 767 |
|
|---|
| 768 |
|
|---|
| 769 |
if( p_sys->exh.i_keyframe_adjust_offset > 0 ) |
|---|
| 770 |
{ |
|---|
| 771 |
msg_Dbg( p_demux, "seeking in stream to %"PRIi64, p_sys->exh.i_keyframe_adjust_offset ); |
|---|
| 772 |
if( stream_Seek( p_demux->s, p_sys->exh.i_keyframe_adjust_offset ) ) |
|---|
| 773 |
{ |
|---|
| 774 |
free( p_seek_table ); |
|---|
| 775 |
return VLC_EGENERIC; |
|---|
| 776 |
} |
|---|
| 777 |
|
|---|
| 778 |
if( FrameHeaderLoad( p_demux, &fh ) ) |
|---|
| 779 |
{ |
|---|
| 780 |
free( p_seek_table ); |
|---|
| 781 |
return VLC_EGENERIC; |
|---|
| 782 |
} |
|---|
| 783 |
|
|---|
| 784 |
if( fh.i_type == 'K' && fh.i_length >= 8 ) |
|---|
| 785 |
{ |
|---|
| 786 |
p_kfa_table = malloc( fh.i_length ); |
|---|
| 787 |
|
|---|
| 788 |
if( p_seek_table == NULL ) |
|---|
| 789 |
{ |
|---|
| 790 |
free( p_seek_table ); |
|---|
| 791 |
return VLC_ENOMEM; |
|---|
| 792 |
} |
|---|
| 793 |
|
|---|
| 794 |
if( stream_Read( p_demux->s, p_kfa_table, fh.i_length ) != fh.i_length ) |
|---|
| 795 |
{ |
|---|
| 796 |
free( p_seek_table ); |
|---|
| 797 |
free( p_kfa_table ); |
|---|
| 798 |
return VLC_EGENERIC; |
|---|
| 799 |
} |
|---|
| 800 |
|
|---|
| 801 |
i_kfa_elements = fh.i_length / 8; |
|---|
| 802 |
} |
|---|
| 803 |
} |
|---|
| 804 |
|
|---|
| 805 |
|
|---|
| 806 |
if( i_kfa_elements > 0 ) |
|---|
| 807 |
msg_Warn( p_demux, "untested keyframe adjust support, upload samples" ); |
|---|
| 808 |
|
|---|
| 809 |
for(j=0; j < i_seek_elements; j++) |
|---|
| 810 |
{ |
|---|
| 811 |
#if 0 |
|---|
| 812 |
uint8_t* p = p_seek_table + j * 12; |
|---|
| 813 |
msg_Dbg( p_demux, "%x %x %x %x %x %x %x %x %x %x %x %x", |
|---|
| 814 |
p[0], p[1], p[2], p[3], p[4], p[5], p[6], p[7], p[8], p[9], p[10], p[11]); |
|---|
| 815 |
#endif |
|---|
| 816 |
keyframe = GetDWLE( p_seek_table + j * 12 + 8 ); |
|---|
| 817 |
|
|---|
| 818 |
frame += (keyframe - last_keyframe) * p_sys->hdr.i_keyframe_distance; |
|---|
| 819 |
|
|---|
| 820 |
if( kfa_entry_id < i_kfa_elements && *(int32_t*)(p_kfa_table + kfa_entry_id * 12 + 4) == j ) |
|---|
| 821 |
{ |
|---|
| 822 |
frame -= *(int32_t*)(p_kfa_table + kfa_entry_id * 12); |
|---|
| 823 |
msg_Dbg( p_demux, "corrected keyframe %d with current frame number %d (corrected with %d)", |
|---|
| 824 |
keyframe, frame, *(int32_t*)(p_kfa_table + kfa_entry_id * 12) ); |
|---|
| 825 |
kfa_entry_id++; |
|---|
| 826 |
} |
|---|
| 827 |
|
|---|
| 828 |
i_time = (double)( (int64_t)frame * 1000000 ) / p_sys->hdr.d_fps; |
|---|
| 829 |
i_offset = GetQWLE( p_seek_table + j * 12 ); |
|---|
| 830 |
|
|---|
| 831 |
if( i_offset == 0 && i_time != 0 ) |
|---|
| 832 |
msg_Dbg( p_demux, "invalid file offset %d %"PRIi64, keyframe, i_offset ); |
|---|
| 833 |
else |
|---|
| 834 |
{ |
|---|
| 835 |
demux_IndexAppend( &p_sys->idx, i_time , i_offset ); |
|---|
| 836 |
#if 0 |
|---|
| 837 |
msg_Dbg( p_demux, "adding entry position %d %"PRIi64 " file offset %"PRIi64, keyframe, i_time, i_offset ); |
|---|
| 838 |
#endif |
|---|
| 839 |
} |
|---|
| 840 |
|
|---|
| 841 |
last_keyframe = keyframe; |
|---|
| 842 |
} |
|---|
| 843 |
|
|---|
| 844 |
p_sys->i_total_frames = (int64_t)frame; |
|---|
| 845 |
|
|---|
| 846 |
p_sys->b_index = true; |
|---|
| 847 |
|
|---|
| 848 |
p_sys->i_total_length = p_sys->i_total_frames * 1000000 / p_sys->hdr.d_fps; |
|---|
| 849 |
|
|---|
| 850 |
msg_Dbg( p_demux, "index table loaded (%d elements)", i_seek_elements ); |
|---|
| 851 |
|
|---|
| 852 |
if( i_kfa_elements ) |
|---|
| 853 |
free ( p_kfa_table ); |
|---|
| 854 |
|
|---|
| 855 |
free ( p_seek_table ); |
|---|
| 856 |
|
|---|
| 857 |
|
|---|
| 858 |
if( stream_Seek( p_demux->s, i_original_pos ) ) |
|---|
| 859 |
return VLC_EGENERIC; |
|---|
| 860 |
|
|---|
| 861 |
return VLC_SUCCESS; |
|---|
| 862 |
|
|---|
| 863 |
} |
|---|
| 864 |
|
|---|
| 865 |
|
|---|
| 866 |
#define DEMUX_INDEX_SIZE_MAX (100000) |
|---|
| 867 |
static void demux_IndexInit( demux_index_t *p_idx ) |
|---|
| 868 |
{ |
|---|
| 869 |
p_idx->i_idx = 0; |
|---|
| 870 |
p_idx->i_idx_max = 0; |
|---|
| 871 |
p_idx->idx = NULL; |
|---|
| 872 |
} |
|---|
| 873 |
static void demux_IndexClean( demux_index_t *p_idx ) |
|---|
| 874 |
{ |
|---|
| 875 |
free( p_idx->idx ); |
|---|
| 876 |
p_idx->idx = NULL; |
|---|
| 877 |
} |
|---|
| 878 |
static void demux_IndexAppend( demux_index_t *p_idx, |
|---|
| 879 |
int64_t i_time, int64_t i_offset ) |
|---|
| 880 |
{ |
|---|
| 881 |
|
|---|
| 882 |
if( p_idx->i_idx > 0 && p_idx->idx[p_idx->i_idx-1].i_time >= i_time ) |
|---|
| 883 |
return; |
|---|
| 884 |
|
|---|
| 885 |
|
|---|
| 886 |
if( p_idx->i_idx >= p_idx->i_idx_max ) |
|---|
| 887 |
{ |
|---|
| 888 |
if( p_idx->i_idx >= DEMUX_INDEX_SIZE_MAX ) |
|---|
| 889 |
{ |
|---|
| 890 |
|
|---|
| 891 |
const int64_t i_length = p_idx->idx[p_idx->i_idx-1].i_time - |
|---|
| 892 |
p_idx->idx[0].i_time; |
|---|
| 893 |
const int i_count = DEMUX_INDEX_SIZE_MAX/2; |
|---|
| 894 |
int i, j; |
|---|
| 895 |
|
|---|
| 896 |
|
|---|
| 897 |
for( i = 1, j = 1; i < p_idx->i_idx; i++ ) |
|---|
| 898 |
{ |
|---|
| 899 |
if( p_idx->idx[i].i_time < j * i_length / i_count ) |
|---|
| 900 |
continue; |
|---|
| 901 |
|
|---|
| 902 |
p_idx->idx[j++] = p_idx->idx[i]; |
|---|
| 903 |
} |
|---|
| 904 |
p_idx->i_idx = j; |
|---|
| 905 |
|
|---|
| 906 |
if( p_idx->i_idx > 3 * DEMUX_INDEX_SIZE_MAX / 4 ) |
|---|
| 907 |
{ |
|---|
| 908 |
|
|---|
| 909 |
|
|---|
| 910 |
for( i = 0; i < p_idx->i_idx/2; i++ ) |
|---|
| 911 |
p_idx->idx[i] = p_idx->idx[2*i]; |
|---|
| 912 |
p_idx->i_idx /= 2; |
|---|
| 913 |
} |
|---|
| 914 |
} |
|---|
| 915 |
else |
|---|
| 916 |
{ |
|---|
| 917 |
p_idx->i_idx_max += 1000; |
|---|
| 918 |
p_idx->idx = realloc( p_idx->idx, |
|---|
| 919 |
p_idx->i_idx_max*sizeof(demux_index_entry_t)); |
|---|
| 920 |
} |
|---|
| 921 |
|---|