1 /* GStreamer
2 * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
3 * Copyright (C) <2004> Wim Taymans <wim@fluendo.com>
4 * Copyright (C) <2005> Thomas Vander Stichele <thomas at apestaart dot org>
5 * Copyright (C) <2008> Sebastian Dröge <sebastian.droege@collabora.co.uk>
6 *
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Library General Public
9 * License as published by the Free Software Foundation; either
10 * version 2 of the License, or (at your option) any later version.
11 *
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Library General Public License for more details.
16 *
17 * You should have received a copy of the GNU Library General Public
18 * License along with this library; if not, write to the
19 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
20 * Boston, MA 02110-1301, USA.
21 */
22
23 /*
24 * Based on the lame element.
25 */
26
27 /**
28 * SECTION:element-twolame
29 * @see_also: mad, lame
30 *
31 * This element encodes raw integer audio into an MPEG-1 layer 2 (MP2) stream.
32 *
33 * <refsect2>
34 * <title>Example pipelines</title>
35 * |[
36 * gst-launch-1.0 -v audiotestsrc wave=sine num-buffers=100 ! audioconvert ! twolame ! filesink location=sine.mp2
37 * ]| Encode a test sine signal to MP2.
38 * |[
39 * gst-launch-1.0 -v alsasrc ! audioconvert ! twolame bitrate=192 ! filesink location=alsasrc.mp2
40 * ]| Record from a sound card using ALSA and encode to MP2
41 * |[
42 * gst-launch-1.0 -v filesrc location=music.wav ! decodebin ! audioconvert ! audioresample ! twolame bitrate=192 ! id3v2mux ! filesink location=music.mp2
43 * ]| Transcode from a .wav file to MP2 (the id3v2mux element is optional)
44 * |[
45 * gst-launch-1.0 -v cdda://5 ! audioconvert ! twolame bitrate=192 ! filesink location=track5.mp2
46 * ]| Encode Audio CD track 5 to MP2
47 * </refsect2>
48 *
49 */
50
51 #ifdef HAVE_CONFIG_H
52 #include "config.h"
53 #endif
54
55 #include "string.h"
56 #include "gsttwolamemp2enc.h"
57 #include "gst/gst-i18n-plugin.h"
58
59 GST_DEBUG_CATEGORY_STATIC (debug);
60 #define GST_CAT_DEFAULT debug
61
62 /* TwoLAME can do MPEG-1, MPEG-2 so it has 6 possible
63 * sample rates it supports */
64 static GstStaticPadTemplate gst_two_lame_sink_template =
65 GST_STATIC_PAD_TEMPLATE ("sink",
66 GST_PAD_SINK,
67 GST_PAD_ALWAYS,
68 GST_STATIC_CAPS ("audio/x-raw, "
69 "format = (string) { " GST_AUDIO_NE (F32) ", " GST_AUDIO_NE (S16) " }, "
70 "layout = (string) interleaved, "
71 "rate = (int) { 16000, 22050, 24000, 32000, 44100, 48000 }, "
72 "channels = (int) 1; "
73 "audio/x-raw, "
74 "format = (string) { " GST_AUDIO_NE (F32) ", " GST_AUDIO_NE (S16) " }, "
75 "layout = (string) interleaved, "
76 "rate = (int) { 16000, 22050, 24000, 32000, 44100, 48000 }, "
77 "channels = (int) 2," "channel-mask = (bitmask) 0x3")
78 );
79
80 static GstStaticPadTemplate gst_two_lame_src_template =
81 GST_STATIC_PAD_TEMPLATE ("src",
82 GST_PAD_SRC,
83 GST_PAD_ALWAYS,
84 GST_STATIC_CAPS ("audio/mpeg, "
85 "mpegversion = (int) 1, "
86 "layer = (int) 2, "
87 "rate = (int) { 16000, 22050, 24000, 32000, 44100, 48000 }, "
88 "channels = (int) [ 1, 2 ]")
89 );
90
91 static struct
92 {
93 gint mode;
94 gint psymodel;
95 gint bitrate;
96 gint padding;
97 gboolean energy_level_extension;
98 gint emphasis;
99 gboolean error_protection;
100 gboolean copyright;
101 gboolean original;
102 gboolean vbr;
103 gfloat vbr_level;
104 gfloat ath_level;
105 gint vbr_max_bitrate;
106 gboolean quick_mode;
107 gint quick_mode_count;
108 } gst_two_lame_default_settings;
109
110 /********** Define useful types for non-programmatic interfaces **********/
111 #define GST_TYPE_TWO_LAME_MODE (gst_two_lame_mode_get_type())
112 static GType
gst_two_lame_mode_get_type(void)113 gst_two_lame_mode_get_type (void)
114 {
115 static GType two_lame_mode_type = 0;
116 static const GEnumValue two_lame_modes[] = {
117 {TWOLAME_AUTO_MODE, "Auto", "auto"},
118 {TWOLAME_STEREO, "Stereo", "stereo"},
119 {TWOLAME_JOINT_STEREO, "Joint Stereo", "joint"},
120 {TWOLAME_DUAL_CHANNEL, "Dual Channel", "dual"},
121 {TWOLAME_MONO, "Mono", "mono"},
122 {0, NULL, NULL}
123 };
124
125 if (!two_lame_mode_type) {
126 two_lame_mode_type =
127 g_enum_register_static ("GstTwoLameMode", two_lame_modes);
128 }
129 return two_lame_mode_type;
130 }
131
132 #define GST_TYPE_TWO_LAME_PADDING (gst_two_lame_padding_get_type())
133 static GType
gst_two_lame_padding_get_type(void)134 gst_two_lame_padding_get_type (void)
135 {
136 static GType two_lame_padding_type = 0;
137 static const GEnumValue two_lame_padding[] = {
138 {TWOLAME_PAD_NO, "No Padding", "never"},
139 {TWOLAME_PAD_ALL, "Always Pad", "always"},
140 {0, NULL, NULL}
141 };
142
143 if (!two_lame_padding_type) {
144 two_lame_padding_type =
145 g_enum_register_static ("GstTwoLamePadding", two_lame_padding);
146 }
147 return two_lame_padding_type;
148 }
149
150 #define GST_TYPE_TWO_LAME_EMPHASIS (gst_two_lame_emphasis_get_type())
151 static GType
gst_two_lame_emphasis_get_type(void)152 gst_two_lame_emphasis_get_type (void)
153 {
154 static GType two_lame_emphasis_type = 0;
155 static const GEnumValue two_lame_emphasis[] = {
156 {TWOLAME_EMPHASIS_N, "No emphasis", "none"},
157 {TWOLAME_EMPHASIS_5, "50/15 ms", "5"},
158 {TWOLAME_EMPHASIS_C, "CCIT J.17", "ccit"},
159 {0, NULL, NULL}
160 };
161
162 if (!two_lame_emphasis_type) {
163 two_lame_emphasis_type =
164 g_enum_register_static ("GstTwoLameEmphasis", two_lame_emphasis);
165 }
166
167 return two_lame_emphasis_type;
168 }
169
170 /********** Standard stuff for signals and arguments **********/
171
172 enum
173 {
174 ARG_0,
175 ARG_MODE,
176 ARG_PSYMODEL,
177 ARG_BITRATE,
178 ARG_PADDING,
179 ARG_ENERGY_LEVEL_EXTENSION,
180 ARG_EMPHASIS,
181 ARG_ERROR_PROTECTION,
182 ARG_COPYRIGHT,
183 ARG_ORIGINAL,
184 ARG_VBR,
185 ARG_VBR_LEVEL,
186 ARG_ATH_LEVEL,
187 ARG_VBR_MAX_BITRATE,
188 ARG_QUICK_MODE,
189 ARG_QUICK_MODE_COUNT
190 };
191
192 static gboolean gst_two_lame_start (GstAudioEncoder * enc);
193 static gboolean gst_two_lame_stop (GstAudioEncoder * enc);
194 static gboolean gst_two_lame_set_format (GstAudioEncoder * enc,
195 GstAudioInfo * info);
196 static GstFlowReturn gst_two_lame_handle_frame (GstAudioEncoder * enc,
197 GstBuffer * in_buf);
198 static void gst_two_lame_flush (GstAudioEncoder * enc);
199
200 static void gst_two_lame_set_property (GObject * object, guint prop_id,
201 const GValue * value, GParamSpec * pspec);
202 static void gst_two_lame_get_property (GObject * object, guint prop_id,
203 GValue * value, GParamSpec * pspec);
204 static gboolean gst_two_lame_setup (GstTwoLame * twolame);
205
206 G_DEFINE_TYPE (GstTwoLame, gst_two_lame, GST_TYPE_AUDIO_ENCODER);
207
208 static void
gst_two_lame_release_memory(GstTwoLame * twolame)209 gst_two_lame_release_memory (GstTwoLame * twolame)
210 {
211 if (twolame->glopts) {
212 twolame_close (&twolame->glopts);
213 twolame->glopts = NULL;
214 }
215 }
216
217 static void
gst_two_lame_finalize(GObject * obj)218 gst_two_lame_finalize (GObject * obj)
219 {
220 gst_two_lame_release_memory (GST_TWO_LAME (obj));
221
222 G_OBJECT_CLASS (gst_two_lame_parent_class)->finalize (obj);
223 }
224
225 static void
gst_two_lame_class_init(GstTwoLameClass * klass)226 gst_two_lame_class_init (GstTwoLameClass * klass)
227 {
228 GObjectClass *gobject_class;
229 GstAudioEncoderClass *gstbase_class;
230
231 gobject_class = (GObjectClass *) klass;
232 gstbase_class = (GstAudioEncoderClass *) klass;
233
234 gobject_class->set_property = gst_two_lame_set_property;
235 gobject_class->get_property = gst_two_lame_get_property;
236 gobject_class->finalize = gst_two_lame_finalize;
237
238 gstbase_class->start = GST_DEBUG_FUNCPTR (gst_two_lame_start);
239 gstbase_class->stop = GST_DEBUG_FUNCPTR (gst_two_lame_stop);
240 gstbase_class->set_format = GST_DEBUG_FUNCPTR (gst_two_lame_set_format);
241 gstbase_class->handle_frame = GST_DEBUG_FUNCPTR (gst_two_lame_handle_frame);
242 gstbase_class->flush = GST_DEBUG_FUNCPTR (gst_two_lame_flush);
243
244 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_MODE,
245 g_param_spec_enum ("mode", "Mode", "Encoding mode",
246 GST_TYPE_TWO_LAME_MODE, gst_two_lame_default_settings.mode,
247 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
248
249 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_PSYMODEL,
250 g_param_spec_int ("psymodel", "Psychoacoustic Model",
251 "Psychoacoustic model used to encode the audio",
252 -1, 4, gst_two_lame_default_settings.psymodel,
253 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
254
255 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_BITRATE,
256 g_param_spec_int ("bitrate", "Bitrate (kb/s)",
257 "Bitrate in kbit/sec (8, 16, 24, 32, 40, 48, 56, 64, 80, 96, "
258 "112, 128, 144, 160, 192, 224, 256, 320, 384)",
259 8, 384, gst_two_lame_default_settings.bitrate,
260 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
261
262 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_PADDING,
263 g_param_spec_enum ("padding", "Padding", "Padding type",
264 GST_TYPE_TWO_LAME_PADDING, gst_two_lame_default_settings.padding,
265 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
266
267 g_object_class_install_property (G_OBJECT_CLASS (klass),
268 ARG_ENERGY_LEVEL_EXTENSION,
269 g_param_spec_boolean ("energy-level-extension", "Energy Level Extension",
270 "Write peak PCM level to each frame",
271 gst_two_lame_default_settings.energy_level_extension,
272 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
273
274 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_EMPHASIS,
275 g_param_spec_enum ("emphasis", "Emphasis",
276 "Pre-emphasis to apply to the decoded audio",
277 GST_TYPE_TWO_LAME_EMPHASIS, gst_two_lame_default_settings.emphasis,
278 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
279
280 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_ERROR_PROTECTION,
281 g_param_spec_boolean ("error-protection", "Error protection",
282 "Adds checksum to every frame",
283 gst_two_lame_default_settings.error_protection,
284 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
285
286 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_COPYRIGHT,
287 g_param_spec_boolean ("copyright", "Copyright", "Mark as copyright",
288 gst_two_lame_default_settings.copyright,
289 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
290
291 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_ORIGINAL,
292 g_param_spec_boolean ("original", "Original", "Mark as original",
293 gst_two_lame_default_settings.original,
294 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
295
296 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_VBR,
297 g_param_spec_boolean ("vbr", "VBR", "Enable variable bitrate mode",
298 gst_two_lame_default_settings.vbr,
299 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
300
301 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_VBR_LEVEL,
302 g_param_spec_float ("vbr-level", "VBR Level", "VBR Level",
303 -10.0, 10.0, gst_two_lame_default_settings.vbr_level,
304 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
305
306 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_ATH_LEVEL,
307 g_param_spec_float ("ath-level", "ATH Level", "ATH Level in dB",
308 -G_MAXFLOAT, G_MAXFLOAT, gst_two_lame_default_settings.ath_level,
309 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
310
311 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_VBR_MAX_BITRATE,
312 g_param_spec_int ("vbr-max-bitrate", "VBR max bitrate",
313 "Specify maximum VBR bitrate (0=off, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, "
314 "112, 128, 144, 160, 192, 224, 256, 320, 384)",
315 0, 384, gst_two_lame_default_settings.vbr_max_bitrate,
316 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
317
318 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_QUICK_MODE,
319 g_param_spec_boolean ("quick-mode", "Quick mode",
320 "Calculate Psymodel every frames",
321 gst_two_lame_default_settings.quick_mode,
322 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
323
324 g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_QUICK_MODE_COUNT,
325 g_param_spec_int ("quick-mode-count", "Quick mode count",
326 "Calculate Psymodel every n frames",
327 0, G_MAXINT, gst_two_lame_default_settings.quick_mode_count,
328 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
329
330 gst_element_class_add_static_pad_template (GST_ELEMENT_CLASS (klass),
331 &gst_two_lame_src_template);
332 gst_element_class_add_static_pad_template (GST_ELEMENT_CLASS (klass),
333 &gst_two_lame_sink_template);
334
335 gst_element_class_set_static_metadata (GST_ELEMENT_CLASS (klass),
336 "TwoLAME mp2 encoder", "Codec/Encoder/Audio",
337 "High-quality free MP2 encoder",
338 "Sebastian Dröge <sebastian.droege@collabora.co.uk>");
339 }
340
341 static gboolean
gst_two_lame_set_format(GstAudioEncoder * enc,GstAudioInfo * info)342 gst_two_lame_set_format (GstAudioEncoder * enc, GstAudioInfo * info)
343 {
344 GstTwoLame *twolame;
345 gint out_samplerate;
346 gint version;
347 GstCaps *othercaps;
348
349 twolame = GST_TWO_LAME (enc);
350
351 /* parameters already parsed for us */
352 twolame->samplerate = GST_AUDIO_INFO_RATE (info);
353 twolame->num_channels = GST_AUDIO_INFO_CHANNELS (info);
354 twolame->float_input = !GST_AUDIO_INFO_IS_INTEGER (info);
355
356 /* but we might be asked to reconfigure, so reset */
357 gst_two_lame_release_memory (twolame);
358
359 GST_DEBUG_OBJECT (twolame, "setting up twolame");
360 if (!gst_two_lame_setup (twolame))
361 goto setup_failed;
362
363 out_samplerate = twolame_get_out_samplerate (twolame->glopts);
364 if (out_samplerate == 0)
365 goto zero_output_rate;
366
367 if (out_samplerate != twolame->samplerate) {
368 GST_WARNING_OBJECT (twolame,
369 "output samplerate %d is different from incoming samplerate %d",
370 out_samplerate, twolame->samplerate);
371 }
372
373 version = twolame_get_version (twolame->glopts);
374 if (version == TWOLAME_MPEG2)
375 version = 2;
376 else
377 version = 1;
378
379 othercaps =
380 gst_caps_new_simple ("audio/mpeg",
381 "mpegversion", G_TYPE_INT, 1,
382 "mpegaudioversion", G_TYPE_INT, version,
383 "layer", G_TYPE_INT, 2,
384 "channels", G_TYPE_INT,
385 twolame->mode == TWOLAME_MONO ? 1 : twolame->num_channels, "rate",
386 G_TYPE_INT, out_samplerate, NULL);
387
388 /* and use these caps */
389 gst_audio_encoder_set_output_format (GST_AUDIO_ENCODER (twolame), othercaps);
390 gst_caps_unref (othercaps);
391
392 /* report needs to base class:
393 * hand one frame at a time, if we are pretty sure what a frame is */
394 if (out_samplerate == twolame->samplerate) {
395 gst_audio_encoder_set_frame_samples_min (enc, 1152);
396 gst_audio_encoder_set_frame_samples_max (enc, 1152);
397 gst_audio_encoder_set_frame_max (enc, 1);
398 }
399
400 return TRUE;
401
402 zero_output_rate:
403 {
404 GST_ELEMENT_ERROR (twolame, LIBRARY, SETTINGS, (NULL),
405 ("TwoLAME decided on a zero sample rate"));
406 return FALSE;
407 }
408 setup_failed:
409 {
410 GST_ELEMENT_ERROR (twolame, LIBRARY, SETTINGS,
411 (_("Failed to configure TwoLAME encoder. Check your encoding parameters.")), (NULL));
412 return FALSE;
413 }
414 }
415
416 static void
gst_two_lame_init(GstTwoLame * twolame)417 gst_two_lame_init (GstTwoLame * twolame)
418 {
419 GST_DEBUG_OBJECT (twolame, "starting initialization");
420
421 GST_PAD_SET_ACCEPT_TEMPLATE (GST_AUDIO_ENCODER_SINK_PAD (twolame));
422
423 twolame->mode = gst_two_lame_default_settings.mode;
424 twolame->psymodel = gst_two_lame_default_settings.psymodel;
425 twolame->bitrate = gst_two_lame_default_settings.bitrate;
426 twolame->padding = gst_two_lame_default_settings.padding;
427 twolame->energy_level_extension =
428 gst_two_lame_default_settings.energy_level_extension;
429 twolame->emphasis = gst_two_lame_default_settings.emphasis;
430 twolame->error_protection = gst_two_lame_default_settings.error_protection;
431 twolame->copyright = gst_two_lame_default_settings.copyright;
432 twolame->original = gst_two_lame_default_settings.original;
433 twolame->vbr = gst_two_lame_default_settings.vbr;
434 twolame->vbr_level = gst_two_lame_default_settings.vbr_level;
435 twolame->ath_level = gst_two_lame_default_settings.ath_level;
436 twolame->vbr_max_bitrate = gst_two_lame_default_settings.vbr_max_bitrate;
437 twolame->quick_mode = gst_two_lame_default_settings.quick_mode;
438 twolame->quick_mode_count = gst_two_lame_default_settings.quick_mode_count;
439
440 GST_DEBUG_OBJECT (twolame, "done initializing");
441 }
442
443 static gboolean
gst_two_lame_start(GstAudioEncoder * enc)444 gst_two_lame_start (GstAudioEncoder * enc)
445 {
446 GstTwoLame *twolame = GST_TWO_LAME (enc);
447
448 GST_DEBUG_OBJECT (twolame, "start");
449 return TRUE;
450 }
451
452 static gboolean
gst_two_lame_stop(GstAudioEncoder * enc)453 gst_two_lame_stop (GstAudioEncoder * enc)
454 {
455 GstTwoLame *twolame = GST_TWO_LAME (enc);
456
457 GST_DEBUG_OBJECT (twolame, "stop");
458
459 gst_two_lame_release_memory (twolame);
460 return TRUE;
461 }
462
463 /* <php-emulation-mode>three underscores for ___rate is really really really
464 * private as opposed to one underscore<php-emulation-mode> */
465 /* call this MACRO outside of the NULL state so that we have a higher chance
466 * of actually having a pipeline and bus to get the message through */
467
468 #define CHECK_AND_FIXUP_BITRATE(obj,param,rate) \
469 G_STMT_START { \
470 gint ___rate = rate; \
471 gint maxrate = 320; \
472 gint multiplier = 64; \
473 if (rate <= 64) { \
474 maxrate = 64; multiplier = 8; \
475 if ((rate % 8) != 0) ___rate = GST_ROUND_UP_8 (rate); \
476 } else if (rate <= 144) { \
477 maxrate = 144; multiplier = 16; \
478 if ((rate % 16) != 0) ___rate = GST_ROUND_UP_16 (rate); \
479 } else if (rate <= 256) { \
480 maxrate = 256; multiplier = 32; \
481 if ((rate % 32) != 0) ___rate = GST_ROUND_UP_32 (rate); \
482 } else if (rate <= 384) { \
483 maxrate = 384; multiplier = 64; \
484 if ((rate % 64) != 0) ___rate = GST_ROUND_UP_64 (rate); \
485 } \
486 if (___rate != rate) { \
487 GST_ELEMENT_WARNING (obj, LIBRARY, SETTINGS, \
488 (_("The requested bitrate %d kbit/s for property '%s' " \
489 "is not allowed. " \
490 "The bitrate was changed to %d kbit/s."), rate, \
491 param, ___rate), \
492 ("A bitrate below %d should be a multiple of %d.", \
493 maxrate, multiplier)); \
494 rate = ___rate; \
495 } \
496 } G_STMT_END
497
498 static void
gst_two_lame_set_property(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)499 gst_two_lame_set_property (GObject * object, guint prop_id,
500 const GValue * value, GParamSpec * pspec)
501 {
502 GstTwoLame *twolame = GST_TWO_LAME (object);
503
504 switch (prop_id) {
505 case ARG_MODE:
506 twolame->mode = g_value_get_enum (value);
507 break;
508 case ARG_PSYMODEL:
509 twolame->psymodel = g_value_get_int (value);
510 break;
511 case ARG_BITRATE:
512 twolame->bitrate = g_value_get_int (value);
513 break;
514 case ARG_PADDING:
515 twolame->padding = g_value_get_enum (value);
516 break;
517 case ARG_ENERGY_LEVEL_EXTENSION:
518 twolame->energy_level_extension = g_value_get_boolean (value);
519 break;
520 case ARG_EMPHASIS:
521 twolame->emphasis = g_value_get_enum (value);
522 break;
523 case ARG_ERROR_PROTECTION:
524 twolame->error_protection = g_value_get_boolean (value);
525 break;
526 case ARG_COPYRIGHT:
527 twolame->copyright = g_value_get_boolean (value);
528 break;
529 case ARG_ORIGINAL:
530 twolame->original = g_value_get_boolean (value);
531 break;
532 case ARG_VBR:
533 twolame->vbr = g_value_get_boolean (value);
534 break;
535 case ARG_VBR_LEVEL:
536 twolame->vbr_level = g_value_get_float (value);
537 break;
538 case ARG_ATH_LEVEL:
539 twolame->ath_level = g_value_get_float (value);
540 break;
541 case ARG_VBR_MAX_BITRATE:
542 twolame->vbr_max_bitrate = g_value_get_int (value);
543 break;
544 case ARG_QUICK_MODE:
545 twolame->quick_mode = g_value_get_boolean (value);
546 break;
547 case ARG_QUICK_MODE_COUNT:
548 twolame->quick_mode_count = g_value_get_int (value);
549 break;
550 default:
551 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
552 break;
553 }
554 }
555
556 static void
gst_two_lame_get_property(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)557 gst_two_lame_get_property (GObject * object, guint prop_id, GValue * value,
558 GParamSpec * pspec)
559 {
560 GstTwoLame *twolame = GST_TWO_LAME (object);
561
562 switch (prop_id) {
563 case ARG_MODE:
564 g_value_set_enum (value, twolame->mode);
565 break;
566 case ARG_PSYMODEL:
567 g_value_set_int (value, twolame->psymodel);
568 break;
569 case ARG_BITRATE:
570 g_value_set_int (value, twolame->bitrate);
571 break;
572 case ARG_PADDING:
573 g_value_set_enum (value, twolame->padding);
574 break;
575 case ARG_ENERGY_LEVEL_EXTENSION:
576 g_value_set_boolean (value, twolame->energy_level_extension);
577 break;
578 case ARG_EMPHASIS:
579 g_value_set_enum (value, twolame->emphasis);
580 break;
581 case ARG_ERROR_PROTECTION:
582 g_value_set_boolean (value, twolame->error_protection);
583 break;
584 case ARG_COPYRIGHT:
585 g_value_set_boolean (value, twolame->copyright);
586 break;
587 case ARG_ORIGINAL:
588 g_value_set_boolean (value, twolame->original);
589 break;
590 case ARG_VBR:
591 g_value_set_boolean (value, twolame->vbr);
592 break;
593 case ARG_VBR_LEVEL:
594 g_value_set_float (value, twolame->vbr_level);
595 break;
596 case ARG_ATH_LEVEL:
597 g_value_set_float (value, twolame->ath_level);
598 break;
599 case ARG_VBR_MAX_BITRATE:
600 g_value_set_int (value, twolame->vbr_max_bitrate);
601 break;
602 case ARG_QUICK_MODE:
603 g_value_set_boolean (value, twolame->quick_mode);
604 break;
605 case ARG_QUICK_MODE_COUNT:
606 g_value_set_int (value, twolame->quick_mode_count);
607 break;
608 default:
609 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
610 break;
611 }
612 }
613
614 static GstFlowReturn
gst_two_lame_flush_full(GstTwoLame * lame,gboolean push)615 gst_two_lame_flush_full (GstTwoLame * lame, gboolean push)
616 {
617 GstBuffer *buf;
618 GstMapInfo map;
619 gint size;
620 GstFlowReturn result = GST_FLOW_OK;
621
622 if (!lame->glopts)
623 return GST_FLOW_OK;
624
625 buf = gst_buffer_new_and_alloc (16384);
626 gst_buffer_map (buf, &map, GST_MAP_WRITE);
627 size = twolame_encode_flush (lame->glopts, map.data, 16384);
628 gst_buffer_unmap (buf, &map);
629
630 if (size > 0 && push) {
631 gst_buffer_set_size (buf, size);
632 GST_DEBUG_OBJECT (lame, "pushing final packet of %u bytes", size);
633 result = gst_audio_encoder_finish_frame (GST_AUDIO_ENCODER (lame), buf, -1);
634 } else {
635 GST_DEBUG_OBJECT (lame, "no final packet (size=%d, push=%d)", size, push);
636 gst_buffer_unref (buf);
637 result = GST_FLOW_OK;
638 }
639 return result;
640 }
641
642 static void
gst_two_lame_flush(GstAudioEncoder * enc)643 gst_two_lame_flush (GstAudioEncoder * enc)
644 {
645 gst_two_lame_flush_full (GST_TWO_LAME (enc), FALSE);
646 }
647
648 static GstFlowReturn
gst_two_lame_handle_frame(GstAudioEncoder * enc,GstBuffer * buf)649 gst_two_lame_handle_frame (GstAudioEncoder * enc, GstBuffer * buf)
650 {
651 GstTwoLame *twolame;
652 gint mp3_buffer_size, mp3_size;
653 GstBuffer *mp3_buf;
654 GstFlowReturn result;
655 gint num_samples;
656 GstMapInfo map, mp3_map;
657
658 twolame = GST_TWO_LAME (enc);
659
660 /* squeeze remaining and push */
661 if (G_UNLIKELY (buf == NULL))
662 return gst_two_lame_flush_full (twolame, TRUE);
663
664 gst_buffer_map (buf, &map, GST_MAP_READ);
665
666 if (twolame->float_input)
667 num_samples = map.size / 4;
668 else
669 num_samples = map.size / 2;
670
671 /* allocate space for output */
672 mp3_buffer_size = 1.25 * num_samples + 16384;
673 mp3_buf = gst_buffer_new_and_alloc (mp3_buffer_size);
674 gst_buffer_map (mp3_buf, &mp3_map, GST_MAP_WRITE);
675
676 if (twolame->num_channels == 1) {
677 if (twolame->float_input)
678 mp3_size = twolame_encode_buffer_float32 (twolame->glopts,
679 (float *) map.data,
680 (float *) map.data, num_samples, mp3_map.data, mp3_buffer_size);
681 else
682 mp3_size = twolame_encode_buffer (twolame->glopts,
683 (short int *) map.data,
684 (short int *) map.data, num_samples, mp3_map.data, mp3_buffer_size);
685 } else {
686 if (twolame->float_input)
687 mp3_size = twolame_encode_buffer_float32_interleaved (twolame->glopts,
688 (float *) map.data,
689 num_samples / twolame->num_channels, mp3_map.data, mp3_buffer_size);
690 else
691 mp3_size = twolame_encode_buffer_interleaved (twolame->glopts,
692 (short int *) map.data,
693 num_samples / twolame->num_channels, mp3_map.data, mp3_buffer_size);
694 }
695
696 GST_LOG_OBJECT (twolame, "encoded %" G_GSIZE_FORMAT " bytes of audio "
697 "to %d bytes of mp3", map.size, mp3_size);
698
699 gst_buffer_unmap (buf, &map);
700 gst_buffer_unmap (mp3_buf, &mp3_map);
701
702 if (mp3_size > 0) {
703 gst_buffer_set_size (mp3_buf, mp3_size);
704 result = gst_audio_encoder_finish_frame (enc, mp3_buf, -1);
705 } else {
706 if (mp3_size < 0) {
707 /* eat error ? */
708 g_warning ("error %d", mp3_size);
709 }
710 gst_buffer_unref (mp3_buf);
711 result = GST_FLOW_OK;
712 }
713
714 return result;
715 }
716
717 /* set up the encoder state */
718 static gboolean
gst_two_lame_setup(GstTwoLame * twolame)719 gst_two_lame_setup (GstTwoLame * twolame)
720 {
721
722 #define CHECK_ERROR(command) G_STMT_START {\
723 if ((command) < 0) { \
724 GST_ERROR_OBJECT (twolame, "setup failed: " G_STRINGIFY (command)); \
725 return FALSE; \
726 } \
727 }G_STMT_END
728
729 int retval;
730 GstCaps *allowed_caps;
731
732 GST_DEBUG_OBJECT (twolame, "starting setup");
733
734 /* check if we're already setup; if we are, we might want to check
735 * if this initialization is compatible with the previous one */
736 /* FIXME: do this */
737 if (twolame->setup) {
738 GST_WARNING_OBJECT (twolame, "already setup");
739 twolame->setup = FALSE;
740 }
741
742 twolame->glopts = twolame_init ();
743
744 if (twolame->glopts == NULL)
745 return FALSE;
746
747 /* copy the parameters over */
748 twolame_set_in_samplerate (twolame->glopts, twolame->samplerate);
749
750 /* let twolame choose default samplerate unless outgoing sample rate is fixed */
751 allowed_caps = gst_pad_get_allowed_caps (GST_AUDIO_ENCODER_SRC_PAD (twolame));
752
753 if (allowed_caps != NULL) {
754 GstStructure *structure;
755 gint samplerate;
756
757 structure = gst_caps_get_structure (allowed_caps, 0);
758
759 if (gst_structure_get_int (structure, "rate", &samplerate)) {
760 GST_DEBUG_OBJECT (twolame,
761 "Setting sample rate to %d as fixed in src caps", samplerate);
762 twolame_set_out_samplerate (twolame->glopts, samplerate);
763 } else {
764 GST_DEBUG_OBJECT (twolame, "Letting twolame choose sample rate");
765 twolame_set_out_samplerate (twolame->glopts, 0);
766 }
767 gst_caps_unref (allowed_caps);
768 allowed_caps = NULL;
769 } else {
770 GST_DEBUG_OBJECT (twolame,
771 "No peer yet, letting twolame choose sample rate");
772 twolame_set_out_samplerate (twolame->glopts, 0);
773 }
774
775 /* force mono encoding if we only have one channel */
776 if (twolame->num_channels == 1)
777 twolame->mode = 3;
778
779 /* Fix bitrates and MPEG version */
780
781 CHECK_ERROR (twolame_set_num_channels (twolame->glopts,
782 twolame->num_channels));
783
784 CHECK_ERROR (twolame_set_mode (twolame->glopts, twolame->mode));
785 CHECK_ERROR (twolame_set_psymodel (twolame->glopts, twolame->psymodel));
786 CHECK_AND_FIXUP_BITRATE (twolame, "bitrate", twolame->bitrate);
787 CHECK_ERROR (twolame_set_bitrate (twolame->glopts, twolame->bitrate));
788 CHECK_ERROR (twolame_set_padding (twolame->glopts, twolame->padding));
789 CHECK_ERROR (twolame_set_energy_levels (twolame->glopts,
790 twolame->energy_level_extension));
791 CHECK_ERROR (twolame_set_emphasis (twolame->glopts, twolame->emphasis));
792 CHECK_ERROR (twolame_set_error_protection (twolame->glopts,
793 twolame->error_protection));
794 CHECK_ERROR (twolame_set_copyright (twolame->glopts, twolame->copyright));
795 CHECK_ERROR (twolame_set_original (twolame->glopts, twolame->original));
796 CHECK_ERROR (twolame_set_VBR (twolame->glopts, twolame->vbr));
797 CHECK_ERROR (twolame_set_VBR_level (twolame->glopts, twolame->vbr_level));
798 CHECK_ERROR (twolame_set_ATH_level (twolame->glopts, twolame->ath_level));
799 CHECK_AND_FIXUP_BITRATE (twolame, "vbr-max-bitrate",
800 twolame->vbr_max_bitrate);
801 CHECK_ERROR (twolame_set_VBR_max_bitrate_kbps (twolame->glopts,
802 twolame->vbr_max_bitrate));
803 CHECK_ERROR (twolame_set_quick_mode (twolame->glopts, twolame->quick_mode));
804 CHECK_ERROR (twolame_set_quick_count (twolame->glopts,
805 twolame->quick_mode_count));
806
807 /* initialize the twolame encoder */
808 if ((retval = twolame_init_params (twolame->glopts)) >= 0) {
809 twolame->setup = TRUE;
810 /* FIXME: it would be nice to print out the mode here */
811 GST_INFO ("twolame encoder setup (%d kbit/s, %d Hz, %d channels)",
812 twolame->bitrate, twolame->samplerate, twolame->num_channels);
813 } else {
814 GST_ERROR_OBJECT (twolame, "twolame_init_params returned %d", retval);
815 }
816
817 GST_DEBUG_OBJECT (twolame, "done with setup");
818
819 return twolame->setup;
820 #undef CHECK_ERROR
821 }
822
823 static gboolean
gst_two_lame_get_default_settings(void)824 gst_two_lame_get_default_settings (void)
825 {
826 twolame_options *glopts = NULL;
827
828 glopts = twolame_init ();
829 if (glopts == NULL) {
830 GST_ERROR ("Couldn't initialize TwoLAME");
831 return FALSE;
832 }
833
834 twolame_set_num_channels (glopts, 2);
835 twolame_set_in_samplerate (glopts, 44100);
836
837 if (twolame_init_params (glopts) != 0) {
838 GST_ERROR ("Couldn't set default parameters");
839 return FALSE;
840 }
841
842 gst_two_lame_default_settings.mode = TWOLAME_JOINT_STEREO; /* twolame_get_mode (glopts); */
843 gst_two_lame_default_settings.psymodel = twolame_get_psymodel (glopts);
844 gst_two_lame_default_settings.bitrate = twolame_get_bitrate (glopts);
845 gst_two_lame_default_settings.padding = twolame_get_padding (glopts);
846 gst_two_lame_default_settings.energy_level_extension =
847 twolame_get_energy_levels (glopts);
848 gst_two_lame_default_settings.emphasis = twolame_get_emphasis (glopts);
849 gst_two_lame_default_settings.error_protection =
850 twolame_get_error_protection (glopts);
851 gst_two_lame_default_settings.copyright = twolame_get_copyright (glopts);
852 gst_two_lame_default_settings.original = twolame_get_original (glopts);
853 gst_two_lame_default_settings.vbr = twolame_get_VBR (glopts);
854 gst_two_lame_default_settings.vbr_level = twolame_get_VBR_level (glopts);
855 gst_two_lame_default_settings.ath_level = twolame_get_ATH_level (glopts);
856 gst_two_lame_default_settings.vbr_max_bitrate =
857 twolame_get_VBR_max_bitrate_kbps (glopts);
858 gst_two_lame_default_settings.quick_mode = twolame_get_quick_mode (glopts);
859 gst_two_lame_default_settings.quick_mode_count =
860 twolame_get_quick_count (glopts);
861
862 twolame_close (&glopts);
863
864 return TRUE;
865 }
866
867 static gboolean
plugin_init(GstPlugin * plugin)868 plugin_init (GstPlugin * plugin)
869 {
870 GST_DEBUG_CATEGORY_INIT (debug, "twolame", 0, "twolame mp2 encoder");
871
872 if (!gst_two_lame_get_default_settings ())
873 return FALSE;
874
875 #ifdef ENABLE_NLS
876 GST_DEBUG ("binding text domain %s to locale dir %s", GETTEXT_PACKAGE,
877 LOCALEDIR);
878 bindtextdomain (GETTEXT_PACKAGE, LOCALEDIR);
879 bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
880 #endif /* ENABLE_NLS */
881
882 if (!gst_element_register (plugin, "twolamemp2enc", GST_RANK_PRIMARY,
883 GST_TYPE_TWO_LAME))
884 return FALSE;
885
886 return TRUE;
887 }
888
889 GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
890 GST_VERSION_MINOR,
891 twolame,
892 "Encode MP2s with TwoLAME",
893 plugin_init, VERSION, "LGPL", GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN);
894