changeset 6821:1b9c458d6d60 libavcodec

LPCM 24 bits support, patch by Lars T¸«£uber, lars.taeuber gmx net
author diego
date Sat, 17 May 2008 11:42:03 +0000
parents 6e7d36c1f94a
children 5dc77fe205ac
files Makefile allcodecs.c avcodec.h pcm.c
diffstat 4 files changed, 30 insertions(+), 6 deletions(-) [+]
line wrap: on
line diff
--- a/Makefile	Sat May 17 11:38:30 2008 +0000
+++ b/Makefile	Sat May 17 11:42:03 2008 +0000
@@ -236,6 +236,7 @@
 
 OBJS-$(CONFIG_PCM_ALAW_DECODER)        += pcm.o
 OBJS-$(CONFIG_PCM_ALAW_ENCODER)        += pcm.o
+OBJS-$(CONFIG_PCM_DVD_DECODER)         += pcm.o
 OBJS-$(CONFIG_PCM_MULAW_DECODER)       += pcm.o
 OBJS-$(CONFIG_PCM_MULAW_ENCODER)       += pcm.o
 OBJS-$(CONFIG_PCM_S8_DECODER)          += pcm.o
--- a/allcodecs.c	Sat May 17 11:38:30 2008 +0000
+++ b/allcodecs.c	Sat May 17 11:42:03 2008 +0000
@@ -214,6 +214,7 @@
 
     /* PCM codecs */
     REGISTER_ENCDEC  (PCM_ALAW, pcm_alaw);
+    REGISTER_DECODER (PCM_DVD, pcm_dvd);
     REGISTER_ENCDEC  (PCM_MULAW, pcm_mulaw);
     REGISTER_ENCDEC  (PCM_S8, pcm_s8);
     REGISTER_ENCDEC  (PCM_S16BE, pcm_s16be);
--- a/avcodec.h	Sat May 17 11:38:30 2008 +0000
+++ b/avcodec.h	Sat May 17 11:42:03 2008 +0000
@@ -30,7 +30,7 @@
 #include "libavutil/avutil.h"
 
 #define LIBAVCODEC_VERSION_MAJOR 51
-#define LIBAVCODEC_VERSION_MINOR 56
+#define LIBAVCODEC_VERSION_MINOR 57
 #define LIBAVCODEC_VERSION_MICRO  0
 
 #define LIBAVCODEC_VERSION_INT  AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \
@@ -207,6 +207,7 @@
     CODEC_ID_PCM_S24DAUD,
     CODEC_ID_PCM_ZORK,
     CODEC_ID_PCM_S16LE_PLANAR,
+    CODEC_ID_PCM_DVD,
 
     /* various ADPCM codecs */
     CODEC_ID_ADPCM_IMA_QT= 0x11000,
--- a/pcm.c	Sat May 17 11:38:30 2008 +0000
+++ b/pcm.c	Sat May 17 11:42:03 2008 +0000
@@ -383,15 +383,21 @@
     samples = data;
     src = buf;
 
-    n= av_get_bits_per_sample(avctx->codec_id)/8;
+    if(avctx->channels <= 0 || avctx->channels > MAX_CHANNELS){
+        av_log(avctx, AV_LOG_ERROR, "PCM channels out of bounds\n");
+        return -1;
+    }
+
+    n = avctx->channels * av_get_bits_per_sample(avctx->codec_id)/8;
+    /* av_get_bits_per_sample returns 0 for CODEC_ID_PCM_DVD */
+    if (CODEC_ID_PCM_DVD == avctx->codec_id)
+        /* 2 samples are interleaved per block in PCM_DVD */
+        n = 2 * avctx->channels * avctx->bits_per_sample/8;
+
     if(n && buf_size % n){
         av_log(avctx, AV_LOG_ERROR, "invalid PCM packet\n");
         return -1;
     }
-    if(avctx->channels <= 0 || avctx->channels > MAX_CHANNELS){
-        av_log(avctx, AV_LOG_ERROR, "PCM channels out of bounds\n");
-        return -1;
-    }
 
     buf_size= FFMIN(buf_size, *data_size/2);
     *data_size=0;
@@ -492,6 +498,20 @@
             *samples++ = s->table[*src++];
         }
         break;
+    case CODEC_ID_PCM_DVD:
+        if(avctx->bits_per_sample != 20 && avctx->bits_per_sample != 24) {
+            av_log(avctx, AV_LOG_ERROR, "PCM DVD unsupported sample depth\n");
+            return -1;
+        } else {
+            int jump = avctx->channels * (avctx->bits_per_sample-16) / 4;
+            n = buf_size / (avctx->channels * 2 * avctx->bits_per_sample / 8);
+            while (n--) {
+                for (c=0; c < 2*avctx->channels; c++)
+                    *samples++ = bytestream_get_be16(&src);
+                src += jump;
+            }
+        }
+        break;
     default:
         return -1;
     }
@@ -537,6 +557,7 @@
     PCM_ENCODER(id,name,long_name_) PCM_DECODER(id,name,long_name_)
 
 PCM_CODEC  (CODEC_ID_PCM_ALAW, pcm_alaw, "A-law PCM");
+PCM_CODEC  (CODEC_ID_PCM_DVD, pcm_dvd, "signed 16|20|24-bit big-endian PCM");
 PCM_CODEC  (CODEC_ID_PCM_MULAW, pcm_mulaw, "mu-law PCM");
 PCM_CODEC  (CODEC_ID_PCM_S8, pcm_s8, "signed 8-bit PCM");
 PCM_CODEC  (CODEC_ID_PCM_S16BE, pcm_s16be, "signed 16-bit big-endian PCM");