~ubuntu-branches/ubuntu/trusty/aegisub/trusty

« back to all changes in this revision

Viewing changes to src/audio_provider_downmix.cpp

  • Committer: Package Import Robot
  • Author(s): Sebastian Reichel
  • Date: 2012-03-16 22:58:00 UTC
  • Revision ID: package-import@ubuntu.com-20120316225800-yfb8h9e5n04rk46a
Tags: upstream-2.1.9
ImportĀ upstreamĀ versionĀ 2.1.9

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
// Copyright (c) 2007-2008, Niels Martin Hansen
 
2
// All rights reserved.
 
3
//
 
4
// Redistribution and use in source and binary forms, with or without
 
5
// modification, are permitted provided that the following conditions are met:
 
6
//
 
7
//   * Redistributions of source code must retain the above copyright notice,
 
8
//     this list of conditions and the following disclaimer.
 
9
//   * Redistributions in binary form must reproduce the above copyright notice,
 
10
//     this list of conditions and the following disclaimer in the documentation
 
11
//     and/or other materials provided with the distribution.
 
12
//   * Neither the name of the Aegisub Group nor the names of its contributors
 
13
//     may be used to endorse or promote products derived from this software
 
14
//     without specific prior written permission.
 
15
//
 
16
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 
17
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 
18
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 
19
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 
20
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 
21
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 
22
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 
23
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 
24
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 
25
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 
26
// POSSIBILITY OF SUCH DAMAGE.
 
27
//
 
28
// -----------------------------------------------------------------------------
 
29
//
 
30
// AEGISUB
 
31
//
 
32
// Website: http://aegisub.cellosoft.com
 
33
// Contact: mailto:jiifurusu@gmail.com
 
34
 
 
35
 
 
36
//////////////////
 
37
// Headers
 
38
#include "config.h"
 
39
 
 
40
#include "audio_provider_downmix.h"
 
41
 
 
42
 
 
43
//////////////////
 
44
// Constructor
 
45
DownmixingAudioProvider::DownmixingAudioProvider(AudioProvider *source) {
 
46
        filename = source->GetFilename();
 
47
        channels = 1; // target
 
48
        src_channels = source->GetChannels();
 
49
        num_samples = source->GetNumSamples();
 
50
        bytes_per_sample = source->GetBytesPerSample();
 
51
        sample_rate = source->GetSampleRate();
 
52
 
 
53
        // We now own this
 
54
        provider = source;
 
55
 
 
56
        if (!(bytes_per_sample == 1 || bytes_per_sample == 2))
 
57
                throw _T("Downmixing Audio Provider: Can only downmix 8 and 16 bit audio");
 
58
        if (!source->AreSamplesNativeEndian())
 
59
                throw _T("Downmixing Audio Provider: Source must have machine endian samples");
 
60
}
 
61
 
 
62
/////////////////
 
63
// Destructor
 
64
DownmixingAudioProvider::~DownmixingAudioProvider()     {
 
65
                delete provider;
 
66
}
 
67
 
 
68
////////////////
 
69
// Actual work happens here
 
70
void DownmixingAudioProvider::GetAudio(void *buf, int64_t start, int64_t count) {
 
71
        if (count == 0) return;
 
72
 
 
73
        // We can do this ourselves
 
74
        if (start >= num_samples) {
 
75
                if (bytes_per_sample == 1)
 
76
                        // 8 bit formats are usually unsigned with bias 127
 
77
                        memset(buf, 127, count);
 
78
                else
 
79
                        // While everything else is signed
 
80
                        memset(buf, 0, count*bytes_per_sample);
 
81
 
 
82
                return;
 
83
        }
 
84
 
 
85
        // So alloc some temporary memory for this
 
86
        // Depending on use, this might be made faster by using
 
87
        // a pre-allocced block of memory...?
 
88
        char *tmp = new char[count*bytes_per_sample*src_channels];
 
89
 
 
90
        provider->GetAudio(tmp, start, count);
 
91
 
 
92
        // Now downmix
 
93
        // Just average the samples over the channels (really bad if they're out of phase!)
 
94
        // XXX: Assuming here that sample data are in machine endian, an upstream provider should ensure that
 
95
        if (bytes_per_sample == 1) {
 
96
                uint8_t *src = (uint8_t *)tmp;
 
97
                uint8_t *dst = (uint8_t *)buf;
 
98
 
 
99
                while (count > 0) {
 
100
                        int sum = 0;
 
101
                        for (int c = 0; c < src_channels; c++)
 
102
                                sum += *(src++);
 
103
                        *(dst++) = (uint8_t)(sum / src_channels);
 
104
                        count--;
 
105
                }
 
106
        }
 
107
        else if (bytes_per_sample == 2) {
 
108
                int16_t *src = (int16_t *)tmp;
 
109
                int16_t *dst = (int16_t *)buf;
 
110
 
 
111
                while (count > 0) {
 
112
                        int sum = 0;
 
113
                        for (int c = 0; c < src_channels; c++)
 
114
                                sum += *(src++);
 
115
                        *(dst++) = (int16_t)(sum / src_channels);
 
116
                        count--;
 
117
                }
 
118
        }
 
119
 
 
120
        // Done downmixing, free the work buffer
 
121
        delete[] tmp;
 
122
}