123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640 |
- /*
- * Copyright (C) 2003-2007 Benny Prijono <benny@prijono.org>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- */
- /*
- * Contributed by:
- * Toni < buldozer at aufbix dot org >
- */
- #include "mp3_port.h"
- #include <pjmedia/errno.h>
- #include <pj/assert.h>
- #include <pj/file_access.h>
- #include <pj/file_io.h>
- #include <pj/log.h>
- #include <pj/pool.h>
- #include <pj/string.h>
- #include <pj/unicode.h>
- /* Include BladeDLL declarations */
- #include "BladeMP3EncDLL.h"
- #define THIS_FILE "mp3_writer.c"
- #define SIGNATURE PJMEDIA_SIG_CLASS_PORT_AUD('M','W')
- #define BYTES_PER_SAMPLE 2
- static struct BladeDLL
- {
- void *hModule;
- int refCount;
- BEINITSTREAM beInitStream;
- BEENCODECHUNK beEncodeChunk;
- BEDEINITSTREAM beDeinitStream;
- BECLOSESTREAM beCloseStream;
- BEVERSION beVersion;
- BEWRITEVBRHEADER beWriteVBRHeader;
- BEWRITEINFOTAG beWriteInfoTag;
- } BladeDLL;
- struct mp3_file_port
- {
- pjmedia_port base;
- pj_size_t total;
- pj_oshandle_t fd;
- pj_size_t cb_size;
- pj_status_t (*cb)(pjmedia_port*, void*);
- pj_bool_t subscribed;
- pj_bool_t cb_called;
- void (*cb2)(pjmedia_port*, void*);
- unsigned silence_duration;
- pj_str_t mp3_filename;
- pjmedia_mp3_encoder_option mp3_option;
- unsigned mp3_samples_per_frame;
- pj_int16_t *mp3_sample_buf;
- unsigned mp3_sample_pos;
- HBE_STREAM mp3_stream;
- unsigned char *mp3_buf;
- };
- static pj_status_t file_put_frame(pjmedia_port *this_port,
- const pjmedia_frame *frame);
- static pj_status_t file_get_frame(pjmedia_port *this_port,
- pjmedia_frame *frame);
- static pj_status_t file_on_destroy(pjmedia_port *this_port);
- #if defined(PJ_WIN32) || defined(_WIN32) || defined(WIN32)
- #include <windows.h>
- #define DLL_NAME PJ_T("LAME_ENC.DLL")
- /*
- * Load BladeEncoder DLL.
- */
- static pj_status_t init_blade_dll(void)
- {
- if (BladeDLL.refCount == 0) {
- #define GET_PROC(type, name) \
- BladeDLL.name = (type)GetProcAddress(BladeDLL.hModule, PJ_T(#name)); \
- if (BladeDLL.name == NULL) { \
- PJ_LOG(1,(THIS_FILE, "Unable to find %s in %s", #name, DLL_NAME)); \
- return PJ_RETURN_OS_ERROR(GetLastError()); \
- }
- BE_VERSION beVersion;
- BladeDLL.hModule = (void*)LoadLibrary(DLL_NAME);
- if (BladeDLL.hModule == NULL) {
- pj_status_t status = PJ_RETURN_OS_ERROR(GetLastError());
- char errmsg[PJ_ERR_MSG_SIZE];
- pj_strerror(status, errmsg, sizeof(errmsg));
- PJ_LOG(1,(THIS_FILE, "Unable to load %s: %s", DLL_NAME, errmsg));
- return status;
- }
- GET_PROC(BEINITSTREAM, beInitStream);
- GET_PROC(BEENCODECHUNK, beEncodeChunk);
- GET_PROC(BEDEINITSTREAM, beDeinitStream);
- GET_PROC(BECLOSESTREAM, beCloseStream);
- GET_PROC(BEVERSION, beVersion);
- GET_PROC(BEWRITEVBRHEADER, beWriteVBRHeader);
- GET_PROC(BEWRITEINFOTAG, beWriteInfoTag);
- #undef GET_PROC
- BladeDLL.beVersion(&beVersion);
- PJ_LOG(4,(THIS_FILE, "%s encoder v%d.%d loaded (%s)", DLL_NAME,
- beVersion.byMajorVersion, beVersion.byMinorVersion,
- beVersion.zHomepage));
- }
- ++BladeDLL.refCount;
- return PJ_SUCCESS;
- }
- /*
- * Decrement the reference counter of the DLL.
- */
- static void deinit_blade_dll()
- {
- --BladeDLL.refCount;
- if (BladeDLL.refCount == 0 && BladeDLL.hModule) {
- FreeLibrary(BladeDLL.hModule);
- BladeDLL.hModule = NULL;
- PJ_LOG(4,(THIS_FILE, "%s unloaded", DLL_NAME));
- }
- }
- #else
- static pj_status_t init_blade_dll(void)
- {
- PJ_LOG(1,(THIS_FILE, "Error: MP3 writer port only works on Windows for now"));
- return PJ_ENOTSUP;
- }
- static void deinit_blade_dll()
- {
- }
- #endif
- /*
- * Initialize MP3 encoder.
- */
- static pj_status_t init_mp3_encoder(struct mp3_file_port *fport,
- pj_pool_t *pool)
- {
- BE_CONFIG LConfig;
- unsigned long InSamples;
- unsigned long OutBuffSize;
- long MP3Err;
- /*
- * Initialize encoder configuration.
- */
- pj_bzero(&LConfig, sizeof(BE_CONFIG));
- LConfig.dwConfig = BE_CONFIG_LAME;
- LConfig.format.LHV1.dwStructVersion = 1;
- LConfig.format.LHV1.dwStructSize = sizeof(BE_CONFIG);
- LConfig.format.LHV1.dwSampleRate = PJMEDIA_PIA_SRATE(&fport->base.info);
- LConfig.format.LHV1.dwReSampleRate = 0;
- if (PJMEDIA_PIA_CCNT(&fport->base.info)==1)
- LConfig.format.LHV1.nMode = BE_MP3_MODE_MONO;
- else if (PJMEDIA_PIA_CCNT(&fport->base.info)==2)
- LConfig.format.LHV1.nMode = BE_MP3_MODE_STEREO;
- else
- return PJMEDIA_ENCCHANNEL;
- LConfig.format.LHV1.dwBitrate = fport->mp3_option.bit_rate / 1000;
- LConfig.format.LHV1.nPreset = LQP_NOPRESET;
- LConfig.format.LHV1.bCopyright = 0;
- LConfig.format.LHV1.bCRC = 1;
- LConfig.format.LHV1.bOriginal = 1;
- LConfig.format.LHV1.bPrivate = 0;
- if (!fport->mp3_option.vbr) {
- LConfig.format.LHV1.nVbrMethod = VBR_METHOD_NONE;
- LConfig.format.LHV1.bWriteVBRHeader = 0;
- LConfig.format.LHV1.bEnableVBR = 0;
- } else {
- LConfig.format.LHV1.nVbrMethod = VBR_METHOD_DEFAULT;
- LConfig.format.LHV1.bWriteVBRHeader = 1;
- LConfig.format.LHV1.dwVbrAbr_bps = fport->mp3_option.bit_rate;
- LConfig.format.LHV1.nVBRQuality = (pj_uint16_t)
- fport->mp3_option.quality;
- LConfig.format.LHV1.bEnableVBR = 1;
- }
- LConfig.format.LHV1.nQuality = (pj_uint16_t)
- (((0-fport->mp3_option.quality-1)<<8) |
- fport->mp3_option.quality);
- /*
- * Init MP3 stream.
- */
- InSamples = 0;
- MP3Err = BladeDLL.beInitStream(&LConfig, &InSamples, &OutBuffSize,
- &fport->mp3_stream);
- if (MP3Err != BE_ERR_SUCCESSFUL)
- return PJMEDIA_ERROR;
- /*
- * Allocate sample buffer.
- */
- fport->mp3_samples_per_frame = (unsigned)InSamples;
- fport->mp3_sample_buf = pj_pool_alloc(pool, fport->mp3_samples_per_frame * 2);
- if (!fport->mp3_sample_buf)
- return PJ_ENOMEM;
- /*
- * Allocate encoded MP3 buffer.
- */
- fport->mp3_buf = pj_pool_alloc(pool, (pj_size_t)OutBuffSize);
- if (fport->mp3_buf == NULL)
- return PJ_ENOMEM;
-
- return PJ_SUCCESS;
- }
- /*
- * Create MP3 file writer port.
- */
- PJ_DEF(pj_status_t)
- pjmedia_mp3_writer_port_create( pj_pool_t *pool,
- const char *filename,
- unsigned sampling_rate,
- unsigned channel_count,
- unsigned samples_per_frame,
- unsigned bits_per_sample,
- const pjmedia_mp3_encoder_option *param_option,
- pjmedia_port **p_port )
- {
- struct mp3_file_port *fport;
- pj_status_t status;
- status = init_blade_dll();
- if (status != PJ_SUCCESS)
- return status;
- /* Check arguments. */
- PJ_ASSERT_RETURN(pool && filename && p_port, PJ_EINVAL);
- /* Only supports 16bits per sample for now. */
- PJ_ASSERT_RETURN(bits_per_sample == 16, PJ_EINVAL);
- /* Create file port instance. */
- fport = pj_pool_zalloc(pool, sizeof(struct mp3_file_port));
- PJ_ASSERT_RETURN(fport != NULL, PJ_ENOMEM);
- /* Initialize port info. */
- pj_strdup2_with_null(pool, &fport->mp3_filename, filename);
- pjmedia_port_info_init(&fport->base.info, &fport->mp3_filename, SIGNATURE,
- sampling_rate, channel_count, bits_per_sample,
- samples_per_frame);
- fport->base.get_frame = &file_get_frame;
- fport->base.put_frame = &file_put_frame;
- fport->base.on_destroy = &file_on_destroy;
- /* Open file in write and read mode.
- * We need the read mode because we'll modify the WAVE header once
- * the recording has completed.
- */
- status = pj_file_open(pool, filename, PJ_O_WRONLY | PJ_O_CLOEXEC,
- &fport->fd);
- if (status != PJ_SUCCESS) {
- deinit_blade_dll();
- return status;
- }
- /* Copy and initialize option with default settings */
- if (param_option) {
- pj_memcpy(&fport->mp3_option, param_option,
- sizeof(pjmedia_mp3_encoder_option));
- } else {
- pj_bzero(&fport->mp3_option, sizeof(pjmedia_mp3_encoder_option));
- fport->mp3_option.vbr = PJ_TRUE;
- }
- /* Calculate bitrate if it's not specified, only if it's not VBR. */
- if (fport->mp3_option.bit_rate == 0 && !fport->mp3_option.vbr)
- fport->mp3_option.bit_rate = sampling_rate * channel_count;
- /* Set default quality if it's not specified */
- if (fport->mp3_option.quality == 0)
- fport->mp3_option.quality = 2;
- /* Init mp3 encoder */
- status = init_mp3_encoder(fport, pool);
- if (status != PJ_SUCCESS) {
- pj_file_close(fport->fd);
- deinit_blade_dll();
- return status;
- }
- /* Done. */
- *p_port = &fport->base;
- PJ_LOG(4,(THIS_FILE,
- "MP3 file writer '%.*s' created: samp.rate=%dKHz, "
- "bitrate=%dkbps%s, quality=%d",
- (int)fport->base.info.name.slen,
- fport->base.info.name.ptr,
- PJMEDIA_PIA_SRATE(&fport->base.info),
- fport->mp3_option.bit_rate/1000,
- (fport->mp3_option.vbr ? " (VBR)" : ""),
- fport->mp3_option.quality));
- return PJ_SUCCESS;
- }
- #if !DEPRECATED_FOR_TICKET_2251
- /*
- * Register callback.
- */
- PJ_DEF(pj_status_t)
- pjmedia_mp3_writer_port_set_cb( pjmedia_port *port,
- pj_size_t pos,
- void *user_data,
- pj_status_t (*cb)(pjmedia_port *port,
- void *usr_data))
- {
- struct mp3_file_port *fport;
- /* Sanity check */
- PJ_ASSERT_RETURN(port && cb, PJ_EINVAL);
- /* Check that this is really a writer port */
- PJ_ASSERT_RETURN(port->info.signature == SIGNATURE, PJ_EINVALIDOP);
- PJ_LOG(1, (THIS_FILE, "pjmedia_mp3_writer_port_set_cb() is deprecated. "
- "Use pjmedia_mp3_writer_port_set_cb2() instead."));
- fport = (struct mp3_file_port*) port;
- fport->cb_size = pos;
- fport->base.port_data.pdata = user_data;
- fport->cb = cb;
- return PJ_SUCCESS;
- }
- #endif
- /*
- * Register callback.
- */
- PJ_DEF(pj_status_t)
- pjmedia_mp3_writer_port_set_cb2(pjmedia_port *port,
- pj_size_t pos,
- void *user_data,
- void (*cb)(pjmedia_port *port,
- void *usr_data))
- {
- struct mp3_file_port *fport;
- /* Sanity check */
- PJ_ASSERT_RETURN(port && cb, PJ_EINVAL);
- /* Check that this is really a writer port */
- PJ_ASSERT_RETURN(port->info.signature == SIGNATURE, PJ_EINVALIDOP);
- fport = (struct mp3_file_port*) port;
- fport->cb_size = pos;
- fport->base.port_data.pdata = user_data;
- fport->cb2 = cb;
- fport->cb_called = PJ_FALSE;
- return PJ_SUCCESS;
- }
- static pj_status_t file_on_event(pjmedia_event *event,
- void *user_data)
- {
- struct file_port *fport = (struct file_port*)user_data;
- if (event->type == PJMEDIA_EVENT_CALLBACK) {
- if (fport->cb2)
- (*fport->cb2)(&fport->base, fport->base.port_data.pdata);
- }
-
- return PJ_SUCCESS;
- }
- /*
- * Put a frame into the buffer. When the buffer is full, flush the buffer
- * to the file.
- */
- static pj_status_t file_put_frame(pjmedia_port *this_port,
- const pjmedia_frame *frame)
- {
- struct mp3_file_port *fport = (struct mp3_file_port *)this_port;
- unsigned long MP3Err;
- pj_ssize_t bytes;
- pj_status_t status;
- unsigned long WriteSize;
- /* Record silence if input is no-frame */
- if (frame->type == PJMEDIA_FRAME_TYPE_NONE || frame->size == 0) {
- unsigned samples_left = PJMEDIA_PIA_SPF(&fport->base.info);
- unsigned samples_copied = 0;
- /* Only want to record at most 1 second of silence */
- if (fport->silence_duration >= PJMEDIA_PIA_SRATE(&fport->base.info))
- return PJ_SUCCESS;
- while (samples_left) {
- unsigned samples_needed = fport->mp3_samples_per_frame -
- fport->mp3_sample_pos;
- if (samples_needed > samples_left)
- samples_needed = samples_left;
- pjmedia_zero_samples(fport->mp3_sample_buf + fport->mp3_sample_pos,
- samples_needed);
- fport->mp3_sample_pos += samples_needed;
- samples_left -= samples_needed;
- samples_copied += samples_needed;
- /* Encode if we have full frame */
- if (fport->mp3_sample_pos == fport->mp3_samples_per_frame) {
-
- /* Clear position */
- fport->mp3_sample_pos = 0;
- /* Encode ! */
- MP3Err = BladeDLL.beEncodeChunk(fport->mp3_stream,
- fport->mp3_samples_per_frame,
- fport->mp3_sample_buf,
- fport->mp3_buf,
- &WriteSize);
- if (MP3Err != BE_ERR_SUCCESSFUL)
- return PJMEDIA_ERROR;
- /* Write the chunk */
- bytes = WriteSize;
- status = pj_file_write(fport->fd, fport->mp3_buf, &bytes);
- if (status != PJ_SUCCESS)
- return status;
- /* Increment total written. */
- fport->total += bytes;
- }
- }
- fport->silence_duration += PJMEDIA_PIA_SPF(&fport->base.info);
- }
- /* If encoder is expecting different sample size, then we need to
- * buffer the samples.
- */
- else if (fport->mp3_samples_per_frame !=
- PJMEDIA_PIA_SPF(&fport->base.info))
- {
- unsigned samples_left = frame->size / 2;
- unsigned samples_copied = 0;
- const pj_int16_t *src_samples = frame->buf;
- fport->silence_duration = 0;
- while (samples_left) {
- unsigned samples_needed = fport->mp3_samples_per_frame -
- fport->mp3_sample_pos;
- if (samples_needed > samples_left)
- samples_needed = samples_left;
- pjmedia_copy_samples(fport->mp3_sample_buf + fport->mp3_sample_pos,
- src_samples + samples_copied,
- samples_needed);
- fport->mp3_sample_pos += samples_needed;
- samples_left -= samples_needed;
- samples_copied += samples_needed;
- /* Encode if we have full frame */
- if (fport->mp3_sample_pos == fport->mp3_samples_per_frame) {
-
- /* Clear position */
- fport->mp3_sample_pos = 0;
- /* Encode ! */
- MP3Err = BladeDLL.beEncodeChunk(fport->mp3_stream,
- fport->mp3_samples_per_frame,
- fport->mp3_sample_buf,
- fport->mp3_buf,
- &WriteSize);
- if (MP3Err != BE_ERR_SUCCESSFUL)
- return PJMEDIA_ERROR;
- /* Write the chunk */
- bytes = WriteSize;
- status = pj_file_write(fport->fd, fport->mp3_buf, &bytes);
- if (status != PJ_SUCCESS)
- return status;
- /* Increment total written. */
- fport->total += bytes;
- }
- }
- } else {
- fport->silence_duration = 0;
- /* Encode ! */
- MP3Err = BladeDLL.beEncodeChunk(fport->mp3_stream,
- fport->mp3_samples_per_frame,
- frame->buf,
- fport->mp3_buf,
- &WriteSize);
- if (MP3Err != BE_ERR_SUCCESSFUL)
- return PJMEDIA_ERROR;
- /* Write the chunk */
- bytes = WriteSize;
- status = pj_file_write(fport->fd, fport->mp3_buf, &bytes);
- if (status != PJ_SUCCESS)
- return status;
- /* Increment total written. */
- fport->total += bytes;
- }
- /* Check if we need to call callback */
- if (fport->total >= fport->cb_size) {
- if (fport->cb2) {
- if (!fport->subscribed) {
- pj_status_t status;
- status = pjmedia_event_subscribe(NULL, &file_on_event,
- fport, fport);
- fport->subscribed = (status == PJ_SUCCESS)? PJ_TRUE:
- PJ_FALSE;
- }
- if (fport->subscribed && !fport->cb_called) {
- pjmedia_event event;
- /* To prevent the callback from being called more than once. */
- fport->cb_called = PJ_TRUE;
- pjmedia_event_init(&event, PJMEDIA_EVENT_CALLBACK,
- NULL, fport);
- pjmedia_event_publish(NULL, fport, &event,
- PJMEDIA_EVENT_PUBLISH_POST_EVENT);
- }
- } else if (fport->cb) {
- pj_status_t (*cb)(pjmedia_port*, void*);
- pj_status_t status;
- cb = fport->cb;
- fport->cb = NULL;
- status = (*cb)(this_port, this_port->port_data.pdata);
- return status;
- }
- }
- return PJ_SUCCESS;
- }
- /*
- * Get frame, basicy is a no-op operation.
- */
- static pj_status_t file_get_frame(pjmedia_port *this_port,
- pjmedia_frame *frame)
- {
- PJ_UNUSED_ARG(this_port);
- PJ_UNUSED_ARG(frame);
- return PJ_EINVALIDOP;
- }
- /*
- * Close the port, modify file header with updated file length.
- */
- static pj_status_t file_on_destroy(pjmedia_port *this_port)
- {
- struct mp3_file_port *fport = (struct mp3_file_port*)this_port;
- pj_status_t status;
- unsigned long WriteSize;
- unsigned long MP3Err;
- if (fport->subscribed) {
- pjmedia_event_unsubscribe(NULL, &file_on_event, fport, fport);
- fport->subscribed = PJ_FALSE;
- }
- /* Close encoder */
- MP3Err = BladeDLL.beDeinitStream(fport->mp3_stream, fport->mp3_buf,
- &WriteSize);
- if (MP3Err == BE_ERR_SUCCESSFUL) {
- pj_ssize_t bytes = WriteSize;
- status = pj_file_write(fport->fd, fport->mp3_buf, &bytes);
- }
- /* Close file */
- status = pj_file_close(fport->fd);
- /* Write additional VBR header */
- if (fport->mp3_option.vbr) {
- MP3Err = BladeDLL.beWriteVBRHeader(fport->mp3_filename.ptr);
- }
- /* Decrement DLL reference counter */
- deinit_blade_dll();
- /* Done. */
- return PJ_SUCCESS;
- }
|