summaryrefslogtreecommitdiff
path: root/src/opusencoder.cc
blob: 77ea0d108e747b4abdeb260e740d33d160d9d2f8 (plain)
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/***************************************************************************
 *            opusencoder.cc
 *
 *  Fri Sep 19 19:21:50 CEST 2014
 *  Copyright 2014 Bent Bisballe Nyeng
 *  deva@aasimon.org
 ****************************************************************************/

/*
 *  This file is part of SimpleRTP.
 *
 *  SimpleRTP 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 of the License, or
 *  (at your option) any later version.
 *
 *  SimpleRTP 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 SimpleRTP; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA.
 */
#include "opusencoder.h"

#include <string.h>
#include <opus/opus.h>
#include <stdio.h>

#include "samplecache.h"

#define OPUS_BITRATE 64000

struct opus_encoder_private_t {
  OpusEncoder *ce;
  SampleCache *cache;
};

OpusEncoder::OpusEncoder()
{
  prv = new struct opus_encoder_private_t;

  prv->cache = new SampleCache(5760 * 10);

  int error = 0;

  samplerate_t srate = opus_samplerate();

  prv->ce =
    opus_encoder_create(srate, 1/*channels()*/, OPUS_APPLICATION_AUDIO, &error);
  if(!prv->ce || error != OPUS_OK) {
    printf("srate: %d, ch: %d, Error: %d\n", srate, 1/*channels()*/, error);
    //throw InitException();
  }

  // 500 to 512000
  opus_encoder_ctl(prv->ce, OPUS_SET_BITRATE(OPUS_BITRATE/*bitrate()*/));

  // [0-10] with 10 representing the highest complexity.
  opus_encoder_ctl(prv->ce, OPUS_SET_COMPLEXITY(10));

  /*
  printf("Encoder(fs: %d, ch: %d, bitrate: %d)\n", srate, channels(),
         bitrate());
  */
}

OpusEncoder::~OpusEncoder()
{
  if(prv) {
    if(prv->ce) opus_encoder_destroy(prv->ce);
    if(prv->cache) delete prv->cache;
    delete prv;
  }
}

unsigned int OpusEncoder::framesize()
{
  return opus_samplerate() / 1000 * OPUS_FRAME_SIZE; // 10ms audio data
}

samplerate_t OpusEncoder::opus_samplerate()
{
  return 48000;
}

framelist_t OpusEncoder::encode(const char *pcm, size_t size)
{
  framelist_t list;

  //  printf("PCM size: %d\n", size);

  samplerate_t srate = opus_samplerate();

  if(prv->ce == NULL) {
    //throw InitException();
    printf("prv->ce == NULL\n");
  }

  size_t bytes_per_packet = ((OPUS_BITRATE/*bitrate()*/ *framesize())/srate+4)/8;

  prv->cache->addSamples((short*)pcm, size / sizeof(short));

  //size_t p = 0;
  opus_int16 *samples = new opus_int16[framesize()];
  while(prv->cache->cacheSize() >= framesize()) {
    size_t sz = prv->cache->getSamples(samples, framesize());

    Frame *frame = new Frame(bytes_per_packet);

    int n = opus_encode(prv->ce, samples, sz / 1/*channels()*//*sizeof(short)*/,
                        (unsigned char *)frame->data, (opus_int32)frame->size);

    //printf("Compressed to %d\n", n);

    if(n < 0) {
      printf("n < 0\n");
      //throw EncodingException();
    }

    frame->size = n;

    list.push_back(frame);
  }
  delete[] samples;

  //printf("%d frames in list\n", list.size());

  return list;
}