1 /*****************************************************************************
2 * daala.c: daala codec module making use of libdaala.
3 *****************************************************************************
4 * Copyright (C) 2014 VLC authors and VideoLAN
6 * Authors: Tristan Matthews <le.businessman@gmail.com>
7 * * Based on theora.c by: Gildas Bazin <gbazin@videolan.org>
9 * This program is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU Lesser General Public License as published by
11 * the Free Software Foundation; either version 2.1 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public License
20 * along with this program; if not, write to the Free Software Foundation,
21 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
24 /*****************************************************************************
26 *****************************************************************************/
31 #include <vlc_common.h>
32 #include <vlc_plugin.h>
33 #include <vlc_codec.h>
34 #include <vlc_input.h>
35 #include "../demux/xiph.h"
37 #include <daala/codec.h>
38 #include <daala/daaladec.h>
40 #include <daala/daalaenc.h>
45 /*****************************************************************************
46 * decoder_sys_t : daala decoder descriptor
47 *****************************************************************************/
61 daala_info di
; /* daala bitstream settings */
62 daala_comment dc
; /* daala comment information */
63 daala_dec_ctx
*dcx
; /* daala decoder context */
68 bool b_decoded_first_keyframe
;
76 /*****************************************************************************
78 *****************************************************************************/
79 static int OpenDecoder ( vlc_object_t
* );
80 static int OpenPacketizer( vlc_object_t
* );
81 static void CloseDecoder ( vlc_object_t
* );
83 static int DecodeVideo( decoder_t
*p_dec
, block_t
*p_block
);
84 static block_t
*Packetize ( decoder_t
*, block_t
** );
85 static int ProcessHeaders( decoder_t
* );
86 static void *ProcessPacket ( decoder_t
*, daala_packet
*, block_t
* );
88 static picture_t
*DecodePacket( decoder_t
*, daala_packet
* );
90 static void ParseDaalaComments( decoder_t
* );
91 static void daala_CopyPicture( picture_t
*, daala_image
* );
94 static int OpenEncoder( vlc_object_t
*p_this
);
95 static void CloseEncoder( vlc_object_t
*p_this
);
96 static block_t
*Encode( encoder_t
*p_enc
, picture_t
*p_pict
);
98 static const char *const enc_chromafmt_list
[] = {
101 static const char *const enc_chromafmt_list_text
[] = {
106 /*****************************************************************************
108 *****************************************************************************/
109 #define ENC_QUALITY_TEXT N_("Encoding quality")
110 #define ENC_QUALITY_LONGTEXT N_( \
111 "Enforce a quality between 0 (lossless) and 511 (worst)." )
112 #define ENC_KEYINT_TEXT N_("Keyframe interval")
113 #define ENC_KEYINT_LONGTEXT N_( \
114 "Enforce a keyframe interval between 1 and 1000." )
117 set_category( CAT_INPUT
)
118 set_subcategory( SUBCAT_INPUT_VCODEC
)
119 set_shortname( "Daala" )
120 set_description( N_("Daala video decoder") )
121 set_capability( "video decoder", 100 )
122 set_callbacks( OpenDecoder
, CloseDecoder
)
123 add_shortcut( "daala" )
125 set_description( N_("Daala video packetizer") )
126 set_capability( "packetizer", 100 )
127 set_callbacks( OpenPacketizer
, CloseDecoder
)
128 add_shortcut( "daala" )
132 set_description( N_("Daala video encoder") )
133 set_capability( "encoder", 150 )
134 set_callbacks( OpenEncoder
, CloseEncoder
)
135 add_shortcut( "daala" )
137 # define ENC_CFG_PREFIX "sout-daala-"
138 add_integer_with_range( ENC_CFG_PREFIX
"quality", 10, 0, 511,
139 ENC_QUALITY_TEXT
, ENC_QUALITY_LONGTEXT
, false )
140 add_integer_with_range( ENC_CFG_PREFIX
"keyint", 256, 1, 1000,
141 ENC_KEYINT_TEXT
, ENC_KEYINT_LONGTEXT
, false )
143 # define ENC_CHROMAFMT_TEXT N_("Chroma format")
144 # define ENC_CHROMAFMT_LONGTEXT N_("Picking chroma format will force a " \
145 "conversion of the video into that format")
147 add_string( ENC_CFG_PREFIX
"chroma-fmt", "420", ENC_CHROMAFMT_TEXT
,
148 ENC_CHROMAFMT_LONGTEXT
, false )
149 change_string_list( enc_chromafmt_list
, enc_chromafmt_list_text
)
152 static const char *const ppsz_enc_options
[] = {
153 "quality", "keyint", "chroma-fmt", NULL
157 static int OpenCommon( vlc_object_t
*p_this
, bool b_packetizer
)
159 decoder_t
*p_dec
= (decoder_t
*)p_this
;
160 decoder_sys_t
*p_sys
;
162 if( p_dec
->fmt_in
.i_codec
!= VLC_CODEC_DAALA
)
167 /* Allocate the memory needed to store the decoder's structure */
168 p_sys
= malloc(sizeof(*p_sys
));
172 p_dec
->p_sys
= p_sys
;
173 p_dec
->p_sys
->b_packetizer
= b_packetizer
;
174 p_sys
->b_has_headers
= false;
175 p_sys
->i_pts
= VLC_TICK_INVALID
;
176 p_sys
->b_decoded_first_keyframe
= false;
181 p_dec
->fmt_out
.i_codec
= VLC_CODEC_DAALA
;
182 p_dec
->pf_packetize
= Packetize
;
186 p_dec
->fmt_out
.i_codec
= VLC_CODEC_I420
;
187 p_dec
->pf_decode
= DecodeVideo
;
190 /* Init supporting Daala structures needed in header parsing */
191 daala_comment_init( &p_sys
->dc
);
192 daala_info_init( &p_sys
->di
);
197 /*****************************************************************************
198 * OpenDecoder: probe the decoder and return score
199 *****************************************************************************/
200 static int OpenDecoder( vlc_object_t
*p_this
)
202 return OpenCommon( p_this
, false );
205 static int OpenPacketizer( vlc_object_t
*p_this
)
207 return OpenCommon( p_this
, true );
210 /****************************************************************************
211 * DecodeBlock: the whole thing
212 ****************************************************************************
213 * This function must be fed with Daala packets.
214 ****************************************************************************/
215 static void *DecodeBlock( decoder_t
*p_dec
, block_t
*p_block
)
217 decoder_sys_t
*p_sys
= p_dec
->p_sys
;
218 daala_packet dpacket
;
220 /* Block to Daala packet */
221 dpacket
.packet
= p_block
->p_buffer
;
222 dpacket
.bytes
= p_block
->i_buffer
;
223 dpacket
.granulepos
= p_block
->i_dts
;
226 dpacket
.packetno
= 0;
228 /* Check for headers */
229 if( !p_sys
->b_has_headers
)
231 if( ProcessHeaders( p_dec
) )
233 block_Release( p_block
);
236 p_sys
->b_has_headers
= true;
239 /* If we haven't seen a single keyframe yet, set to preroll,
240 * otherwise we'll get display artifacts. (This is impossible
241 * in the general case, but can happen if e.g. we play a network stream
242 * using a timed URL, such that the server doesn't start the video with a
244 if( !p_sys
->b_decoded_first_keyframe
)
245 p_block
->i_flags
|= BLOCK_FLAG_PREROLL
; /* Wait until we've decoded the first keyframe */
247 return ProcessPacket( p_dec
, &dpacket
, p_block
);
250 static int DecodeVideo( decoder_t
*p_dec
, block_t
*p_block
)
252 if( p_block
== NULL
) /* No Drain */
253 return VLCDEC_SUCCESS
;
255 picture_t
*p_pic
= DecodeBlock( p_dec
, p_block
);
257 decoder_QueueVideo( p_dec
, p_pic
);
258 return VLCDEC_SUCCESS
;
261 static block_t
*Packetize( decoder_t
*p_dec
, block_t
**pp_block
)
263 if( pp_block
== NULL
) /* No Drain */
265 block_t
*p_block
= *pp_block
; *pp_block
= NULL
;
266 if( p_block
== NULL
)
268 return DecodeBlock( p_dec
, p_block
);
271 /*****************************************************************************
272 * ProcessHeaders: process Daala headers.
273 *****************************************************************************/
274 static int ProcessHeaders( decoder_t
*p_dec
)
276 int ret
= VLC_SUCCESS
;
277 decoder_sys_t
*p_sys
= p_dec
->p_sys
;
278 daala_packet dpacket
;
279 daala_setup_info
*ds
= NULL
; /* daala setup information */
281 unsigned pi_size
[XIPH_MAX_HEADER_COUNT
];
282 void *pp_data
[XIPH_MAX_HEADER_COUNT
];
284 if( xiph_SplitHeaders( pi_size
, pp_data
, &i_count
,
285 p_dec
->fmt_in
.i_extra
, p_dec
->fmt_in
.p_extra
) )
290 dpacket
.granulepos
= -1;
292 dpacket
.packetno
= 0;
294 /* Take care of the initial info header */
295 dpacket
.b_o_s
= 1; /* yes this actually is a b_o_s packet :) */
296 dpacket
.bytes
= pi_size
[0];
297 dpacket
.packet
= pp_data
[0];
298 if( daala_decode_header_in( &p_sys
->di
, &p_sys
->dc
, &ds
, &dpacket
) < 0 )
300 msg_Err( p_dec
, "this bitstream does not contain Daala video data" );
305 /* Set output properties */
306 if( !p_sys
->b_packetizer
)
308 if( p_sys
->di
.plane_info
[0].xdec
== 0 && p_sys
->di
.plane_info
[0].ydec
== 0 &&
309 p_sys
->di
.plane_info
[1].xdec
== 1 && p_sys
->di
.plane_info
[1].ydec
== 1 &&
310 p_sys
->di
.plane_info
[2].xdec
== 1 && p_sys
->di
.plane_info
[2].ydec
== 1 )
312 p_dec
->fmt_out
.i_codec
= VLC_CODEC_I420
;
314 else if( p_sys
->di
.plane_info
[0].xdec
== 0 && p_sys
->di
.plane_info
[0].ydec
== 0 &&
315 p_sys
->di
.plane_info
[1].xdec
== 0 && p_sys
->di
.plane_info
[1].ydec
== 0 &&
316 p_sys
->di
.plane_info
[2].xdec
== 0 && p_sys
->di
.plane_info
[2].ydec
== 0 )
318 p_dec
->fmt_out
.i_codec
= VLC_CODEC_I444
;
322 msg_Err( p_dec
, "unknown chroma in daala sample" );
326 p_dec
->fmt_out
.video
.i_width
= p_sys
->di
.pic_width
;
327 p_dec
->fmt_out
.video
.i_height
= p_sys
->di
.pic_height
;
328 if( p_sys
->di
.pic_width
&& p_sys
->di
.pic_height
)
330 p_dec
->fmt_out
.video
.i_visible_width
= p_sys
->di
.pic_width
;
331 p_dec
->fmt_out
.video
.i_visible_height
= p_sys
->di
.pic_height
;
334 if( p_sys
->di
.pixel_aspect_denominator
&& p_sys
->di
.pixel_aspect_numerator
)
336 p_dec
->fmt_out
.video
.i_sar_num
= p_sys
->di
.pixel_aspect_numerator
;
337 p_dec
->fmt_out
.video
.i_sar_den
= p_sys
->di
.pixel_aspect_denominator
;
341 p_dec
->fmt_out
.video
.i_sar_num
= 1;
342 p_dec
->fmt_out
.video
.i_sar_den
= 1;
345 if( p_sys
->di
.timebase_numerator
> 0 && p_sys
->di
.timebase_denominator
> 0 )
347 p_dec
->fmt_out
.video
.i_frame_rate
= p_sys
->di
.timebase_numerator
;
348 p_dec
->fmt_out
.video
.i_frame_rate_base
= p_sys
->di
.timebase_denominator
;
351 msg_Dbg( p_dec
, "%dx%d %.02f fps video, frame content ",
352 p_sys
->di
.pic_width
, p_sys
->di
.pic_height
,
353 (double)p_sys
->di
.timebase_numerator
/p_sys
->di
.timebase_denominator
);
355 /* The next packet in order is the comments header */
357 dpacket
.bytes
= pi_size
[1];
358 dpacket
.packet
= pp_data
[1];
360 if( daala_decode_header_in( &p_sys
->di
, &p_sys
->dc
, &ds
, &dpacket
) < 0 )
362 msg_Err( p_dec
, "Daala comment header is corrupted" );
367 ParseDaalaComments( p_dec
);
369 /* The next packet in order is the setup header
370 * We need to watch out that this packet is not missing as a
371 * missing or corrupted header is fatal. */
373 dpacket
.bytes
= pi_size
[2];
374 dpacket
.packet
= pp_data
[2];
375 if( daala_decode_header_in( &p_sys
->di
, &p_sys
->dc
, &ds
, &dpacket
) < 0 )
377 msg_Err( p_dec
, "Daala setup header is corrupted" );
382 if( !p_sys
->b_packetizer
)
384 /* We have all the headers, initialize decoder */
385 if ( ( p_sys
->dcx
= daala_decode_create( &p_sys
->di
, ds
) ) == NULL
)
387 msg_Err( p_dec
, "Could not allocate Daala decoder" );
394 void* p_extra
= realloc( p_dec
->fmt_out
.p_extra
,
395 p_dec
->fmt_in
.i_extra
);
396 if( unlikely( p_extra
== NULL
) )
401 p_dec
->fmt_out
.p_extra
= p_extra
;
402 p_dec
->fmt_out
.i_extra
= p_dec
->fmt_in
.i_extra
;
403 memcpy( p_dec
->fmt_out
.p_extra
,
404 p_dec
->fmt_in
.p_extra
, p_dec
->fmt_out
.i_extra
);
408 /* Clean up the decoder setup info... we're done with it */
409 daala_setup_free( ds
);
414 /*****************************************************************************
415 * ProcessPacket: processes a daala packet.
416 *****************************************************************************/
417 static void *ProcessPacket( decoder_t
*p_dec
, daala_packet
*p_dpacket
,
420 decoder_sys_t
*p_sys
= p_dec
->p_sys
;
423 if( ( p_block
->i_flags
&(BLOCK_FLAG_CORRUPTED
) ) != 0 )
425 /* Don't send the the first packet after a discontinuity to
426 * daala_decode, otherwise we get purple/green display artifacts
427 * appearing in the video output */
428 block_Release(p_block
);
432 /* Date management */
433 if( p_block
->i_pts
!= VLC_TICK_INVALID
&& p_block
->i_pts
!= p_sys
->i_pts
)
435 p_sys
->i_pts
= p_block
->i_pts
;
438 if( p_sys
->b_packetizer
)
440 /* Date management */
441 p_block
->i_dts
= p_block
->i_pts
= p_sys
->i_pts
;
443 p_block
->i_length
= p_sys
->i_pts
- p_block
->i_pts
;
449 p_buf
= DecodePacket( p_dec
, p_dpacket
);
450 block_Release( p_block
);
453 /* Date management */
454 p_sys
->i_pts
+= vlc_tick_from_samples( p_sys
->di
.timebase_denominator
,
455 p_sys
->di
.timebase_numerator
); /* 1 frame per packet */
460 /*****************************************************************************
461 * DecodePacket: decodes a Daala packet.
462 *****************************************************************************/
463 static picture_t
*DecodePacket( decoder_t
*p_dec
, daala_packet
*p_dpacket
)
465 decoder_sys_t
*p_sys
= p_dec
->p_sys
;
469 if (daala_decode_packet_in( p_sys
->dcx
, p_dpacket
) < 0)
470 return NULL
; /* bad packet */
472 if (!daala_decode_img_out( p_sys
->dcx
, &ycbcr
))
475 /* Check for keyframe */
476 if( daala_packet_iskeyframe( p_dpacket
) )
477 p_sys
->b_decoded_first_keyframe
= true;
479 /* Get a new picture */
480 if( decoder_UpdateVideoFormat( p_dec
) )
482 p_pic
= decoder_NewPicture( p_dec
);
483 if( !p_pic
) return NULL
;
485 daala_CopyPicture( p_pic
, &ycbcr
);
487 p_pic
->date
= p_sys
->i_pts
;
492 /*****************************************************************************
493 * ParseDaalaComments:
494 *****************************************************************************/
495 static void ParseDaalaComments( decoder_t
*p_dec
)
497 char *psz_name
, *psz_value
, *psz_comment
;
498 /* Regarding the daala_comment structure: */
500 /* The metadata is stored as a series of (tag, value) pairs, in
501 length-encoded string vectors. The first occurrence of the '='
502 character delimits the tag and value. A particular tag may
503 occur more than once, and order is significant. The character
504 set encoding for the strings is always UTF-8, but the tag names
505 are limited to ASCII, and treated as case-insensitive. See the
506 Theora specification, Section 6.3.3 for details. */
508 /* In filling in this structure, daala_decode_header_in() will
509 null-terminate the user_comment strings for safety. However,
510 the bitstream format itself treats them as 8-bit clean vectors,
511 possibly containing null characters, and so the length array
512 should be treated as their authoritative length. */
513 for ( int i
= 0; i
< p_dec
->p_sys
->dc
.comments
; i
++ )
515 int clen
= p_dec
->p_sys
->dc
.comment_lengths
[i
];
516 if ( clen
<= 0 || clen
>= INT_MAX
) { continue; }
517 psz_comment
= malloc( clen
+ 1 );
520 memcpy( (void*)psz_comment
, (void*)p_dec
->p_sys
->dc
.user_comments
[i
], clen
+ 1 );
521 psz_comment
[clen
] = '\0';
523 psz_name
= psz_comment
;
524 psz_value
= strchr( psz_comment
, '=' );
530 if( !p_dec
->p_description
)
531 p_dec
->p_description
= vlc_meta_New();
532 /* TODO: Since psz_value can contain NULLs see if there is an
533 * instance where we need to preserve the full length of this string */
534 if( p_dec
->p_description
)
535 vlc_meta_AddExtra( p_dec
->p_description
, psz_name
, psz_value
);
541 /*****************************************************************************
542 * CloseDecoder: daala decoder destruction
543 *****************************************************************************/
544 static void CloseDecoder( vlc_object_t
*p_this
)
546 decoder_t
*p_dec
= (decoder_t
*)p_this
;
547 decoder_sys_t
*p_sys
= p_dec
->p_sys
;
549 daala_info_clear(&p_sys
->di
);
550 daala_comment_clear(&p_sys
->dc
);
551 daala_decode_free(p_sys
->dcx
);
555 /*****************************************************************************
556 * daala_CopyPicture: copy a picture from daala internal buffers to a
557 * picture_t structure.
558 *****************************************************************************/
559 static void daala_CopyPicture( picture_t
*p_pic
,
562 const int i_planes
= p_pic
->i_planes
< 3 ? p_pic
->i_planes
: 3;
563 for( int i_plane
= 0; i_plane
< i_planes
; i_plane
++ )
565 const int i_total_lines
= __MIN(p_pic
->p
[i_plane
].i_lines
,
566 ycbcr
->height
>> ycbcr
->planes
[i_plane
].ydec
);
567 uint8_t *p_dst
= p_pic
->p
[i_plane
].p_pixels
;
568 uint8_t *p_src
= ycbcr
->planes
[i_plane
].data
;
569 const int i_dst_stride
= p_pic
->p
[i_plane
].i_pitch
;
570 const int i_src_stride
= ycbcr
->planes
[i_plane
].ystride
;
571 for( int i_line
= 0; i_line
< i_total_lines
; i_line
++ )
573 memcpy( p_dst
, p_src
, i_src_stride
);
574 p_src
+= i_src_stride
;
575 p_dst
+= i_dst_stride
;
583 daala_info di
; /* daala bitstream settings */
584 daala_comment dc
; /* daala comment header */
585 daala_enc_ctx
*dcx
; /* daala context */
588 static int OpenEncoder( vlc_object_t
*p_this
)
590 encoder_t
*p_enc
= (encoder_t
*)p_this
;
591 encoder_sys_t
*p_sys
;
595 if( p_enc
->fmt_out
.i_codec
!= VLC_CODEC_DAALA
&&
601 /* Allocate the memory needed to store the encoder's structure */
602 p_sys
= malloc( sizeof( encoder_sys_t
) );
605 p_enc
->p_sys
= p_sys
;
607 p_enc
->pf_encode_video
= Encode
;
608 p_enc
->fmt_in
.i_codec
= VLC_CODEC_I420
;
609 p_enc
->fmt_out
.i_codec
= VLC_CODEC_DAALA
;
611 config_ChainParse( p_enc
, ENC_CFG_PREFIX
, ppsz_enc_options
, p_enc
->p_cfg
);
613 char *psz_tmp
= var_GetString( p_enc
, ENC_CFG_PREFIX
"chroma-fmt" );
619 if( !strcmp( psz_tmp
, "420" ) ) {
620 i_codec
= VLC_CODEC_I420
;
622 else if( !strcmp( psz_tmp
, "444" ) ) {
623 i_codec
= VLC_CODEC_I444
;
626 msg_Err( p_enc
, "Invalid chroma format: %s", psz_tmp
);
632 p_enc
->fmt_in
.i_codec
= i_codec
;
633 /* update bits_per_pixel */
634 video_format_Setup(&p_enc
->fmt_in
.video
, i_codec
,
635 p_enc
->fmt_in
.video
.i_width
,
636 p_enc
->fmt_in
.video
.i_height
,
637 p_enc
->fmt_in
.video
.i_visible_width
,
638 p_enc
->fmt_in
.video
.i_visible_height
,
639 p_enc
->fmt_in
.video
.i_sar_num
,
640 p_enc
->fmt_in
.video
.i_sar_den
);
643 daala_info_init( &p_sys
->di
);
645 p_sys
->di
.pic_width
= p_enc
->fmt_in
.video
.i_visible_width
;
646 p_sys
->di
.pic_height
= p_enc
->fmt_in
.video
.i_visible_height
;
648 p_sys
->di
.nplanes
= 3;
649 for (int i
= 0; i
< p_sys
->di
.nplanes
; i
++)
651 p_sys
->di
.plane_info
[i
].xdec
= i
> 0 && i_codec
!= VLC_CODEC_I444
;
652 p_sys
->di
.plane_info
[i
].ydec
= i_codec
== VLC_CODEC_I420
?
653 p_sys
->di
.plane_info
[i
].xdec
: 0;
655 p_sys
->di
.frame_duration
= 1;
657 if( !p_enc
->fmt_in
.video
.i_frame_rate
||
658 !p_enc
->fmt_in
.video
.i_frame_rate_base
)
660 p_sys
->di
.timebase_numerator
= 25;
661 p_sys
->di
.timebase_denominator
= 1;
665 p_sys
->di
.timebase_numerator
= p_enc
->fmt_in
.video
.i_frame_rate
;
666 p_sys
->di
.timebase_denominator
= p_enc
->fmt_in
.video
.i_frame_rate_base
;
669 if( p_enc
->fmt_in
.video
.i_sar_num
> 0 && p_enc
->fmt_in
.video
.i_sar_den
> 0 )
671 unsigned i_dst_num
, i_dst_den
;
672 vlc_ureduce( &i_dst_num
, &i_dst_den
,
673 p_enc
->fmt_in
.video
.i_sar_num
,
674 p_enc
->fmt_in
.video
.i_sar_den
, 0 );
675 p_sys
->di
.pixel_aspect_numerator
= i_dst_num
;
676 p_sys
->di
.pixel_aspect_denominator
= i_dst_den
;
680 p_sys
->di
.pixel_aspect_numerator
= 4;
681 p_sys
->di
.pixel_aspect_denominator
= 3;
684 p_sys
->di
.keyframe_rate
= var_GetInteger( p_enc
, ENC_CFG_PREFIX
"keyint" );
687 p_sys
->dcx
= dcx
= daala_encode_create( &p_sys
->di
);
694 daala_comment_init( &p_sys
->dc
);
696 int i_quality
= var_GetInteger( p_enc
, ENC_CFG_PREFIX
"quality" );
697 daala_encode_ctl( dcx
, OD_SET_QUANT
, &i_quality
, sizeof(i_quality
) );
699 /* Create and store headers */
700 while( ( status
= daala_encode_flush_header( dcx
, &p_sys
->dc
, &header
) ) )
704 CloseEncoder( p_this
);
707 if( xiph_AppendHeaders( &p_enc
->fmt_out
.i_extra
,
708 &p_enc
->fmt_out
.p_extra
, header
.bytes
,
711 p_enc
->fmt_out
.i_extra
= 0;
712 p_enc
->fmt_out
.p_extra
= NULL
;
718 static block_t
*Encode( encoder_t
*p_enc
, picture_t
*p_pict
)
720 encoder_sys_t
*p_sys
= p_enc
->p_sys
;
721 daala_packet dpacket
;
725 if( !p_pict
) return NULL
;
727 const int i_width
= p_sys
->di
.pic_width
;
728 const int i_height
= p_sys
->di
.pic_height
;
731 if( p_pict
->p
[0].i_pitch
< i_width
||
732 p_pict
->p
[0].i_lines
< i_height
)
734 msg_Err( p_enc
, "frame is smaller than encoding size"
735 "(%ix%i->%ix%i) -> dropping frame",
736 p_pict
->p
[0].i_pitch
, p_pict
->p
[0].i_lines
,
741 /* Daala is a one-frame-in, one-frame-out system. Submit a frame
742 * for compression and pull out the packet. */
744 img
.nplanes
= p_sys
->di
.nplanes
;
746 img
.height
= i_height
;
747 for( int i
= 0; i
< img
.nplanes
; i
++ )
749 img
.planes
[i
].data
= p_pict
->p
[i
].p_pixels
;
750 img
.planes
[i
].xdec
= p_sys
->di
.plane_info
[i
].xdec
;
751 img
.planes
[i
].ydec
= p_sys
->di
.plane_info
[i
].ydec
;
752 img
.planes
[i
].xstride
= 1;
753 img
.planes
[i
].ystride
= p_pict
->p
[i
].i_pitch
;
754 img
.planes
[i
].bitdepth
= 8; /*FIXME: support higher bit depths*/
757 if( daala_encode_img_in( p_sys
->dcx
, &img
, 0 ) < 0 )
759 msg_Warn( p_enc
, "failed encoding a frame" );
763 daala_encode_packet_out( p_sys
->dcx
, 0, &dpacket
);
765 /* Daala packet to block */
766 p_block
= block_Alloc( dpacket
.bytes
);
767 memcpy( p_block
->p_buffer
, dpacket
.packet
, dpacket
.bytes
);
768 p_block
->i_dts
= p_block
->i_pts
= p_pict
->date
;
770 if( daala_packet_iskeyframe( &dpacket
) )
771 p_block
->i_flags
|= BLOCK_FLAG_TYPE_I
;
776 static void CloseEncoder( vlc_object_t
*p_this
)
778 encoder_t
*p_enc
= (encoder_t
*)p_this
;
779 encoder_sys_t
*p_sys
= p_enc
->p_sys
;
781 daala_info_clear(&p_sys
->di
);
782 daala_comment_clear(&p_sys
->dc
);
783 daala_encode_free(p_sys
->dcx
);