File: ap_dca.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 (119 lines) | stat: -rw-r--r-- 3,840 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
117
118
119
/*******************************************************************************
*                         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 "dca.h"
}

namespace ap {

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

DCADecoder::DCADecoder(DecoderContext * e) : DecoderPlugin(e) {
  state = dca_init(0);
  }

DCADecoder::~DCADecoder() {
  }

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


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

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

    int dflags=DCA_STEREO;
    level_t level = 1.0f;
    dca_frame (state,buffer.data(),&dflags,&level,0.0f);

    int nblocks = dca_blocks_num (state);
    for (int i=0;i<nblocks;i++) {
      dca_block(state);
      sample_t * samples = dca_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=stream_length;
        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_dca_decoder(DecoderContext * ctx) {
  return new DCADecoder(ctx);
  }

}