File: ap_a52.cpp

package info (click to toggle)
gogglesmm 1.2.5-6
  • links: PTS
  • area: main
  • in suites: forky, sid
  • size: 16,812 kB
  • sloc: cpp: 231,960; ansic: 893; xml: 222; makefile: 33
file content (116 lines) | stat: -rw-r--r-- 3,705 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*******************************************************************************
*                         Goggles Audio Player Library                         *
********************************************************************************
*           Copyright (C) 2010-2015 by Sander Jansen. All Rights Reserved      *
*                               ---                                            *
* 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 3 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, see http://www.gnu.org/licenses.           *
********************************************************************************/
#include "ap_defs.h"
#include "ap_event_private.h"
#include "ap_packet.h"
#include "ap_decoder_plugin.h"

extern "C" {
#include <stdint.h>
#include "a52dec/a52.h"
}

namespace ap {

class A52Decoder : public DecoderPlugin {
protected:
  MemoryBuffer  buffer;
  a52_state_t*  state  = nullptr;
  Packet*       out    = nullptr;
  FXlong  stream_position=0;
public:
  A52Decoder(DecoderContext*);
  FXuchar codec() const override { return Codec::A52; }
  FXbool init(ConfigureEvent*) override;
  FXbool process(Packet*) override;
  virtual ~A52Decoder();
  };

A52Decoder::A52Decoder(DecoderContext * e) : DecoderPlugin(e) {
  state = a52_init(0);
  }

A52Decoder::~A52Decoder() {
  }

FXbool A52Decoder::init(ConfigureEvent*event) {
  DecoderPlugin::init(event);
  event->af.setChannels(2);
  af=event->af;
  return true;
  }


FXbool A52Decoder::process(Packet*in) {
  FXbool eos    = (in->flags&FLAG_EOS);
  buffer.append(in->data(),in->size());
  in->unref();

  while (buffer.size()>=7) {
    int flags;
    int samplerate;
    int bitrate;
    int length = a52_syncinfo(buffer.data(),&flags,&samplerate,&bitrate);
    if (length<=0) {
      GM_DEBUG_PRINT("[a52] length returned %d\n",length);
      return false;
      }
    else if (buffer.size()<length) return true;

    int dflags=A52_STEREO;
    sample_t level = 1.0f;
    a52_frame (state,buffer.data(),&dflags,&level,0.0f);

    for (int i=0;i<6;i++) {
      a52_block(state);
      sample_t * samples = a52_samples(state);

      /// Get new buffer
      if (out==nullptr) {
        out = context->get_output_packet();
        if (out==nullptr) return true;
        out->stream_position=stream_position;
        out->stream_length=0;
        out->af=af;
        }

      FXfloat * data = out->flt();
      for (FXint i=0,d=0;i<256;i++) {
        data[d++] = samples[i];
        data[d++] = samples[256+i];
        }
      out->wroteFrames(256);
      if (out->availableFrames()<256) {
        context->post_output_packet(out);
        }
      stream_position+=256;
      }
    buffer.readBytes(length);
    }
  if (eos) {
    context->post_output_packet(out,true);
    }
  return true;
  }

DecoderPlugin * ap_a52_decoder(DecoderContext * ctx) {
  return new A52Decoder(ctx);
  }

}