mpeg2enc: add --mode option to command line
[profile/ivi/libva.git] / test / encode / mpeg2enc.c
1 /*
2  * Copyright (c) 2012 Intel Corporation. All Rights Reserved.
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the
6  * "Software"), to deal in the Software without restriction, including
7  * without limitation the rights to use, copy, modify, merge, publish,
8  * distribute, sub license, and/or sell copies of the Software, and to
9  * permit persons to whom the Software is furnished to do so, subject to
10  * the following conditions:
11  *
12  * The above copyright notice and this permission notice (including the
13  * next paragraph) shall be included in all copies or substantial portions
14  * of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
17  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
19  * IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR
20  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
21  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
22  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23  */
24 /*
25  * Simple MPEG-2 encoder based on libVA.
26  *
27  */  
28
29 #include "sysdeps.h"
30
31 #include <getopt.h>
32 #include <unistd.h>
33
34 #include <sys/time.h>
35 #include <sys/types.h>
36 #include <fcntl.h>
37 #include <time.h>
38 #include <pthread.h>
39
40 #include <va/va.h>
41 #include <va/va_enc_mpeg2.h>
42
43 #include "va_display.h"
44
45 #define START_CODE_PICUTRE      0x00000100
46 #define START_CODE_SLICE        0x00000101
47 #define START_CODE_USER         0x000001B2
48 #define START_CODE_SEQ          0x000001B3
49 #define START_CODE_EXT          0x000001B5
50 #define START_CODE_GOP          0x000001B8
51
52 #define CHROMA_FORMAT_RESERVED  0
53 #define CHROMA_FORMAT_420       1
54 #define CHROMA_FORMAT_422       2
55 #define CHROMA_FORMAT_444       3
56
57 #define MAX_SLICES              128
58
59 #define CHECK_VASTATUS(va_status, func)                                 \
60     if (va_status != VA_STATUS_SUCCESS) {                               \
61         fprintf(stderr, "%s:%s (%d) failed, exit\n", __func__, func, __LINE__); \
62         exit(1);                                                        \
63     }
64
65 #define INTRA_PERIOD            30
66
67 static int const picture_type_patter[][2] = {{VAEncPictureTypeIntra, 1}, 
68                                              {VAEncPictureTypePredictive, 3}, {VAEncPictureTypePredictive, 3},{VAEncPictureTypePredictive, 3},
69                                              {VAEncPictureTypePredictive, 3}, {VAEncPictureTypePredictive, 3},{VAEncPictureTypePredictive, 3},
70                                              {VAEncPictureTypePredictive, 3}, {VAEncPictureTypePredictive, 3},{VAEncPictureTypePredictive, 3},
71                                              {VAEncPictureTypePredictive, 2}};
72
73 struct mpeg2enc_context {
74     /* args */
75     int rate_control_mode;
76     int fps;
77     int mode; /* 0:I, 1:I/P, 2:I/P/B */
78     int width;
79     int height;
80     int frame_size;
81     int num_pictures;
82     int qp;
83     FILE *ifp;
84     FILE *ofp;
85     unsigned char *frame_data_buffer;
86     int intra_period;
87     int bit_rate; /* in kbps */
88
89     /* VA resource */
90     VADisplay va_dpy;
91     VAEncSequenceParameterBufferMPEG2 seq_param;
92     VAEncPictureParameterBufferMPEG2 pic_param;
93     VAEncSliceParameterBufferMPEG2 slice_param[MAX_SLICES];
94     VAContextID context_id;
95     VAConfigID config_id;
96     VABufferID seq_param_buf_id;                /* Sequence level parameter */
97     VABufferID pic_param_buf_id;                /* Picture level parameter */
98     VABufferID slice_param_buf_id[MAX_SLICES];  /* Slice level parameter, multil slices */
99     VABufferID codedbuf_buf_id;                 /* Output buffer, compressed data */
100     VABufferID packed_seq_header_param_buf_id;
101     VABufferID packed_seq_buf_id;
102     VABufferID packed_pic_header_param_buf_id;
103     VABufferID packed_pic_buf_id;
104     int num_slice_groups;
105     int codedbuf_i_size;
106     int codedbuf_pb_size;
107
108     /* thread */
109     pthread_t upload_thread_id;
110     int upload_thread_value;
111     int current_input_surface;
112     int current_upload_surface;
113 };
114
115 /*
116  * mpeg2enc helpers
117  */
118 #define BITSTREAM_ALLOCATE_STEPPING     4096
119
120 struct __bitstream {
121     unsigned int *buffer;
122     int bit_offset;
123     int max_size_in_dword;
124 };
125
126 typedef struct __bitstream bitstream;
127
128 static unsigned int 
129 swap32(unsigned int val)
130 {
131     unsigned char *pval = (unsigned char *)&val;
132
133     return ((pval[0] << 24)     |
134             (pval[1] << 16)     |
135             (pval[2] << 8)      |
136             (pval[3] << 0));
137 }
138
139 static void
140 bitstream_start(bitstream *bs)
141 {
142     bs->max_size_in_dword = BITSTREAM_ALLOCATE_STEPPING;
143     bs->buffer = calloc(bs->max_size_in_dword * sizeof(int), 1);
144     bs->bit_offset = 0;
145 }
146
147 static void
148 bitstream_end(bitstream *bs)
149 {
150     int pos = (bs->bit_offset >> 5);
151     int bit_offset = (bs->bit_offset & 0x1f);
152     int bit_left = 32 - bit_offset;
153
154     if (bit_offset) {
155         bs->buffer[pos] = swap32((bs->buffer[pos] << bit_left));
156     }
157 }
158  
159 static void
160 bitstream_put_ui(bitstream *bs, unsigned int val, int size_in_bits)
161 {
162     int pos = (bs->bit_offset >> 5);
163     int bit_offset = (bs->bit_offset & 0x1f);
164     int bit_left = 32 - bit_offset;
165
166     if (!size_in_bits)
167         return;
168
169     if (size_in_bits < 32)
170         val &= ((1 << size_in_bits) - 1);
171
172     bs->bit_offset += size_in_bits;
173
174     if (bit_left > size_in_bits) {
175         bs->buffer[pos] = (bs->buffer[pos] << size_in_bits | val);
176     } else {
177         size_in_bits -= bit_left;
178         bs->buffer[pos] = (bs->buffer[pos] << bit_left) | (val >> size_in_bits);
179         bs->buffer[pos] = swap32(bs->buffer[pos]);
180
181         if (pos + 1 == bs->max_size_in_dword) {
182             bs->max_size_in_dword += BITSTREAM_ALLOCATE_STEPPING;
183             bs->buffer = realloc(bs->buffer, bs->max_size_in_dword * sizeof(unsigned int));
184         }
185
186         bs->buffer[pos + 1] = val;
187     }
188 }
189
190 static void
191 bitstream_byte_aligning(bitstream *bs, int bit)
192 {
193     int bit_offset = (bs->bit_offset & 0x7);
194     int bit_left = 8 - bit_offset;
195     int new_val;
196
197     if (!bit_offset)
198         return;
199
200     assert(bit == 0 || bit == 1);
201
202     if (bit)
203         new_val = (1 << bit_left) - 1;
204     else
205         new_val = 0;
206
207     bitstream_put_ui(bs, new_val, bit_left);
208 }
209
210 static struct mpeg2_frame_rate {
211     int code;
212     float value;
213 } frame_rate_tab[] = {
214     {1, 23.976},
215     {2, 24.0},
216     {3, 25.0},
217     {4, 29.97},
218     {5, 30},
219     {6, 50},
220     {7, 59.94},
221     {8, 60}
222 };
223
224 static int
225 find_frame_rate_code(const VAEncSequenceParameterBufferMPEG2 *seq_param)
226 {
227     unsigned int delta = -1;
228     int code = 1, i;
229     float frame_rate_value = seq_param->frame_rate * 
230         (seq_param->sequence_extension.bits.frame_rate_extension_d + 1) / 
231         (seq_param->sequence_extension.bits.frame_rate_extension_n + 1);
232
233     for (i = 0; i < sizeof(frame_rate_tab) / sizeof(frame_rate_tab[0]); i++) {
234
235         if (abs(1000 * frame_rate_tab[i].value - 1000 * frame_rate_value) < delta) {
236             code = frame_rate_tab[i].code;
237             delta = abs(1000 * frame_rate_tab[i].value - 1000 * frame_rate_value);
238         }
239     }
240
241     return code;
242 }
243
244 static void 
245 sps_rbsp(const VAEncSequenceParameterBufferMPEG2 *seq_param,
246          bitstream *bs)
247 {
248     int frame_rate_code = find_frame_rate_code(seq_param);
249
250     bitstream_put_ui(bs, START_CODE_SEQ, 32);
251     bitstream_put_ui(bs, seq_param->picture_width, 12);
252     bitstream_put_ui(bs, seq_param->picture_height, 12);
253     bitstream_put_ui(bs, seq_param->aspect_ratio_information, 4);
254     bitstream_put_ui(bs, frame_rate_code, 4); /* frame_rate_code */
255     bitstream_put_ui(bs, (seq_param->bits_per_second + 399) / 400, 18); /* the low 18 bits of bit_rate */
256     bitstream_put_ui(bs, 1, 1); /* marker_bit */
257     bitstream_put_ui(bs, seq_param->vbv_buffer_size, 10);
258     bitstream_put_ui(bs, 0, 1); /* constraint_parameter_flag, always 0 for MPEG-2 */
259     bitstream_put_ui(bs, 0, 1); /* load_intra_quantiser_matrix */
260     bitstream_put_ui(bs, 0, 1); /* load_non_intra_quantiser_matrix */
261
262     bitstream_byte_aligning(bs, 0);
263
264     bitstream_put_ui(bs, START_CODE_EXT, 32);
265     bitstream_put_ui(bs, 1, 4); /* sequence_extension id */
266     bitstream_put_ui(bs, seq_param->sequence_extension.bits.profile_and_level_indication, 8);
267     bitstream_put_ui(bs, seq_param->sequence_extension.bits.progressive_sequence, 1);
268     bitstream_put_ui(bs, seq_param->sequence_extension.bits.chroma_format, 2);
269     bitstream_put_ui(bs, seq_param->picture_width >> 12, 2);
270     bitstream_put_ui(bs, seq_param->picture_height >> 12, 2);
271     bitstream_put_ui(bs, ((seq_param->bits_per_second + 399) / 400) >> 18, 12); /* bit_rate_extension */
272     bitstream_put_ui(bs, 1, 1); /* marker_bit */
273     bitstream_put_ui(bs, seq_param->vbv_buffer_size >> 10, 8);
274     bitstream_put_ui(bs, seq_param->sequence_extension.bits.low_delay, 1);
275     bitstream_put_ui(bs, seq_param->sequence_extension.bits.frame_rate_extension_n, 2);
276     bitstream_put_ui(bs, seq_param->sequence_extension.bits.frame_rate_extension_d, 5);
277
278     bitstream_byte_aligning(bs, 0);
279
280     bitstream_put_ui(bs, START_CODE_GOP, 32);
281     bitstream_put_ui(bs, seq_param->gop_header.bits.time_code, 25);
282     bitstream_put_ui(bs, seq_param->gop_header.bits.closed_gop, 1);
283     bitstream_put_ui(bs, seq_param->gop_header.bits.broken_link, 1);
284
285     bitstream_byte_aligning(bs, 0);
286 }
287
288 static void 
289 pps_rbsp(const VAEncSequenceParameterBufferMPEG2 *seq_param,
290          const VAEncPictureParameterBufferMPEG2 *pic_param,
291          bitstream *bs)
292 {
293     int i;
294     int chroma_420_type;
295
296     if (seq_param->sequence_extension.bits.chroma_format == CHROMA_FORMAT_420)
297         chroma_420_type = pic_param->picture_coding_extension.bits.progressive_frame;
298     else
299         chroma_420_type = 0;
300
301     bitstream_put_ui(bs, START_CODE_PICUTRE, 32);
302     bitstream_put_ui(bs, pic_param->temporal_reference, 10);
303     bitstream_put_ui(bs,
304                      pic_param->picture_type == VAEncPictureTypeIntra ? 1 :
305                      pic_param->picture_type == VAEncPictureTypePredictive ? 2 : 3,
306                      3);
307     bitstream_put_ui(bs, 0xFFFF, 16); /* vbv_delay, always 0xFFFF */
308     
309     if (pic_param->picture_type == VAEncPictureTypePredictive) {
310         bitstream_put_ui(bs, 0, 1); /* full_pel_forward_vector, always 0 for MPEG-2 */
311         bitstream_put_ui(bs, 7, 3); /* forward_f_code, always 7 for MPEG-2 */
312     }
313
314     if (pic_param->picture_type == VAEncPictureTypeBidirectional) {
315         bitstream_put_ui(bs, 0, 1); /* full_pel_backward_vector, always 0 for MPEG-2 */
316         bitstream_put_ui(bs, 7, 3); /* backward_f_code, always 7 for MPEG-2 */
317     }
318      
319     bitstream_put_ui(bs, 0, 1); /* extra_bit_picture, 0 */
320
321     bitstream_byte_aligning(bs, 0);
322
323     bitstream_put_ui(bs, START_CODE_EXT, 32);
324     bitstream_put_ui(bs, 8, 4); /* Picture Coding Extension ID: 8 */
325     bitstream_put_ui(bs, pic_param->f_code[0][0], 4);
326     bitstream_put_ui(bs, pic_param->f_code[0][1], 4);
327     bitstream_put_ui(bs, pic_param->f_code[1][0], 4);
328     bitstream_put_ui(bs, pic_param->f_code[1][1], 4);
329
330     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.intra_dc_precision, 2);
331     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.picture_structure, 2);
332     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.top_field_first, 1);
333     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.frame_pred_frame_dct, 1);
334     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.concealment_motion_vectors, 1);
335     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.q_scale_type, 1);
336     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.intra_vlc_format, 1);
337     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.alternate_scan, 1);
338     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.repeat_first_field, 1);
339     bitstream_put_ui(bs, chroma_420_type, 1);
340     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.progressive_frame, 1);
341     bitstream_put_ui(bs, pic_param->picture_coding_extension.bits.composite_display_flag, 1);
342
343     bitstream_byte_aligning(bs, 0);
344
345     if (pic_param->user_data_length) {
346         bitstream_put_ui(bs, START_CODE_USER, 32);
347
348         for (i = 0; i < pic_param->user_data_length; i++) {
349             bitstream_put_ui(bs, pic_param->user_data[i], 8);
350         }
351
352         bitstream_byte_aligning(bs, 0);
353     }
354 }
355
356 static int
357 build_packed_pic_buffer(const VAEncSequenceParameterBufferMPEG2 *seq_param,
358                         const VAEncPictureParameterBufferMPEG2 *pic_param,
359                         unsigned char **header_buffer)
360 {
361     bitstream bs;
362
363     bitstream_start(&bs);
364     pps_rbsp(seq_param, pic_param, &bs);
365     bitstream_end(&bs);
366
367     *header_buffer = (unsigned char *)bs.buffer;
368     return bs.bit_offset;
369 }
370
371 static int
372 build_packed_seq_buffer(const VAEncSequenceParameterBufferMPEG2 *seq_param,
373                         unsigned char **header_buffer)
374 {
375     bitstream bs;
376
377     bitstream_start(&bs);
378     sps_rbsp(seq_param, &bs);
379     bitstream_end(&bs);
380
381     *header_buffer = (unsigned char *)bs.buffer;
382     return bs.bit_offset;
383 }
384
385 /*
386  * mpeg2enc
387  */
388 #define SID_INPUT_PICTURE_0                     0
389 #define SID_INPUT_PICTURE_1                     1
390 #define SID_REFERENCE_PICTURE_L0                2
391 #define SID_REFERENCE_PICTURE_L1                3
392 #define SID_RECON_PICTURE                       4
393 #define SID_NUMBER                              SID_RECON_PICTURE + 1
394
395 static VASurfaceID surface_ids[SID_NUMBER];
396
397 /*
398  * upload thread function
399  */
400 static void *
401 upload_yuv_to_surface(void *data)
402 {
403     struct mpeg2enc_context *ctx = data;
404     VAImage surface_image;
405     VAStatus va_status;
406     void *surface_p = NULL;
407     unsigned char *y_src, *u_src, *v_src;
408     unsigned char *y_dst, *u_dst, *v_dst;
409     int y_size = ctx->width * ctx->height;
410     int u_size = (ctx->width >> 1) * (ctx->height >> 1);
411     int row, col;
412     size_t n_items;
413
414     do {
415         n_items = fread(ctx->frame_data_buffer, ctx->frame_size, 1, ctx->ifp);
416     } while (n_items != 1);
417
418     va_status = vaDeriveImage(ctx->va_dpy, surface_ids[ctx->current_upload_surface], &surface_image);
419     CHECK_VASTATUS(va_status,"vaDeriveImage");
420
421     vaMapBuffer(ctx->va_dpy, surface_image.buf, &surface_p);
422     assert(VA_STATUS_SUCCESS == va_status);
423         
424     y_src = ctx->frame_data_buffer;
425     u_src = ctx->frame_data_buffer + y_size; /* UV offset for NV12 */
426     v_src = ctx->frame_data_buffer + y_size + u_size;
427
428     y_dst = surface_p + surface_image.offsets[0];
429     u_dst = surface_p + surface_image.offsets[1]; /* UV offset for NV12 */
430     v_dst = surface_p + surface_image.offsets[2];
431
432     /* Y plane */
433     for (row = 0; row < surface_image.height; row++) {
434         memcpy(y_dst, y_src, surface_image.width);
435         y_dst += surface_image.pitches[0];
436         y_src += ctx->width;
437     }
438
439     if (surface_image.format.fourcc == VA_FOURCC_NV12) { /* UV plane */
440         for (row = 0; row < surface_image.height / 2; row++) {
441             for (col = 0; col < surface_image.width / 2; col++) {
442                 u_dst[col * 2] = u_src[col];
443                 u_dst[col * 2 + 1] = v_src[col];
444             }
445
446             u_dst += surface_image.pitches[1];
447             u_src += (ctx->width / 2);
448             v_src += (ctx->width / 2);
449         }
450     } else {
451         for (row = 0; row < surface_image.height / 2; row++) {
452             for (col = 0; col < surface_image.width / 2; col++) {
453                 u_dst[col] = u_src[col];
454                 v_dst[col] = v_src[col];
455             }
456
457             u_dst += surface_image.pitches[1];
458             v_dst += surface_image.pitches[2];
459             u_src += (ctx->width / 2);
460             v_src += (ctx->width / 2);
461         }
462     }
463
464     vaUnmapBuffer(ctx->va_dpy, surface_image.buf);
465     vaDestroyImage(ctx->va_dpy, surface_image.image_id);
466
467     return NULL;
468 }
469
470 static void 
471 mpeg2enc_exit(struct mpeg2enc_context *ctx, int exit_code)
472 {
473     if (ctx->frame_data_buffer) {
474         free(ctx->frame_data_buffer);
475         ctx->frame_data_buffer = NULL;
476     }
477
478     if (ctx->ifp) {
479         fclose(ctx->ifp);
480         ctx->ifp = NULL;
481     }
482
483     if (ctx->ofp) {
484         fclose(ctx->ofp);
485         ctx->ofp = NULL;
486     }
487
488     exit(exit_code);
489 }
490
491 static void 
492 usage(char *program)
493 {   
494     fprintf(stderr, "Usage: %s --help\n", program);
495     fprintf(stderr, "\t--help   print this message\n");
496     fprintf(stderr, "Usage: %s <width> <height> <ifile> <ofile> [options]\n", program);
497     fprintf(stderr, "\t<width>  specifies the frame width\n");
498     fprintf(stderr, "\t<height> specifies the frame height\n");
499     fprintf(stderr, "\t<ifile>  specifies the I420/IYUV YUV file\n");
500     fprintf(stderr, "\t<ofile>  specifies the encoded MPEG-2 file\n");
501     fprintf(stderr, "where options include:\n");
502     fprintf(stderr, "\t--cqp <QP>       const qp mode with specified <QP>\n");
503     fprintf(stderr, "\t--fps <FPS>      specify the frame rate\n");
504     fprintf(stderr, "\t--mode <MODE>    specify the mode 0 (I), 1 (I/P) and 2 (I/P/B)\n");
505 }
506
507 static void 
508 parse_args(struct mpeg2enc_context *ctx, int argc, char **argv)
509 {
510     int c, tmp, i = 0;
511     int option_index = 0;
512     long file_size;
513     static struct option long_options[] = {
514         {"help",        no_argument,            0,      'h'},
515         {"cqp",         required_argument,      0,      'c'},
516         {"fps",         required_argument,      0,      'f'},
517         {"mode",        required_argument,      0,      'm'},
518         { NULL,         0,                      NULL,   0 }
519     };
520
521     if ((argc == 2 && strcmp(argv[1], "--help") == 0) ||
522         (argc < 5))
523         goto print_usage;
524
525     ctx->width = atoi(argv[1]);
526     ctx->height = atoi(argv[2]);
527
528     if (ctx->width <= 0 || ctx->height <= 0) {
529         fprintf(stderr, "<width> and <height> must be greater than 0\n");
530         goto err_exit;
531     }
532
533     ctx->ifp = fopen(argv[3], "rb");
534
535     if (ctx->ifp == NULL) {
536         fprintf(stderr, "Can't open the input file\n");
537         goto err_exit;
538     }
539
540     fseek(ctx->ifp, 0l, SEEK_END);
541     file_size = ftell(ctx->ifp);
542     ctx->frame_size = ctx->width * ctx->height * 3 / 2;
543
544     if ((file_size < ctx->frame_size) ||
545         (file_size % ctx->frame_size)) {
546         fprintf(stderr, "The input file size %ld isn't a multiple of the frame size %d\n", file_size, ctx->frame_size);
547         goto err_exit;
548     }
549
550     ctx->num_pictures = file_size / ctx->frame_size;
551     fseek(ctx->ifp, 0l, SEEK_SET);
552     
553     ctx->ofp = fopen(argv[4], "wb");
554     
555     if (ctx->ofp == NULL) {
556         fprintf(stderr, "Can't create the output file\n");
557         goto err_exit;
558     }
559
560     opterr = 0;
561     ctx->fps = 30;
562     ctx->qp = 28;
563     ctx->rate_control_mode = VA_RC_CQP;
564     ctx->mode = 1;
565
566     optind = 5;
567
568     while((c = getopt_long(argc, argv,
569                            "",
570                            long_options, 
571                            &option_index)) != -1) {
572         switch(c) {
573         case 'c':
574             tmp = atoi(optarg);
575
576             /* only support q_scale_type = 0 */
577             if (tmp > 62 || tmp < 2) {
578                 fprintf(stderr, "Warning: QP must be in [2, 62]\n");
579
580                 if (tmp > 62)
581                     tmp = 62;
582
583                 if (tmp < 2)
584                     tmp = 2;
585             }
586
587             ctx->qp = tmp & 0xFE;
588             ctx->rate_control_mode = VA_RC_CQP;
589
590             break;
591
592         case 'f':
593             tmp = atoi(optarg);
594
595             if (tmp <= 0)
596                 fprintf(stderr, "Warning: FPS must be greater than 0\n");
597             else
598                 ctx->fps = tmp;
599
600             ctx->rate_control_mode = VA_RC_CBR;
601
602             break;
603
604         case 'm':
605             tmp = atoi(optarg);
606             
607             if (tmp < 0 || tmp > 2)
608                 fprintf(stderr, "Waning: MODE must be 0, 1, or 2\n");
609             else
610                 ctx->mode = tmp;
611
612             break;
613
614         case '?':
615             fprintf(stderr, "Error: unkown command options\n");
616
617         case 'h':
618             goto print_usage;
619         }
620     }
621
622     return;
623
624 print_usage:    
625     usage(argv[0]);
626 err_exit:
627     mpeg2enc_exit(ctx, 1);
628 }
629
630 /*
631  * init
632  */
633 static void
634 mpeg2enc_init_sequence_parameter(struct mpeg2enc_context *ctx,
635                                 VAEncSequenceParameterBufferMPEG2 *seq_param)
636 {
637     seq_param->intra_period = ctx->intra_period;
638     seq_param->ip_period = 0;   /* FIXME: ??? */
639     seq_param->picture_width = ctx->width;
640     seq_param->picture_height = ctx->height;
641
642     if (ctx->bit_rate > 0)
643         seq_param->bits_per_second = 1024 * ctx->bit_rate; /* use kbps as input */
644     else
645         seq_param->bits_per_second = 0x3FFFF * 400;
646
647     seq_param->frame_rate = ctx->fps;
648     seq_param->aspect_ratio_information = 1;
649     seq_param->vbv_buffer_size = 3; /* B = 16 * 1024 * vbv_buffer_size */
650
651     seq_param->sequence_extension.bits.profile_and_level_indication = 4 << 4 | 0x8; /* FIXME: Main/Main ??? */
652     seq_param->sequence_extension.bits.progressive_sequence = 1; /* progressive frame-pictures */
653     seq_param->sequence_extension.bits.chroma_format = CHROMA_FORMAT_420; /* 4:2:0 */
654     seq_param->sequence_extension.bits.low_delay = 0; /* FIXME */
655     seq_param->sequence_extension.bits.frame_rate_extension_n = 0;
656     seq_param->sequence_extension.bits.frame_rate_extension_d = 0;
657
658     seq_param->gop_header.bits.time_code = (1 << 12); /* bit12: marker_bit */
659     seq_param->gop_header.bits.closed_gop = 0;
660     seq_param->gop_header.bits.broken_link = 0;    
661 }
662
663 static void
664 mpeg2enc_init_picture_parameter(struct mpeg2enc_context *ctx,
665                                VAEncPictureParameterBufferMPEG2 *pic_param)
666 {
667     pic_param->forward_reference_picture = VA_INVALID_ID;
668     pic_param->backward_reference_picture = VA_INVALID_ID;
669     pic_param->reconstructed_picture = VA_INVALID_ID;
670     pic_param->coded_buf = VA_INVALID_ID;
671     pic_param->picture_type = VAEncPictureTypeIntra;
672
673     pic_param->temporal_reference = 0;
674     pic_param->f_code[0][0] = 0xf;
675     pic_param->f_code[0][1] = 0xf;
676     pic_param->f_code[1][0] = 0xf;
677     pic_param->f_code[1][1] = 0xf;
678
679     pic_param->picture_coding_extension.bits.intra_dc_precision = 0; /* 8bits */
680     pic_param->picture_coding_extension.bits.picture_structure = 3; /* frame picture */
681     pic_param->picture_coding_extension.bits.top_field_first = 0; 
682     pic_param->picture_coding_extension.bits.frame_pred_frame_dct = 1; /* FIXME */
683     pic_param->picture_coding_extension.bits.concealment_motion_vectors = 0;
684     pic_param->picture_coding_extension.bits.q_scale_type = 0;
685     pic_param->picture_coding_extension.bits.intra_vlc_format = 0;
686     pic_param->picture_coding_extension.bits.alternate_scan = 0;
687     pic_param->picture_coding_extension.bits.repeat_first_field = 0;
688     pic_param->picture_coding_extension.bits.progressive_frame = 1;
689     pic_param->picture_coding_extension.bits.composite_display_flag = 0;
690
691     pic_param->user_data_length = 0;
692 }
693
694 static void 
695 mpeg2enc_alloc_va_resources(struct mpeg2enc_context *ctx)
696 {
697     VAEntrypoint *entrypoint_list;
698     VAConfigAttrib attrib_list[2];
699     VAStatus va_status;
700     int max_entrypoints, num_entrypoints, entrypoint;
701     int major_ver, minor_ver;
702
703     ctx->va_dpy = va_open_display();
704     va_status = vaInitialize(ctx->va_dpy,
705                              &major_ver,
706                              &minor_ver);
707     CHECK_VASTATUS(va_status, "vaInitialize");
708
709     max_entrypoints = vaMaxNumEntrypoints(ctx->va_dpy);
710     entrypoint_list = malloc(max_entrypoints * sizeof(VAEntrypoint));
711     vaQueryConfigEntrypoints(ctx->va_dpy,
712                              VAProfileMPEG2Simple,
713                              entrypoint_list,
714                              &num_entrypoints);
715
716     for (entrypoint = 0; entrypoint < num_entrypoints; entrypoint++) {
717         if (entrypoint_list[entrypoint] == VAEntrypointEncSlice)
718             break;
719     }
720
721     free(entrypoint_list);
722
723     if (entrypoint == num_entrypoints) {
724         /* not find Slice entry point */
725         assert(0);
726     }
727
728     /* find out the format for the render target, and rate control mode */
729     attrib_list[0].type = VAConfigAttribRTFormat;
730     attrib_list[1].type = VAConfigAttribRateControl;
731     vaGetConfigAttributes(ctx->va_dpy,
732                           VAProfileMPEG2Simple,
733                           VAEntrypointEncSlice,
734                           &attrib_list[0],
735                           2);
736
737     if ((attrib_list[0].value & VA_RT_FORMAT_YUV420) == 0) {
738         /* not find desired YUV420 RT format */
739         assert(0);
740     }
741
742     if ((attrib_list[1].value & ctx->rate_control_mode) == 0) {
743         /* Can't find matched RC mode */
744         fprintf(stderr, "RC mode %d isn't found, exit\n", ctx->rate_control_mode);
745         assert(0);
746     }
747
748     attrib_list[0].value = VA_RT_FORMAT_YUV420; /* set to desired RT format */
749     attrib_list[1].value = ctx->rate_control_mode; /* set to desired RC mode */
750
751     va_status = vaCreateConfig(ctx->va_dpy,
752                                VAProfileMPEG2Simple,
753                                VAEntrypointEncSlice,
754                                attrib_list,
755                                2,
756                                &ctx->config_id);
757     CHECK_VASTATUS(va_status, "vaCreateConfig");
758
759     /* Create a context for this decode pipe */
760     va_status = vaCreateContext(ctx->va_dpy,
761                                 ctx->config_id,
762                                 ctx->width,
763                                 ctx->height,
764                                 VA_PROGRESSIVE, 
765                                 0,
766                                 0,
767                                 &ctx->context_id);
768     CHECK_VASTATUS(va_status, "vaCreateContext");
769
770     va_status = vaCreateSurfaces(ctx->va_dpy,
771                                  VA_RT_FORMAT_YUV420,
772                                  ctx->width,
773                                  ctx->height,
774                                  surface_ids,
775                                  SID_NUMBER,
776                                  NULL,
777                                  0);
778     CHECK_VASTATUS(va_status, "vaCreateSurfaces");
779 }
780
781 static void 
782 mpeg2enc_init(struct mpeg2enc_context *ctx)
783 {
784     int i;
785
786     ctx->frame_data_buffer = (unsigned char *)malloc(ctx->frame_size);
787     ctx->seq_param_buf_id = VA_INVALID_ID;
788     ctx->pic_param_buf_id = VA_INVALID_ID;
789     ctx->packed_seq_header_param_buf_id = VA_INVALID_ID;
790     ctx->packed_seq_buf_id = VA_INVALID_ID;
791     ctx->packed_pic_header_param_buf_id = VA_INVALID_ID;
792     ctx->packed_pic_buf_id = VA_INVALID_ID;
793     ctx->codedbuf_buf_id = VA_INVALID_ID;
794     ctx->codedbuf_i_size = ctx->frame_size;
795     ctx->codedbuf_pb_size = 0;
796     ctx->intra_period = INTRA_PERIOD;
797     ctx->bit_rate = -1;
798
799     for (i = 0; i < MAX_SLICES; i++) {
800         ctx->slice_param_buf_id[i] = VA_INVALID_ID;
801     }
802
803     mpeg2enc_init_sequence_parameter(ctx, &ctx->seq_param);
804     mpeg2enc_init_picture_parameter(ctx, &ctx->pic_param);
805     mpeg2enc_alloc_va_resources(ctx);
806
807     /* thread */
808     ctx->current_input_surface = SID_INPUT_PICTURE_0;
809     ctx->current_upload_surface = SID_INPUT_PICTURE_1;
810     ctx->upload_thread_value = pthread_create(&ctx->upload_thread_id,
811                                               NULL,
812                                               upload_yuv_to_surface,
813                                               ctx);
814 }
815
816 /*
817  * run
818  */
819 static void 
820 mpeg2enc_update_picture_parameter(struct mpeg2enc_context *ctx,
821                                  VAEncPictureType picture_type,
822                                  int coded_order,
823                                  int display_order)
824 {
825     VAEncPictureParameterBufferMPEG2 *pic_param;
826     VAStatus va_status;
827
828     // Picture level
829     pic_param = &ctx->pic_param;
830     pic_param->picture_type = picture_type;
831     pic_param->temporal_reference = display_order;
832     pic_param->reconstructed_picture = surface_ids[SID_RECON_PICTURE];
833     pic_param->forward_reference_picture = surface_ids[SID_REFERENCE_PICTURE_L0];
834     pic_param->backward_reference_picture = surface_ids[SID_REFERENCE_PICTURE_L1];
835     pic_param->coded_buf = ctx->codedbuf_buf_id;
836
837     va_status = vaCreateBuffer(ctx->va_dpy,
838                                ctx->context_id,
839                                VAEncPictureParameterBufferType,
840                                sizeof(*pic_param),
841                                1,
842                                pic_param,
843                                &ctx->pic_param_buf_id);
844     CHECK_VASTATUS(va_status, "vaCreateBuffer");
845 }
846
847 static void 
848 mpeg2enc_update_slice_parameter(struct mpeg2enc_context *ctx, VAEncPictureType picture_type)
849 {
850     VAEncSequenceParameterBufferMPEG2 *seq_param;
851     VAEncPictureParameterBufferMPEG2 *pic_param;
852     VAEncSliceParameterBufferMPEG2 *slice_param;
853     VAStatus va_status;
854     int i, width_in_mbs, height_in_mbs;
855
856     pic_param = &ctx->pic_param;
857     assert(pic_param->picture_coding_extension.bits.q_scale_type == 0);
858
859     seq_param = &ctx->seq_param;
860     width_in_mbs = (seq_param->picture_width + 15) / 16;
861     height_in_mbs = (seq_param->picture_height + 15) / 16;
862     ctx->num_slice_groups = 1;
863
864     for (i = 0; i < height_in_mbs; i++) {
865         slice_param = &ctx->slice_param[i];
866         slice_param->macroblock_address = i * width_in_mbs;
867         slice_param->num_macroblocks = width_in_mbs;
868         slice_param->is_intra_slice = (picture_type == VAEncPictureTypeIntra);
869         slice_param->quantiser_scale_code = ctx->qp / 2;
870     }
871
872     va_status = vaCreateBuffer(ctx->va_dpy,
873                                ctx->context_id,
874                                VAEncSliceParameterBufferType,
875                                sizeof(*slice_param),
876                                height_in_mbs,
877                                ctx->slice_param,
878                                ctx->slice_param_buf_id);
879     CHECK_VASTATUS(va_status, "vaCreateBuffer");;
880 }
881
882 static int 
883 begin_picture(struct mpeg2enc_context *ctx,
884               int coded_order,
885               int display_order,
886               VAEncPictureType picture_type)
887 {
888     VAStatus va_status;
889     int tmp;
890     VAEncPackedHeaderParameterBuffer packed_header_param_buffer;
891     unsigned int length_in_bits;
892     unsigned char *packed_seq_buffer = NULL, *packed_pic_buffer = NULL;
893
894     if (ctx->upload_thread_value != 0) {
895         fprintf(stderr, "FATAL error!!!\n");
896         exit(1);
897     }
898     
899     pthread_join(ctx->upload_thread_id, NULL);
900
901     ctx->upload_thread_value = -1;
902     tmp = ctx->current_input_surface;
903     ctx->current_input_surface = ctx->current_upload_surface;
904     ctx->current_upload_surface = tmp;
905
906     if (coded_order == 0) {
907         assert(picture_type == VAEncPictureTypeIntra);
908         length_in_bits = build_packed_seq_buffer(&ctx->seq_param, &packed_seq_buffer);
909         packed_header_param_buffer.type = VAEncPackedHeaderMPEG2_SPS;
910         packed_header_param_buffer.has_emulation_bytes = 0;
911         packed_header_param_buffer.bit_length = length_in_bits;
912         va_status = vaCreateBuffer(ctx->va_dpy,
913                                    ctx->context_id,
914                                    VAEncPackedHeaderParameterBufferType,
915                                    sizeof(packed_header_param_buffer), 1, &packed_header_param_buffer,
916                                    &ctx->packed_seq_header_param_buf_id);
917         CHECK_VASTATUS(va_status,"vaCreateBuffer");
918
919         va_status = vaCreateBuffer(ctx->va_dpy,
920                                    ctx->context_id,
921                                    VAEncPackedHeaderDataBufferType,
922                                    (length_in_bits + 7) / 8, 1, packed_seq_buffer,
923                                    &ctx->packed_seq_buf_id);
924         CHECK_VASTATUS(va_status,"vaCreateBuffer");
925
926         free(packed_seq_buffer);
927     }
928
929     length_in_bits = build_packed_pic_buffer(&ctx->seq_param, &ctx->pic_param, &packed_pic_buffer);
930     packed_header_param_buffer.type = VAEncPackedHeaderMPEG2_PPS;
931     packed_header_param_buffer.has_emulation_bytes = 0;
932     packed_header_param_buffer.bit_length = length_in_bits;
933
934     va_status = vaCreateBuffer(ctx->va_dpy,
935                                ctx->context_id,
936                                VAEncPackedHeaderParameterBufferType,
937                                sizeof(packed_header_param_buffer), 1, &packed_header_param_buffer,
938                                &ctx->packed_pic_header_param_buf_id);
939     CHECK_VASTATUS(va_status,"vaCreateBuffer");
940
941     va_status = vaCreateBuffer(ctx->va_dpy,
942                                ctx->context_id,
943                                VAEncPackedHeaderDataBufferType,
944                                (length_in_bits + 7) / 8, 1, packed_pic_buffer,
945                                &ctx->packed_pic_buf_id);
946     CHECK_VASTATUS(va_status,"vaCreateBuffer");
947
948     free(packed_pic_buffer);
949
950     /* sequence parameter set */
951     VAEncSequenceParameterBufferMPEG2 *seq_param = &ctx->seq_param;
952     va_status = vaCreateBuffer(ctx->va_dpy,
953                                ctx->context_id,
954                                VAEncSequenceParameterBufferType,
955                                sizeof(*seq_param),
956                                1,
957                                seq_param,
958                                &ctx->seq_param_buf_id);
959     CHECK_VASTATUS(va_status,"vaCreateBuffer");;
960
961     /* slice parameter */
962     mpeg2enc_update_slice_parameter(ctx, picture_type);
963
964     return 0;
965 }
966
967 static int 
968 mpeg2enc_render_picture(struct mpeg2enc_context *ctx)
969 {
970     VAStatus va_status;
971     VABufferID va_buffers[16];
972     unsigned int num_va_buffers = 0;
973
974     va_buffers[num_va_buffers++] = ctx->seq_param_buf_id;
975     va_buffers[num_va_buffers++] = ctx->pic_param_buf_id;
976
977     if (ctx->packed_seq_header_param_buf_id != VA_INVALID_ID)
978         va_buffers[num_va_buffers++] = ctx->packed_seq_header_param_buf_id;
979
980     if (ctx->packed_seq_buf_id != VA_INVALID_ID)
981         va_buffers[num_va_buffers++] = ctx->packed_seq_buf_id;
982
983     if (ctx->packed_pic_header_param_buf_id != VA_INVALID_ID)
984         va_buffers[num_va_buffers++] = ctx->packed_pic_header_param_buf_id;
985
986     if (ctx->packed_pic_buf_id != VA_INVALID_ID)
987         va_buffers[num_va_buffers++] = ctx->packed_pic_buf_id;
988
989     va_status = vaBeginPicture(ctx->va_dpy,
990                                ctx->context_id,
991                                surface_ids[ctx->current_input_surface]);
992     CHECK_VASTATUS(va_status,"vaBeginPicture");
993
994     va_status = vaRenderPicture(ctx->va_dpy,
995                                 ctx->context_id,
996                                 va_buffers,
997                                 num_va_buffers);
998     CHECK_VASTATUS(va_status,"vaRenderPicture");
999
1000     va_status = vaRenderPicture(ctx->va_dpy,
1001                                 ctx->context_id,
1002                                 &ctx->slice_param_buf_id[0],
1003                                 ctx->num_slice_groups);
1004     CHECK_VASTATUS(va_status,"vaRenderPicture");
1005
1006     va_status = vaEndPicture(ctx->va_dpy, ctx->context_id);
1007     CHECK_VASTATUS(va_status,"vaEndPicture");
1008
1009     return 0;
1010 }
1011
1012 static int 
1013 mpeg2enc_destroy_buffers(struct mpeg2enc_context *ctx, VABufferID *va_buffers, unsigned int num_va_buffers)
1014 {
1015     VAStatus va_status;
1016     unsigned int i;
1017
1018     for (i = 0; i < num_va_buffers; i++) {
1019         if (va_buffers[i] != VA_INVALID_ID) {
1020             va_status = vaDestroyBuffer(ctx->va_dpy, va_buffers[i]);
1021             CHECK_VASTATUS(va_status,"vaDestroyBuffer");
1022             va_buffers[i] = VA_INVALID_ID;
1023         }
1024     }
1025
1026     return 0;
1027 }
1028
1029 static void
1030 end_picture(struct mpeg2enc_context *ctx, VAEncPictureType picture_type, int next_is_bpic)
1031 {
1032     VABufferID tempID;
1033
1034     /* Prepare for next picture */
1035     tempID = surface_ids[SID_RECON_PICTURE];  
1036
1037     if (picture_type != VAEncPictureTypeBidirectional) {
1038         if (next_is_bpic) {
1039             surface_ids[SID_RECON_PICTURE] = surface_ids[SID_REFERENCE_PICTURE_L1]; 
1040             surface_ids[SID_REFERENCE_PICTURE_L1] = tempID;     
1041         } else {
1042             surface_ids[SID_RECON_PICTURE] = surface_ids[SID_REFERENCE_PICTURE_L0]; 
1043             surface_ids[SID_REFERENCE_PICTURE_L0] = tempID;
1044         }
1045     } else {
1046         if (!next_is_bpic) {
1047             surface_ids[SID_RECON_PICTURE] = surface_ids[SID_REFERENCE_PICTURE_L0]; 
1048             surface_ids[SID_REFERENCE_PICTURE_L0] = surface_ids[SID_REFERENCE_PICTURE_L1];
1049             surface_ids[SID_REFERENCE_PICTURE_L1] = tempID;
1050         }
1051     }
1052
1053     mpeg2enc_destroy_buffers(ctx, &ctx->seq_param_buf_id, 1);
1054     mpeg2enc_destroy_buffers(ctx, &ctx->pic_param_buf_id, 1);
1055     mpeg2enc_destroy_buffers(ctx, &ctx->packed_seq_header_param_buf_id, 1);
1056     mpeg2enc_destroy_buffers(ctx, &ctx->packed_seq_buf_id, 1);
1057     mpeg2enc_destroy_buffers(ctx, &ctx->packed_pic_header_param_buf_id, 1);
1058     mpeg2enc_destroy_buffers(ctx, &ctx->packed_pic_buf_id, 1);
1059     mpeg2enc_destroy_buffers(ctx, &ctx->slice_param_buf_id[0], ctx->num_slice_groups);
1060     mpeg2enc_destroy_buffers(ctx, &ctx->codedbuf_buf_id, 1);
1061     memset(ctx->slice_param, 0, sizeof(ctx->slice_param));
1062     ctx->num_slice_groups = 0;
1063 }
1064
1065 static int
1066 store_coded_buffer(struct mpeg2enc_context *ctx, VAEncPictureType picture_type)
1067 {
1068     VACodedBufferSegment *coded_buffer_segment;
1069     unsigned char *coded_mem;
1070     int slice_data_length;
1071     VAStatus va_status;
1072     VASurfaceStatus surface_status;
1073     size_t w_items;
1074
1075     va_status = vaSyncSurface(ctx->va_dpy, surface_ids[ctx->current_input_surface]);
1076     CHECK_VASTATUS(va_status,"vaSyncSurface");
1077
1078     surface_status = 0;
1079     va_status = vaQuerySurfaceStatus(ctx->va_dpy, surface_ids[ctx->current_input_surface], &surface_status);
1080     CHECK_VASTATUS(va_status,"vaQuerySurfaceStatus");
1081
1082     va_status = vaMapBuffer(ctx->va_dpy, ctx->codedbuf_buf_id, (void **)(&coded_buffer_segment));
1083     CHECK_VASTATUS(va_status,"vaMapBuffer");
1084     coded_mem = coded_buffer_segment->buf;
1085
1086     if (coded_buffer_segment->status & VA_CODED_BUF_STATUS_SLICE_OVERFLOW_MASK) {
1087         if (picture_type == VAEncPictureTypeIntra)
1088             ctx->codedbuf_i_size *= 2;
1089         else
1090             ctx->codedbuf_pb_size *= 2;
1091
1092         vaUnmapBuffer(ctx->va_dpy, ctx->codedbuf_buf_id);
1093         return -1;
1094     }
1095
1096     slice_data_length = coded_buffer_segment->size;
1097
1098     do {
1099         w_items = fwrite(coded_mem, slice_data_length, 1, ctx->ofp);
1100     } while (w_items != 1);
1101
1102     if (picture_type == VAEncPictureTypeIntra) {
1103         if (ctx->codedbuf_i_size > slice_data_length * 3 / 2) {
1104             ctx->codedbuf_i_size = slice_data_length * 3 / 2;
1105         }
1106         
1107         if (ctx->codedbuf_pb_size < slice_data_length) {
1108             ctx->codedbuf_pb_size = slice_data_length;
1109         }
1110     } else {
1111         if (ctx->codedbuf_pb_size > slice_data_length * 3 / 2) {
1112             ctx->codedbuf_pb_size = slice_data_length * 3 / 2;
1113         }
1114     }
1115
1116     vaUnmapBuffer(ctx->va_dpy, ctx->codedbuf_buf_id);
1117
1118     return 0;
1119 }
1120
1121 static void
1122 encode_picture(struct mpeg2enc_context *ctx,
1123                int coded_order,
1124                int display_order,
1125                VAEncPictureType picture_type,
1126                int next_is_bpic,
1127                int next_display_order)
1128 {
1129     VAStatus va_status;
1130     int ret = 0, codedbuf_size;
1131     
1132     begin_picture(ctx, coded_order, display_order, picture_type);
1133
1134     if (1) {
1135         /* upload YUV data to VA surface for next frame */
1136         if (next_display_order >= ctx->num_pictures)
1137             next_display_order = ctx->num_pictures - 1;
1138
1139         fseek(ctx->ifp, ctx->frame_size * next_display_order, SEEK_SET);
1140         ctx->upload_thread_value = pthread_create(&ctx->upload_thread_id,
1141                                                   NULL,
1142                                                   upload_yuv_to_surface,
1143                                                   ctx);
1144     }
1145
1146     do {
1147         mpeg2enc_destroy_buffers(ctx, &ctx->codedbuf_buf_id, 1);
1148         mpeg2enc_destroy_buffers(ctx, &ctx->pic_param_buf_id, 1);
1149
1150
1151         if (VAEncPictureTypeIntra == picture_type) {
1152             codedbuf_size = ctx->codedbuf_i_size;
1153         } else {
1154             codedbuf_size = ctx->codedbuf_pb_size;
1155         }
1156
1157         /* coded buffer */
1158         va_status = vaCreateBuffer(ctx->va_dpy,
1159                                    ctx->context_id,
1160                                    VAEncCodedBufferType,
1161                                    codedbuf_size, 1, NULL,
1162                                    &ctx->codedbuf_buf_id);
1163         CHECK_VASTATUS(va_status,"vaCreateBuffer");
1164
1165         /* picture parameter set */
1166         mpeg2enc_update_picture_parameter(ctx, picture_type, coded_order, display_order);
1167
1168         mpeg2enc_render_picture(ctx);
1169
1170         ret = store_coded_buffer(ctx, picture_type);
1171     } while (ret);
1172
1173     end_picture(ctx, picture_type, next_is_bpic);
1174 }
1175
1176 static void
1177 encode_pb_pictures(struct mpeg2enc_context *ctx,
1178                    int coded_order,
1179                    int f, 
1180                    int nbframes,
1181                    int next_f)
1182 {
1183     int i;
1184
1185     encode_picture(ctx,
1186                    coded_order,
1187                    f + nbframes,
1188                    VAEncPictureTypePredictive,
1189                    1,
1190                    f);
1191
1192     for( i = 0; i < nbframes - 1; i++) {
1193         encode_picture(ctx,
1194                        coded_order + 1,
1195                        f + i,
1196                        VAEncPictureTypeBidirectional,
1197                        1,
1198                        f + i + 1);
1199     }
1200     
1201     encode_picture(ctx,
1202                    coded_order + 1,
1203                    f + nbframes - 1,
1204                    VAEncPictureTypeBidirectional,
1205                    0,
1206                    next_f);
1207 }
1208
1209 static void
1210 mpeg2enc_run(struct mpeg2enc_context *ctx)
1211 {
1212     int display_order = 0, coded_order = 0;
1213     int i_frame_only = 1, i_p_frame_only = 1;
1214
1215     for (display_order = 0; display_order < ctx->num_pictures;) {
1216         if (i_frame_only) {
1217             encode_picture(ctx,
1218                            coded_order,
1219                            display_order,
1220                            VAEncPictureTypeIntra,
1221                            0,
1222                            display_order + 1);
1223             display_order++;
1224             coded_order++;
1225         } else if (i_p_frame_only) {
1226             if ((display_order % ctx->intra_period) == 0) {
1227                 encode_picture(ctx,
1228                                coded_order,
1229                                display_order,
1230                                VAEncPictureTypeIntra,
1231                                0,
1232                                display_order + 1);
1233                 display_order++;
1234                 coded_order++;
1235             } else {
1236                 encode_picture(ctx,
1237                                coded_order,
1238                                display_order,
1239                                VAEncPictureTypePredictive,
1240                                0,
1241                                display_order + 1);
1242                 display_order++;
1243                 coded_order++;
1244             }
1245         } else { // follow the i,p,b pattern
1246             static int fcurrent = 0;
1247             int fnext;
1248             
1249             fcurrent = fcurrent % (sizeof(picture_type_patter)/sizeof(int[2]));
1250             fnext = (fcurrent+1) % (sizeof(picture_type_patter)/sizeof(int[2]));
1251             
1252             if ( picture_type_patter[fcurrent][0] == VAEncPictureTypeIntra ) {
1253                 encode_picture(ctx,
1254                                coded_order,
1255                                display_order,
1256                                VAEncPictureTypeIntra,
1257                                0, 
1258                                display_order + picture_type_patter[fnext][1]);
1259                 display_order++;
1260                 coded_order++;
1261             } else {
1262                 encode_pb_pictures(ctx,
1263                                    coded_order,
1264                                    display_order,
1265                                    picture_type_patter[fcurrent][1] - 1, 
1266                                    display_order + picture_type_patter[fcurrent][1] + picture_type_patter[fnext][1] - 1);
1267                 display_order += picture_type_patter[fcurrent][1];
1268                 coded_order++;
1269             }
1270  
1271             fcurrent++;
1272         }
1273
1274         fprintf(stderr, "\r %d/%d ...", display_order + 1, ctx->num_pictures);
1275         fflush(stdout);
1276     }
1277 }
1278
1279 /*
1280  * end
1281  */
1282 static void
1283 mpeg2enc_release_va_resources(struct mpeg2enc_context *ctx)
1284 {
1285     vaDestroySurfaces(ctx->va_dpy, surface_ids, SID_NUMBER);    
1286     vaDestroyContext(ctx->va_dpy, ctx->context_id);
1287     vaDestroyConfig(ctx->va_dpy, ctx->config_id);
1288     vaTerminate(ctx->va_dpy);
1289     va_close_display(ctx->va_dpy);
1290 }
1291
1292 static void
1293 mpeg2enc_end(struct mpeg2enc_context *ctx)
1294 {
1295     pthread_join(ctx->upload_thread_id, NULL);
1296     mpeg2enc_release_va_resources(ctx);
1297 }
1298
1299 int 
1300 main(int argc, char *argv[])
1301 {
1302     struct mpeg2enc_context ctx;
1303     struct timeval tpstart, tpend; 
1304     float timeuse;
1305
1306     gettimeofday(&tpstart, NULL);
1307
1308     memset(&ctx, 0, sizeof(ctx));
1309     parse_args(&ctx, argc, argv);
1310     mpeg2enc_init(&ctx);
1311     mpeg2enc_run(&ctx);
1312     mpeg2enc_end(&ctx);
1313
1314     gettimeofday(&tpend, NULL);
1315     timeuse = 1000000 * (tpend.tv_sec - tpstart.tv_sec) + tpend.tv_usec - tpstart.tv_usec;
1316     timeuse /= 1000000;
1317     fprintf(stderr, "\ndone!\n");
1318     fprintf(stderr, "encode %d frames in %f secondes, FPS is %.1f\n", ctx.num_pictures, timeuse, ctx.num_pictures / timeuse);
1319
1320     mpeg2enc_exit(&ctx, 0);
1321
1322     return 0;
1323 }