aboutsummaryrefslogtreecommitdiff
path: root/test/audio/helper.h
blob: e0d034aaeae868947aa42dc6d15dfe197082d22f (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
#ifndef TEST_SOUND_HELPER_H
#define TEST_SOUND_HELPER_H

#include "audio/decoders/raw.h"

#include "common/stream.h"
#include "common/endian.h"

#include <math.h>
#include <limits>

template<typename T>
static T *createSine(const int sampleRate, const int time) {
	T *sine = (T *)malloc(sizeof(T) * time * sampleRate);

	const bool isUnsigned = !std::numeric_limits<T>::is_signed;
	const T xorMask = isUnsigned ? (1 << (std::numeric_limits<T>::digits - 1)) : 0;
	const T maxValue = std::numeric_limits<T>::max() ^ xorMask;

	for (int i = 0; i < time * sampleRate; ++i)
		sine[i] = ((T)(sin((double)i / sampleRate * 2 * M_PI) * maxValue)) ^ xorMask;

	return sine;
}

template<typename T>
static Common::SeekableReadStream *createPartitionStream(T *sine, const int samples, Audio::RawStreamBlockList &blockList) {
	const int block1Len = samples / 2;
	const int block1Size = block1Len * sizeof(T);
	const int block2Len = samples - block1Len;
	const int block2Size = block2Len * sizeof(T);

	const int bufferLen = samples * 2;
	const int bufferSize = bufferLen * sizeof(T);
	T *partition = (T *)calloc(1, bufferSize);

	Audio::RawStreamBlock block;

	// The will layout the buffer like the following:
	// [Zero], [Part2], [Zero], [Part1]

	// The first part of the stream is at the end of the memory buffer
	block.pos = bufferSize - block1Size;
	block.len = block1Len;
	memcpy(partition + bufferLen - block1Len, sine, block1Size);
	blockList.push_back(block);

	// The second part of the stream is near the beginning of the memory buffer
	block.pos = block2Size;
	block.len = block2Len;
	memcpy(partition + block2Len, sine + block1Len, block2Size);
	blockList.push_back(block);

	free(sine);

	return new Common::MemoryReadStream((const byte *)partition, bufferSize, DisposeAfterUse::YES);
}

template<typename T>
static Audio::SeekableAudioStream *createSineStream(const int sampleRate, const int time, int16 **comp, bool le, bool isStereo, bool makePartition = false) {
	T *sine = createSine<T>(sampleRate, time * (isStereo ? 2 : 1));

	const bool isUnsigned = !std::numeric_limits<T>::is_signed;
	const T xorMask = isUnsigned ? (1 << (std::numeric_limits<T>::digits - 1)) : 0;
	const bool is16Bits = (sizeof(T) == 2);
	assert(sizeof(T) == 2 || sizeof(T) == 1);

	const int samples = sampleRate * time * (isStereo ? 2 : 1);

	if (comp) {
		*comp = new int16[samples];
		for (int i = 0; i < samples; ++i) {
			if (is16Bits)
				(*comp)[i] = sine[i] ^ xorMask;
			else
				(*comp)[i] = (sine[i] ^ xorMask) << 8;
		}
	}

	if (is16Bits) {
		if (le) {
			for (int i = 0; i < samples; ++i)
				WRITE_LE_UINT16(&sine[i], sine[i]);
		} else {
			for (int i = 0; i < samples; ++i)
				WRITE_BE_UINT16(&sine[i], sine[i]);
		}
	}

	Audio::SeekableAudioStream *s = 0;
	if (makePartition) {
		Audio::RawStreamBlockList blockList;
		Common::SeekableReadStream *sD = createPartitionStream<T>(sine, samples, blockList);
		s = Audio::makeRawStream(sD, blockList, sampleRate,
		                             (is16Bits ? Audio::FLAG_16BITS : 0)
		                             | (isUnsigned ? Audio::FLAG_UNSIGNED : 0)
		                             | (le ? Audio::FLAG_LITTLE_ENDIAN : 0)
		                             | (isStereo ? Audio::FLAG_STEREO : 0));
	} else {
		Common::SeekableReadStream *sD = new Common::MemoryReadStream((const byte *)sine, sizeof(T) * samples, DisposeAfterUse::YES);
		s = Audio::makeRawStream(sD, sampleRate,
		                             (is16Bits ? Audio::FLAG_16BITS : 0)
		                             | (isUnsigned ? Audio::FLAG_UNSIGNED : 0)
		                             | (le ? Audio::FLAG_LITTLE_ENDIAN : 0)
		                             | (isStereo ? Audio::FLAG_STEREO : 0));
	}

	return s;
}

#endif