You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

216 lines
5.0 KiB

/*
* import_ac3.c
*
* Copyright (C) Thomas Oestreich - June 2001
*
* This file is part of transcode, a video stream processing tool
*
* transcode 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, or (at your option)
* any later version.
*
* transcode 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 GNU Make; see the file COPYING. If not, write to
* the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
#define MOD_NAME "import_ac3.so"
#define MOD_VERSION "v0.3.2 (2002-02-15)"
#define MOD_CODEC "(audio) AC3"
#include "transcode.h"
static int verbose_flag = TC_QUIET;
static int capability_flag = TC_CAP_PCM | TC_CAP_AC3;
#define MOD_PRE ac3
#include "import_def.h"
#include "ac3scan.h"
char import_cmd_buf[TC_BUF_MAX];
static FILE *fd;
static int codec, syncf=0;
static int pseudo_frame_size=0, real_frame_size=0, effective_frame_size=0;
static int ac3_bytes_to_go=0;
/* ------------------------------------------------------------
*
* open stream
*
* ------------------------------------------------------------*/
MOD_open
{
long sret;
// audio only
if(param->flag != TC_AUDIO) return(TC_IMPORT_ERROR);
codec = vob->im_a_codec;
syncf = vob->sync;
switch(codec) {
case CODEC_AC3:
// produce a clean sequence of AC3 frames
sret = tc_snprintf(import_cmd_buf, TC_BUF_MAX,
"tcextract -a %d -i \"%s\" -x ac3 -d %d |"
" tcextract -t raw -x ac3 -d %d",
vob->a_track, vob->audio_in_file, vob->verbose, vob->verbose);
if (sret < 0)
return(TC_IMPORT_ERROR);
if(verbose_flag) tc_log_info(MOD_NAME, "AC3->AC3");
break;
case CODEC_PCM:
if(vob->a_codec_flag==CODEC_AC3) {
sret = tc_snprintf(import_cmd_buf, TC_BUF_MAX,
"tcextract -a %d -i \"%s\" -x ac3 -d %d |"
" tcdecode -x ac3 -d %d -s %f,%f,%f -A %d",
vob->a_track, vob->audio_in_file, vob->verbose,
vob->verbose, vob->ac3_gain[0], vob->ac3_gain[1],
vob->ac3_gain[2], vob->a52_mode);
if (sret < 0)
return(TC_IMPORT_ERROR);
if(verbose_flag) tc_log_info(MOD_NAME, "AC3->PCM");
}
break;
default:
tc_log_warn(MOD_NAME, "invalid import codec request 0x%x", codec);
return(TC_IMPORT_ERROR);
}
// print out
if(verbose_flag)
tc_log_info(MOD_NAME, "%s", import_cmd_buf);
// set to NULL if we handle read
param->fd = NULL;
// popen
if((fd = popen(import_cmd_buf, "r"))== NULL) {
tc_log_perror(MOD_NAME, "popen pcm stream");
return(TC_IMPORT_ERROR);
}
return(TC_IMPORT_OK);
}
/* ------------------------------------------------------------
*
* decode stream
*
* ------------------------------------------------------------*/
MOD_decode
{
int ac_bytes=0, ac_off=0;
int num_frames;
// audio only
if(param->flag != TC_AUDIO) return(TC_IMPORT_ERROR);
switch(codec) {
case CODEC_AC3:
// determine frame size at the very beginning of the stream
if (pseudo_frame_size == 0) {
if (ac3scan(fd, param->buffer, param->size, &ac_off, &ac_bytes,
&pseudo_frame_size, &real_frame_size, verbose) != 0)
return(TC_IMPORT_ERROR);
} else {
ac_off = 0;
ac_bytes = pseudo_frame_size;
}
// switch to entire frames:
// bytes_to_go is the difference between requested bytes and
// delivered bytes
//
// pseudo_frame_size = average bytes per audio frame
// real_frame_size = real AC3 frame size in bytes
num_frames = (ac_bytes + ac3_bytes_to_go) / real_frame_size;
effective_frame_size = num_frames * real_frame_size;
ac3_bytes_to_go = ac_bytes + ac3_bytes_to_go - effective_frame_size;
// return effective_frame_size as physical size of audio data
param->size = effective_frame_size;
if (verbose_flag & TC_STATS)
tc_log_info(MOD_NAME, "pseudo=%d, real=%d, frames=%d, effective=%d",
ac_bytes, real_frame_size, num_frames,
effective_frame_size);
// adjust
ac_bytes=effective_frame_size;
if(syncf>0) {
//dump an ac3 frame, instead of a pcm frame
ac_bytes = real_frame_size-ac_off;
param->size = real_frame_size;
--syncf;
}
break;
case CODEC_PCM:
//default:
ac_off = 0;
ac_bytes = param->size;
break;
default:
tc_log_warn(MOD_NAME, "invalid import codec request 0x%x", codec);
return(TC_IMPORT_ERROR);
}
if (fread(param->buffer+ac_off, ac_bytes-ac_off, 1, fd) !=1)
return(TC_IMPORT_ERROR);
return(TC_IMPORT_OK);
}
/* ------------------------------------------------------------
*
* close stream
*
* ------------------------------------------------------------*/
MOD_close
{
if(param->fd != NULL) pclose(param->fd);
return(TC_IMPORT_OK);
}