slesTestFeedback.cpp revision 0ac2a7d4343d98e3cb02180e548a5a4737ba0df1
1f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten/*
2f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * Copyright (C) 2010 The Android Open Source Project
3f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten *
4f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * Licensed under the Apache License, Version 2.0 (the "License");
5f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * you may not use this file except in compliance with the License.
6f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * You may obtain a copy of the License at
7f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten *
8f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten *      http://www.apache.org/licenses/LICENSE-2.0
9f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten *
10f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * Unless required by applicable law or agreed to in writing, software
11f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * distributed under the License is distributed on an "AS IS" BASIS,
12f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * See the License for the specific language governing permissions and
14f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten * limitations under the License.
15f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten */
16f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
17f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten// Test program to record from default audio input and playback to default audio output
18f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
195ef8af762b27b4fa45f59d944b11ad00311cb14bGlenn Kasten#undef NDEBUG
205ef8af762b27b4fa45f59d944b11ad00311cb14bGlenn Kasten
21f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten#include "SLES/OpenSLES.h"
22f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten#include <assert.h>
23f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten#include <stdio.h>
24f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten#include <stdlib.h>
25f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten#include <unistd.h>
26f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
270ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten#define ASSERT_EQ(x, y) do { if ((x) == (y)) ; else { fprintf(stderr, "0x%x != 0x%x\n", \
280ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    (unsigned) (x), (unsigned) (y)); assert((x) == (y)); } } while (0)
29f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
300ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten#if 0
31f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kastentypedef struct {
32f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    short left;
33f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    short right;
34f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten} stereo;
35f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
360ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastentypedef short mono;
370ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten#endif
380ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten
390ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten// default values
400ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 rxBufCount = 1;     // -r#
410ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 txBufCount = 2;     // -t#
420ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 bufSizeInFrames = 512;  // -f#
430ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 channels = 2;       // -c#
440ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 sampleRate = 44100; // -s#
450ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 appBufCount = 0;    // -n#
460ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn KastenSLuint32 bufSizeInBytes = 0;
470ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten
480ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten// Storage area for the buffers
490ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic char *buffers = NULL;
50f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
51f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten// Index of which buffer to enqueue next
52f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kastenstatic SLuint32 whichRecord;
53f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kastenstatic SLuint32 whichPlay;
54f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
55f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn KastenSLBufferQueueItf recorderBufferQueue;
56f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn KastenSLBufferQueueItf playerBufferQueue;
57f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
580ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten// Compute maximum of two values
590ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 max(SLuint32 a, SLuint32 b)
600ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten{
610ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    return a >= b ? a : b;
620ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten}
630ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten
640ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten// Compute minimum of two values
650ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kastenstatic SLuint32 min(SLuint32 a, SLuint32 b)
660ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten{
670ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    return a <= b ? a : b;
680ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten}
690ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten
70f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten// Called after audio recorder fills a buffer with data
71f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kastenstatic void recorderCallback(SLBufferQueueItf caller, void *context)
72f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten{
73f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLresult result;
74f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    putchar('*');
75f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    fflush(stdout);
76f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
77f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // Enqueue the next empty buffer for the recorder to fill
780ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    assert(whichRecord < appBufCount);
790ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    void *buffer = &buffers[bufSizeInBytes * whichRecord];
800ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    result = (*recorderBufferQueue)->Enqueue(recorderBufferQueue, buffer, bufSizeInBytes);
81f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
820ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    if (++whichRecord >= appBufCount)
83f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        whichRecord = 0;
84f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
85f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // Enqueue the just-filled buffer for the player to empty
860ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    assert(whichPlay < appBufCount); // sic not tx
870ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    buffer = &buffers[bufSizeInBytes * whichPlay];
880ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    result = (*playerBufferQueue)->Enqueue(playerBufferQueue, buffer, bufSizeInBytes);
895ef8af762b27b4fa45f59d944b11ad00311cb14bGlenn Kasten    // FIXME not sure yet why this is overflowing
905ef8af762b27b4fa45f59d944b11ad00311cb14bGlenn Kasten    if (SL_RESULT_SUCCESS == result) {
910ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        if (++whichPlay >= appBufCount)
920ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            whichPlay = 0;
930ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    } else {
940ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        ASSERT_EQ(SL_RESULT_BUFFER_INSUFFICIENT, result);
955ef8af762b27b4fa45f59d944b11ad00311cb14bGlenn Kasten    }
96f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
97f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten}
98f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
99f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kastenint main(int argc, char **argv)
100f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten{
1010ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    // process command-line options
1020ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    int i;
1030ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    for (i = 1; i < argc; ++i) {
1040ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        char *arg = argv[i];
1050ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        if (arg[0] != '-')
1060ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            break;
1070ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        // -r# number of receive buffers
1080ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        if (!strncmp(arg, "-r", 2)) {
1090ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            rxBufCount = atoi(&arg[2]);
1100ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            if (rxBufCount < 1 || rxBufCount > 8)
1110ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                fprintf(stderr, "%s: unusual receive buffer queue size (%u buffers)", argv[0],
1120ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                    (unsigned) rxBufCount);
1130ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        // -t# number of receive buffers
1140ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        } else if (!strncmp(arg, "-t", 2)) {
1150ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            txBufCount = atoi(&arg[2]);
1160ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            if (txBufCount < 1 || txBufCount > 8)
1170ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                fprintf(stderr, "%s: unusual transmit buffer queue size (%u buffers)", argv[0],
1180ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                    (unsigned) txBufCount);
1190ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        // -n# number of application buffers
1200ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        } else if (!strncmp(arg, "-n", 2)) {
1210ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            appBufCount = atoi(&arg[2]);
1220ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        // -f# size of each buffer in frames
1230ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        } else if (!strncmp(arg, "-f", 2)) {
1240ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            bufSizeInFrames = atoi(&arg[2]);
1250ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            if (bufSizeInFrames == 0) {
1260ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                fprintf(stderr, "%s: unusual buffer size (%u frames)\n", argv[0],
1270ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                    (unsigned) bufSizeInFrames);
1280ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            }
1290ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        // -c1 mono or -c2 stereo
1300ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        } else if (!strncmp(arg, "-c", 2)) {
1310ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            channels = atoi(&arg[2]);
1320ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            if (channels < 1 || channels > 2) {
1330ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                fprintf(stderr, "%s: unusual channel count ignored (%u)\n", argv[0],
1340ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                    (unsigned) channels);
1350ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                channels = 2;
1360ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            }
1370ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        // -s# sample rate in Hz
1380ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        } else if (!strncmp(arg, "-s", 2)) {
1390ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            sampleRate = atoi(&arg[2]);
1400ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            switch (sampleRate) {
1410ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            case 8000:
1420ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            case 11025:
1430ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            case 16000:
1440ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            case 22050:
1450ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            case 32000:
1460ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            case 44100:
1470ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                break;
1480ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            default:
1490ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                fprintf(stderr, "%s: unusual sample rate (%u Hz)\n", argv[0],
1500ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                    (unsigned) sampleRate);
1510ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten                break;
1520ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            }
1530ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        } else
1540ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            fprintf(stderr, "%s: unknown option %s\n", argv[0], arg);
1550ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    }
1560ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    if (i < argc) {
1570ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "usage: %s -r# -t# -f# -r# -m/-s\n", argv[0]);
1580ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -r# receive buffer queue count for microphone input, default 1\n");
1590ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -t# transmit buffer queue count for speaker output, default 2\n");
1600ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -f# number of frames per buffer, default 512\n");
1610ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -s# sample rate in Hz, default 44100\n");
1620ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -n# number of application-allocated buffers, default max(-r#,-t#)\n");
1630ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -c1 mono\n");
1640ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "  -c2 stereo, default\n");
1650ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    }
1660ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    if (appBufCount == 0)
1670ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        appBufCount = max(rxBufCount, txBufCount);
1680ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    if (appBufCount == 0)
1690ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        appBufCount = 1;
1700ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    if (appBufCount < max(rxBufCount, txBufCount))
1710ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        fprintf(stderr, "%s: unusual application buffer count (%u buffers)", argv[0],
1720ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            (unsigned) appBufCount);
1730ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    bufSizeInBytes = channels * bufSizeInFrames * sizeof(short);
1740ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    buffers = (char *) malloc(bufSizeInBytes * appBufCount);
175f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLresult result;
176f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
177f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // create engine
178f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLObjectItf engineObject;
179f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = slCreateEngine(&engineObject, 0, NULL, 0, NULL, NULL);
180f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
181f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*engineObject)->Realize(engineObject, SL_BOOLEAN_FALSE);
182f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
183f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLEngineItf engineEngine;
184f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*engineObject)->GetInterface(engineObject, SL_IID_ENGINE, &engineEngine);
185f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
186f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
187f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // create output mix
188f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLObjectItf outputmixObject;
189f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*engineEngine)->CreateOutputMix(engineEngine, &outputmixObject, 0, NULL, NULL);
190f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
191f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*outputmixObject)->Realize(outputmixObject, SL_BOOLEAN_FALSE);
192f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
193f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
194f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // create an audio player with buffer queue source and output mix sink
195f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLDataSource audiosrc;
196f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLDataSink audiosnk;
197f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLDataFormat_PCM pcm;
198f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLDataLocator_OutputMix locator_outputmix;
199f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLDataLocator_BufferQueue locator_bufferqueue;
200f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_bufferqueue.locatorType = SL_DATALOCATOR_BUFFERQUEUE;
2010ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    locator_bufferqueue.numBuffers = txBufCount;
202f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_outputmix.locatorType = SL_DATALOCATOR_OUTPUTMIX;
203f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_outputmix.outputMix = outputmixObject;
204f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    pcm.formatType = SL_DATAFORMAT_PCM;
2050ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    pcm.numChannels = channels;
2060ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    pcm.samplesPerSec = sampleRate * 1000;
207f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    pcm.bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_16;
208f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    pcm.containerSize = 16;
2090ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    pcm.channelMask = channels == 1 ? SL_SPEAKER_FRONT_CENTER :
2100ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        (SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT);
211f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    pcm.endianness = SL_BYTEORDER_LITTLEENDIAN;
212f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosrc.pLocator = &locator_bufferqueue;
213f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosrc.pFormat = &pcm;
214f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosnk.pLocator = &locator_outputmix;
215f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosnk.pFormat = NULL;
216f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLObjectItf playerObject;
217f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLInterfaceID ids[1] = {SL_IID_BUFFERQUEUE};
218f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLboolean flags[1] = {SL_BOOLEAN_TRUE};
2190ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audiosrc, &audiosnk,
2200ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        1, ids, flags);
221f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
222f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*playerObject)->Realize(playerObject, SL_BOOLEAN_FALSE);
223f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
224f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLPlayItf playerPlay;
225f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*playerObject)->GetInterface(playerObject, SL_IID_PLAY, &playerPlay);
226f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
227f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*playerObject)->GetInterface(playerObject, SL_IID_BUFFERQUEUE, &playerBufferQueue);
228f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
229f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*playerPlay)->SetPlayState(playerPlay, SL_PLAYSTATE_PLAYING);
230f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
231f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
232f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // Create an audio recorder with microphone device source and buffer queue sink.
233f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // The buffer queue as sink is an Android-specific extension.
234f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
235f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLDataLocator_IODevice locator_iodevice;
236f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_iodevice.locatorType = SL_DATALOCATOR_IODEVICE;
237f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_iodevice.deviceType = SL_IODEVICE_AUDIOINPUT;
238f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_iodevice.deviceID = SL_DEFAULTDEVICEID_AUDIOINPUT;
239f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    locator_iodevice.device = NULL;
240f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosrc.pLocator = &locator_iodevice;
241f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosrc.pFormat = &pcm;
2420ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    locator_bufferqueue.numBuffers = rxBufCount;
243f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosnk.pLocator = &locator_bufferqueue;
244f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    audiosnk.pFormat = &pcm;
245f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLObjectItf recorderObject;
2460ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    result = (*engineEngine)->CreateAudioRecorder(engineEngine, &recorderObject, &audiosrc,
2470ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        &audiosnk, 1, ids, flags);
248f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
249f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*recorderObject)->Realize(recorderObject, SL_BOOLEAN_FALSE);
250f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
251f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    SLRecordItf recorderRecord;
252f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*recorderObject)->GetInterface(recorderObject, SL_IID_RECORD, &recorderRecord);
253f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
2540ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    result = (*recorderObject)->GetInterface(recorderObject, SL_IID_BUFFERQUEUE,
2550ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        &recorderBufferQueue);
256f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
257f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*recorderBufferQueue)->RegisterCallback(recorderBufferQueue, recorderCallback, NULL);
258f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
259f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
260f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // Enqueue some empty buffers for the recorder
2610ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    SLuint32 temp = min(rxBufCount, appBufCount);
2620ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    for (whichRecord = 0; whichRecord < (temp <= 1 ? 1 : temp - 1); ++whichRecord) {
2630ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        result = (*recorderBufferQueue)->Enqueue(recorderBufferQueue,
2640ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            &buffers[bufSizeInBytes * whichRecord], bufSizeInBytes);
265f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        ASSERT_EQ(SL_RESULT_SUCCESS, result);
266f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    }
2670ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten    if (whichRecord >= appBufCount)
2680ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        whichRecord = 0;
269f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
270f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // Kick off the recorder
271f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    whichPlay = 0;
272f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    result = (*recorderRecord)->SetRecordState(recorderRecord, SL_RECORDSTATE_RECORDING);
273f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    ASSERT_EQ(SL_RESULT_SUCCESS, result);
274f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
275f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    // Wait patiently
276f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    for (;;) {
277f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        usleep(1000000);
278f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        putchar('.');
279f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        SLBufferQueueState playerBQState;
280f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        result = (*playerBufferQueue)->GetState(playerBufferQueue, &playerBQState);
281f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        ASSERT_EQ(SL_RESULT_SUCCESS, result);
282f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        SLBufferQueueState recorderBQState;
283f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        result = (*recorderBufferQueue)->GetState(recorderBufferQueue, &recorderBQState);
284f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        ASSERT_EQ(SL_RESULT_SUCCESS, result);
2850ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten        printf("pC%u pI%u rC%u rI%u\n", (unsigned) playerBQState.count,
2860ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            (unsigned) playerBQState.playIndex, (unsigned) recorderBQState.count,
2870ac2a7d4343d98e3cb02180e548a5a4737ba0df1Glenn Kasten            (unsigned) recorderBQState.playIndex);
288f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten        fflush(stdout);
289f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    }
290f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten
291f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten    //return EXIT_SUCCESS;
292f460ec604707d0bdaf8124d84c5f8595cba9c804Glenn Kasten}
293