1 /*
2  * Copyright (C) 2015 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 //#define LOG_NDEBUG 0
18 #define LOG_TAG "SL-Utils"
19 
20 #include "sl-utils.h"
21 #include <utils/Mutex.h>
22 
23 #define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
24 
25 // These will wind up in <SLES/OpenSLES_Android.h>
26 #define SL_ANDROID_SPEAKER_QUAD (SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT \
27  | SL_SPEAKER_BACK_LEFT | SL_SPEAKER_BACK_RIGHT)
28 
29 #define SL_ANDROID_SPEAKER_5DOT1 (SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT \
30  | SL_SPEAKER_FRONT_CENTER  | SL_SPEAKER_LOW_FREQUENCY| SL_SPEAKER_BACK_LEFT \
31  | SL_SPEAKER_BACK_RIGHT)
32 
33 #define SL_ANDROID_SPEAKER_7DOT1 (SL_ANDROID_SPEAKER_5DOT1 | SL_SPEAKER_SIDE_LEFT \
34  |SL_SPEAKER_SIDE_RIGHT)
35 
36 namespace android {
37 
38 static Mutex gLock;
39 static SLObjectItf gEngineObject;
40 static unsigned gRefCount;
41 
42 static const char *gErrorStrings[] = {
43     "SL_RESULT_SUCCESS",                // 0
44     "SL_RESULT_PRECONDITIONS_VIOLATE",  // 1
45     "SL_RESULT_PARAMETER_INVALID",      // 2
46     "SL_RESULT_MEMORY_FAILURE",         // 3
47     "SL_RESULT_RESOURCE_ERROR",         // 4
48     "SL_RESULT_RESOURCE_LOST",          // 5
49     "SL_RESULT_IO_ERROR",               // 6
50     "SL_RESULT_BUFFER_INSUFFICIENT",    // 7
51     "SL_RESULT_CONTENT_CORRUPTED",      // 8
52     "SL_RESULT_CONTENT_UNSUPPORTED",    // 9
53     "SL_RESULT_CONTENT_NOT_FOUND",      // 10
54     "SL_RESULT_PERMISSION_DENIED",      // 11
55     "SL_RESULT_FEATURE_UNSUPPORTED",    // 12
56     "SL_RESULT_INTERNAL_ERROR",         // 13
57     "SL_RESULT_UNKNOWN_ERROR",          // 14
58     "SL_RESULT_OPERATION_ABORTED",      // 15
59     "SL_RESULT_CONTROL_LOST",           // 16
60 };
61 
getSLErrStr(int code)62 const char *getSLErrStr(int code) {
63     if ((size_t)code >= ARRAY_SIZE(gErrorStrings)) {
64         return "SL_RESULT_UNKNOWN";
65     }
66     return gErrorStrings[code];
67 }
68 
channelCountToMask(unsigned channelCount)69 SLuint32 channelCountToMask(unsigned channelCount) {
70     switch (channelCount) {
71     case 1:
72         return SL_SPEAKER_FRONT_LEFT; // we prefer left over center
73     case 2:
74         return SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT;
75     case 3:
76         return SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT | SL_SPEAKER_FRONT_CENTER;
77     case 4:
78         return SL_ANDROID_SPEAKER_QUAD;
79     case 5:
80         return SL_ANDROID_SPEAKER_QUAD | SL_SPEAKER_FRONT_CENTER;
81     case 6:
82         return SL_ANDROID_SPEAKER_5DOT1;
83     case 7:
84         return SL_ANDROID_SPEAKER_5DOT1 | SL_SPEAKER_BACK_CENTER;
85     case 8:
86         return SL_ANDROID_SPEAKER_7DOT1;
87     default:
88         return 0;
89     }
90 }
91 
createEngine()92 static SLObjectItf createEngine() {
93     static SLEngineOption EngineOption[] = {
94             (SLuint32) SL_ENGINEOPTION_THREADSAFE,
95             (SLuint32) SL_BOOLEAN_TRUE
96     };
97     // create engine in thread-safe mode
98     SLObjectItf engine;
99     SLresult result = slCreateEngine(&engine,
100             1 /* numOptions */, EngineOption /* pEngineOptions */,
101             0 /* numInterfaces */, NULL /* pInterfaceIds */, NULL /* pInterfaceRequired */);
102     if (result != SL_RESULT_SUCCESS) {
103         ALOGE("slCreateEngine() failed: %s", getSLErrStr(result));
104         return NULL;
105     }
106     // realize the engine
107     result = (*engine)->Realize(engine, SL_BOOLEAN_FALSE /* async */);
108     if (result != SL_RESULT_SUCCESS) {
109         ALOGE("Realize() failed: %s", getSLErrStr(result));
110         (*engine)->Destroy(engine);
111         return NULL;
112     }
113     return engine;
114 }
115 
OpenSLEngine(bool global)116 SLObjectItf OpenSLEngine(bool global) {
117 
118     if (!global) {
119         return createEngine();
120     }
121     Mutex::Autolock l(gLock);
122     if (gRefCount == 0) {
123         gEngineObject = createEngine();
124     }
125     gRefCount++;
126     return gEngineObject;
127 }
128 
CloseSLEngine(SLObjectItf engine)129 void CloseSLEngine(SLObjectItf engine) {
130     Mutex::Autolock l(gLock);
131     if (engine == gEngineObject) {
132         if (gRefCount == 0) {
133             ALOGE("CloseSLEngine(%p): refcount already 0", engine);
134             return;
135         }
136         if (--gRefCount != 0) {
137             return;
138         }
139         gEngineObject = NULL;
140     }
141     (*engine)->Destroy(engine);
142 }
143 
144 } // namespace android
145 
146