1 /*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 // Play silence and recover from dead servers or disconnected devices.
18
19 #include <memory>
20 #include <stdio.h>
21
22 #include <aaudio/AAudio.h>
23
24
25 #define DEFAULT_TIMEOUT_NANOS ((int64_t)1000000000)
26
main(int argc,char ** argv)27 int main(int argc, char **argv) {
28 (void) argc;
29 (void) argv;
30
31 aaudio_result_t result = AAUDIO_OK;
32
33 int32_t triesLeft = 3;
34 int32_t bufferCapacity;
35 int32_t framesPerBurst = 0;
36
37 int32_t actualChannelCount = 0;
38 int32_t actualSampleRate = 0;
39 aaudio_format_t actualDataFormat = AAUDIO_FORMAT_PCM_FLOAT;
40
41 AAudioStreamBuilder *aaudioBuilder = nullptr;
42 AAudioStream *aaudioStream = nullptr;
43
44 // Make printf print immediately so that debug info is not stuck
45 // in a buffer if we hang or crash.
46 setvbuf(stdout, nullptr, _IONBF, (size_t) 0);
47
48 printf("TestRecovery:\n");
49
50 // Use an AAudioStreamBuilder to contain requested parameters.
51 result = AAudio_createStreamBuilder(&aaudioBuilder);
52 if (result != AAUDIO_OK) {
53 printf("AAudio_createStreamBuilder returned %s",
54 AAudio_convertResultToText(result));
55 goto finish;
56 }
57
58 // Request stream properties.
59 AAudioStreamBuilder_setFormat(aaudioBuilder, AAUDIO_FORMAT_PCM_FLOAT);
60
61 while (triesLeft-- > 0) {
62 // Create an AAudioStream using the Builder.
63 result = AAudioStreamBuilder_openStream(aaudioBuilder, &aaudioStream);
64 if (result != AAUDIO_OK) {
65 printf("AAudioStreamBuilder_openStream returned %s",
66 AAudio_convertResultToText(result));
67 goto finish;
68 }
69
70 // Check to see what kind of stream we actually got.
71 actualSampleRate = AAudioStream_getSampleRate(aaudioStream);
72 actualChannelCount = AAudioStream_getChannelCount(aaudioStream);
73 actualDataFormat = AAudioStream_getFormat(aaudioStream);
74
75 printf("-------- chans = %3d, rate = %6d format = %d\n",
76 actualChannelCount, actualSampleRate, actualDataFormat);
77
78 // This is the number of frames that are read in one chunk by a DMA controller
79 // or a DSP or a mixer.
80 framesPerBurst = AAudioStream_getFramesPerBurst(aaudioStream);
81 bufferCapacity = AAudioStream_getBufferCapacityInFrames(aaudioStream);
82 printf(" bufferCapacity = %d, framesPerBurst = %d\n",
83 bufferCapacity, framesPerBurst);
84
85 int samplesPerBurst = framesPerBurst * actualChannelCount;
86 std::unique_ptr<float[]> buffer(new float[samplesPerBurst]);
87
88 result = AAudioStream_requestStart(aaudioStream);
89 if (result != AAUDIO_OK) {
90 printf("AAudioStream_requestStart returned %s",
91 AAudio_convertResultToText(result));
92 goto finish;
93 }
94
95 // Play silence for awhile.
96 int32_t framesMax = actualSampleRate * 20;
97 int64_t framesTotal = 0;
98 int64_t printAt = actualSampleRate;
99 while (result == AAUDIO_OK && framesTotal < framesMax) {
100 int32_t framesWritten = AAudioStream_write(aaudioStream,
101 buffer.get(), framesPerBurst,
102 DEFAULT_TIMEOUT_NANOS);
103 if (framesWritten < 0) {
104 result = framesWritten;
105 printf("write() returned %s, frames = %d\n",
106 AAudio_convertResultToText(result), (int)framesTotal);
107 printf(" frames = %d\n", (int)framesTotal);
108 } else if (framesWritten != framesPerBurst) {
109 printf("write() returned %d, frames = %d\n", framesWritten, (int)framesTotal);
110 result = AAUDIO_ERROR_TIMEOUT;
111 } else {
112 framesTotal += framesWritten;
113 if (framesTotal >= printAt) {
114 printf("frames = %d\n", (int)framesTotal);
115 printAt += actualSampleRate;
116 }
117 }
118 }
119 result = AAudioStream_requestStop(aaudioStream);
120 if (result != AAUDIO_OK) {
121 printf("AAudioStream_requestStop returned %s\n",
122 AAudio_convertResultToText(result));
123 }
124 result = AAudioStream_close(aaudioStream);
125 if (result != AAUDIO_OK) {
126 printf("AAudioStream_close returned %s\n",
127 AAudio_convertResultToText(result));
128 }
129 aaudioStream = nullptr;
130 }
131
132 finish:
133 if (aaudioStream != nullptr) {
134 AAudioStream_close(aaudioStream);
135 }
136 AAudioStreamBuilder_delete(aaudioBuilder);
137 printf(" result = %d = %s\n", result, AAudio_convertResultToText(result));
138 }
139