/* mkvmerge -- utility for splicing together matroska files from component media subtypes p_pcm.cpp Written by Moritz Bunkus Distributed under the GPL see the file COPYING for details or visit http://www.gnu.org/copyleft/gpl.html */ /*! \file \version \$Id: p_pcm.cpp,v 1.3 2003/02/28 13:00:26 mosu Exp $ \brief PCM output module \author Moritz Bunkus */ #include #include #include #include #include "common.h" #include "queue.h" #include "p_pcm.h" #include "KaxTracks.h" #include "KaxTrackAudio.h" #ifdef DMALLOC #include #endif pcm_packetizer_c::pcm_packetizer_c(void *nprivate_data, int nprivate_size, unsigned long nsamples_per_sec, int nchannels, int nbits_per_sample, audio_sync_t *nasync, range_t *nrange) throw (error_c) : q_c() { packetno = 0; bps = nchannels * nbits_per_sample * nsamples_per_sec / 8; tempbuf = (char *)malloc(bps + 128); if (tempbuf == NULL) die("malloc"); samples_per_sec = nsamples_per_sec; channels = nchannels; bits_per_sample = nbits_per_sample; bytes_output = 0; memcpy(&async, nasync, sizeof(audio_sync_t)); memcpy(&range, nrange, sizeof(range_t)); remaining_sync = 0; set_private_data(nprivate_data, nprivate_size); set_header(); } pcm_packetizer_c::~pcm_packetizer_c() { if (tempbuf != NULL) free(tempbuf); } // void pcm_packetizer_c::reset() { // } #define APCM "A_PCM16IN" void pcm_packetizer_c::set_header() { using namespace LIBMATROSKA_NAMESPACE; if (kax_last_entry == NULL) track_entry = &GetChild(static_cast(*kax_tracks)); else track_entry = &GetNextChild(static_cast(*kax_tracks), static_cast(*kax_last_entry)); kax_last_entry = track_entry; if (serialno <= 0) serialno = track_number++; KaxTrackNumber &tnumber = GetChild(static_cast(*track_entry)); *(static_cast(&tnumber)) = serialno; *(static_cast (&GetChild(static_cast(*track_entry)))) = track_audio; KaxCodecID &codec_id = GetChild(static_cast(*track_entry)); if (bits_per_sample != 16) { fprintf(stderr, "Error: pcm_packetizer: Only files with 16bits per sample " "are supported at the moment.\n"); exit(1); } codec_id.CopyBuffer((binary *)APCM, countof(APCM)); KaxTrackAudio &track_audio = GetChild(static_cast(*track_entry)); KaxAudioSamplingFreq &kax_freq = GetChild(track_audio); *(static_cast(&kax_freq)) = (float)samples_per_sec; KaxAudioChannels &kax_chans = GetChild(track_audio); *(static_cast(&kax_chans)) = channels; } int pcm_packetizer_c::process(char *buf, int size, int last_frame) { int i, bytes_per_packet, remaining_bytes, complete_packets; char *new_buf; if (size > bps) { fprintf(stderr, "FATAL: pcm_packetizer: size (%d) > bps (%d)\n", size, bps); exit(1); } new_buf = buf; if (async.displacement != 0) { if (async.displacement > 0) { // Add silence. int pad_size; pad_size = bps * async.displacement / 1000; new_buf = (char *)malloc(size + pad_size); if (new_buf == NULL) die("malloc"); memset(new_buf, 0, pad_size); memcpy(&new_buf[pad_size], buf, size); size += pad_size; } else // Skip bytes. remaining_sync = -1 * bps * async.displacement / 1000; async.displacement = 0; } if (remaining_sync > 0) { if (remaining_sync > size) { remaining_sync -= size; return EMOREDATA; } memmove(buf, &buf[remaining_sync], size - remaining_sync); size -= remaining_sync; remaining_sync = 0; } bytes_per_packet = bps / pcm_interleave; complete_packets = size / bytes_per_packet; remaining_bytes = size % bytes_per_packet; for (i = 0; i < complete_packets; i++) { add_packet(new_buf + i * bytes_per_packet, bytes_per_packet, (bytes_output * 1000 / bps) * async.linear); bytes_output += bytes_per_packet; packetno++; } if (remaining_bytes != 0) { add_packet(new_buf + complete_packets * bytes_per_packet, remaining_bytes, (bytes_output * 1000 / bps) * async.linear); bytes_output += remaining_bytes; packetno++; } if (new_buf != buf) free(new_buf); if (last_frame) return 0; else return EMOREDATA; }