1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker * Copyright (C) 2017 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker *
4*4d7e907cSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker *
8*4d7e907cSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker *
10*4d7e907cSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker */
16*4d7e907cSAndroid Build Coastguard Worker
17*4d7e907cSAndroid Build Coastguard Worker #ifndef ANDROID_HARDWARE_CAS_V1_0_FACTORY_LOADER_H_
18*4d7e907cSAndroid Build Coastguard Worker #define ANDROID_HARDWARE_CAS_V1_0_FACTORY_LOADER_H_
19*4d7e907cSAndroid Build Coastguard Worker
20*4d7e907cSAndroid Build Coastguard Worker #include <android-base/strings.h>
21*4d7e907cSAndroid Build Coastguard Worker #include <dirent.h>
22*4d7e907cSAndroid Build Coastguard Worker #include <dlfcn.h>
23*4d7e907cSAndroid Build Coastguard Worker #include "SharedLibrary.h"
24*4d7e907cSAndroid Build Coastguard Worker #include <utils/KeyedVector.h>
25*4d7e907cSAndroid Build Coastguard Worker #include <utils/Mutex.h>
26*4d7e907cSAndroid Build Coastguard Worker #include <media/cas/CasAPI.h>
27*4d7e907cSAndroid Build Coastguard Worker
28*4d7e907cSAndroid Build Coastguard Worker using namespace std;
29*4d7e907cSAndroid Build Coastguard Worker
30*4d7e907cSAndroid Build Coastguard Worker namespace android {
31*4d7e907cSAndroid Build Coastguard Worker namespace hardware {
32*4d7e907cSAndroid Build Coastguard Worker namespace cas {
33*4d7e907cSAndroid Build Coastguard Worker namespace V1_0 {
34*4d7e907cSAndroid Build Coastguard Worker namespace implementation {
35*4d7e907cSAndroid Build Coastguard Worker
36*4d7e907cSAndroid Build Coastguard Worker template <class T>
37*4d7e907cSAndroid Build Coastguard Worker class FactoryLoader {
38*4d7e907cSAndroid Build Coastguard Worker public:
FactoryLoader(const char * name)39*4d7e907cSAndroid Build Coastguard Worker FactoryLoader(const char *name) :
40*4d7e907cSAndroid Build Coastguard Worker mFactory(NULL), mCreateFactoryFuncName(name) {}
41*4d7e907cSAndroid Build Coastguard Worker
~FactoryLoader()42*4d7e907cSAndroid Build Coastguard Worker virtual ~FactoryLoader() { closeFactory(); }
43*4d7e907cSAndroid Build Coastguard Worker
44*4d7e907cSAndroid Build Coastguard Worker bool findFactoryForScheme(
45*4d7e907cSAndroid Build Coastguard Worker int32_t CA_system_id,
46*4d7e907cSAndroid Build Coastguard Worker sp<SharedLibrary> *library = NULL,
47*4d7e907cSAndroid Build Coastguard Worker T** factory = NULL);
48*4d7e907cSAndroid Build Coastguard Worker
49*4d7e907cSAndroid Build Coastguard Worker bool enumeratePlugins(vector<HidlCasPluginDescriptor>* results);
50*4d7e907cSAndroid Build Coastguard Worker
51*4d7e907cSAndroid Build Coastguard Worker private:
52*4d7e907cSAndroid Build Coastguard Worker typedef T*(*CreateFactoryFunc)();
53*4d7e907cSAndroid Build Coastguard Worker
54*4d7e907cSAndroid Build Coastguard Worker Mutex mMapLock;
55*4d7e907cSAndroid Build Coastguard Worker T* mFactory;
56*4d7e907cSAndroid Build Coastguard Worker const char *mCreateFactoryFuncName;
57*4d7e907cSAndroid Build Coastguard Worker sp<SharedLibrary> mLibrary;
58*4d7e907cSAndroid Build Coastguard Worker KeyedVector<int32_t, String8> mCASystemIdToLibraryPathMap;
59*4d7e907cSAndroid Build Coastguard Worker KeyedVector<String8, wp<SharedLibrary> > mLibraryPathToOpenLibraryMap;
60*4d7e907cSAndroid Build Coastguard Worker
61*4d7e907cSAndroid Build Coastguard Worker bool loadFactoryForSchemeFromPath(
62*4d7e907cSAndroid Build Coastguard Worker const String8 &path,
63*4d7e907cSAndroid Build Coastguard Worker int32_t CA_system_id,
64*4d7e907cSAndroid Build Coastguard Worker sp<SharedLibrary> *library,
65*4d7e907cSAndroid Build Coastguard Worker T** factory);
66*4d7e907cSAndroid Build Coastguard Worker
67*4d7e907cSAndroid Build Coastguard Worker bool queryPluginsFromPath(
68*4d7e907cSAndroid Build Coastguard Worker const String8 &path,
69*4d7e907cSAndroid Build Coastguard Worker vector<HidlCasPluginDescriptor>* results);
70*4d7e907cSAndroid Build Coastguard Worker
71*4d7e907cSAndroid Build Coastguard Worker bool openFactory(const String8 &path);
72*4d7e907cSAndroid Build Coastguard Worker void closeFactory();
73*4d7e907cSAndroid Build Coastguard Worker };
74*4d7e907cSAndroid Build Coastguard Worker
75*4d7e907cSAndroid Build Coastguard Worker template <class T>
findFactoryForScheme(int32_t CA_system_id,sp<SharedLibrary> * library,T ** factory)76*4d7e907cSAndroid Build Coastguard Worker bool FactoryLoader<T>::findFactoryForScheme(
77*4d7e907cSAndroid Build Coastguard Worker int32_t CA_system_id, sp<SharedLibrary> *library, T** factory) {
78*4d7e907cSAndroid Build Coastguard Worker if (library != NULL) {
79*4d7e907cSAndroid Build Coastguard Worker library->clear();
80*4d7e907cSAndroid Build Coastguard Worker }
81*4d7e907cSAndroid Build Coastguard Worker if (factory != NULL) {
82*4d7e907cSAndroid Build Coastguard Worker *factory = NULL;
83*4d7e907cSAndroid Build Coastguard Worker }
84*4d7e907cSAndroid Build Coastguard Worker
85*4d7e907cSAndroid Build Coastguard Worker Mutex::Autolock autoLock(mMapLock);
86*4d7e907cSAndroid Build Coastguard Worker
87*4d7e907cSAndroid Build Coastguard Worker // first check cache
88*4d7e907cSAndroid Build Coastguard Worker ssize_t index = mCASystemIdToLibraryPathMap.indexOfKey(CA_system_id);
89*4d7e907cSAndroid Build Coastguard Worker if (index >= 0) {
90*4d7e907cSAndroid Build Coastguard Worker return loadFactoryForSchemeFromPath(
91*4d7e907cSAndroid Build Coastguard Worker mCASystemIdToLibraryPathMap[index],
92*4d7e907cSAndroid Build Coastguard Worker CA_system_id, library, factory);
93*4d7e907cSAndroid Build Coastguard Worker }
94*4d7e907cSAndroid Build Coastguard Worker
95*4d7e907cSAndroid Build Coastguard Worker // no luck, have to search
96*4d7e907cSAndroid Build Coastguard Worker #ifdef __LP64__
97*4d7e907cSAndroid Build Coastguard Worker String8 dirPath("/vendor/lib64/mediacas");
98*4d7e907cSAndroid Build Coastguard Worker #else
99*4d7e907cSAndroid Build Coastguard Worker String8 dirPath("/vendor/lib/mediacas");
100*4d7e907cSAndroid Build Coastguard Worker #endif
101*4d7e907cSAndroid Build Coastguard Worker
102*4d7e907cSAndroid Build Coastguard Worker DIR* pDir = opendir(dirPath.c_str());
103*4d7e907cSAndroid Build Coastguard Worker
104*4d7e907cSAndroid Build Coastguard Worker if (pDir == NULL) {
105*4d7e907cSAndroid Build Coastguard Worker ALOGE("Failed to open plugin directory %s", dirPath.c_str());
106*4d7e907cSAndroid Build Coastguard Worker return false;
107*4d7e907cSAndroid Build Coastguard Worker }
108*4d7e907cSAndroid Build Coastguard Worker
109*4d7e907cSAndroid Build Coastguard Worker struct dirent* pEntry;
110*4d7e907cSAndroid Build Coastguard Worker while ((pEntry = readdir(pDir))) {
111*4d7e907cSAndroid Build Coastguard Worker String8 pluginPath = dirPath + "/" + pEntry->d_name;
112*4d7e907cSAndroid Build Coastguard Worker if (base::EndsWith(pluginPath.c_str(), ".so")) {
113*4d7e907cSAndroid Build Coastguard Worker if (loadFactoryForSchemeFromPath(
114*4d7e907cSAndroid Build Coastguard Worker pluginPath, CA_system_id, library, factory)) {
115*4d7e907cSAndroid Build Coastguard Worker mCASystemIdToLibraryPathMap.add(CA_system_id, pluginPath);
116*4d7e907cSAndroid Build Coastguard Worker closedir(pDir);
117*4d7e907cSAndroid Build Coastguard Worker
118*4d7e907cSAndroid Build Coastguard Worker return true;
119*4d7e907cSAndroid Build Coastguard Worker }
120*4d7e907cSAndroid Build Coastguard Worker }
121*4d7e907cSAndroid Build Coastguard Worker }
122*4d7e907cSAndroid Build Coastguard Worker
123*4d7e907cSAndroid Build Coastguard Worker closedir(pDir);
124*4d7e907cSAndroid Build Coastguard Worker
125*4d7e907cSAndroid Build Coastguard Worker ALOGE("Failed to find plugin");
126*4d7e907cSAndroid Build Coastguard Worker return false;
127*4d7e907cSAndroid Build Coastguard Worker }
128*4d7e907cSAndroid Build Coastguard Worker
129*4d7e907cSAndroid Build Coastguard Worker template <class T>
enumeratePlugins(vector<HidlCasPluginDescriptor> * results)130*4d7e907cSAndroid Build Coastguard Worker bool FactoryLoader<T>::enumeratePlugins(
131*4d7e907cSAndroid Build Coastguard Worker vector<HidlCasPluginDescriptor>* results) {
132*4d7e907cSAndroid Build Coastguard Worker ALOGI("enumeratePlugins");
133*4d7e907cSAndroid Build Coastguard Worker
134*4d7e907cSAndroid Build Coastguard Worker results->clear();
135*4d7e907cSAndroid Build Coastguard Worker
136*4d7e907cSAndroid Build Coastguard Worker #ifdef __LP64__
137*4d7e907cSAndroid Build Coastguard Worker String8 dirPath("/vendor/lib64/mediacas");
138*4d7e907cSAndroid Build Coastguard Worker #else
139*4d7e907cSAndroid Build Coastguard Worker String8 dirPath("/vendor/lib/mediacas");
140*4d7e907cSAndroid Build Coastguard Worker #endif
141*4d7e907cSAndroid Build Coastguard Worker
142*4d7e907cSAndroid Build Coastguard Worker DIR* pDir = opendir(dirPath.c_str());
143*4d7e907cSAndroid Build Coastguard Worker
144*4d7e907cSAndroid Build Coastguard Worker if (pDir == NULL) {
145*4d7e907cSAndroid Build Coastguard Worker ALOGE("Failed to open plugin directory %s", dirPath.c_str());
146*4d7e907cSAndroid Build Coastguard Worker return false;
147*4d7e907cSAndroid Build Coastguard Worker }
148*4d7e907cSAndroid Build Coastguard Worker
149*4d7e907cSAndroid Build Coastguard Worker Mutex::Autolock autoLock(mMapLock);
150*4d7e907cSAndroid Build Coastguard Worker
151*4d7e907cSAndroid Build Coastguard Worker struct dirent* pEntry;
152*4d7e907cSAndroid Build Coastguard Worker while ((pEntry = readdir(pDir))) {
153*4d7e907cSAndroid Build Coastguard Worker String8 pluginPath = dirPath + "/" + pEntry->d_name;
154*4d7e907cSAndroid Build Coastguard Worker if (base::EndsWith(pluginPath.c_str(), ".so")) {
155*4d7e907cSAndroid Build Coastguard Worker queryPluginsFromPath(pluginPath, results);
156*4d7e907cSAndroid Build Coastguard Worker }
157*4d7e907cSAndroid Build Coastguard Worker }
158*4d7e907cSAndroid Build Coastguard Worker return true;
159*4d7e907cSAndroid Build Coastguard Worker }
160*4d7e907cSAndroid Build Coastguard Worker
161*4d7e907cSAndroid Build Coastguard Worker template <class T>
loadFactoryForSchemeFromPath(const String8 & path,int32_t CA_system_id,sp<SharedLibrary> * library,T ** factory)162*4d7e907cSAndroid Build Coastguard Worker bool FactoryLoader<T>::loadFactoryForSchemeFromPath(
163*4d7e907cSAndroid Build Coastguard Worker const String8 &path, int32_t CA_system_id,
164*4d7e907cSAndroid Build Coastguard Worker sp<SharedLibrary> *library, T** factory) {
165*4d7e907cSAndroid Build Coastguard Worker closeFactory();
166*4d7e907cSAndroid Build Coastguard Worker
167*4d7e907cSAndroid Build Coastguard Worker if (!openFactory(path) || !mFactory->isSystemIdSupported(CA_system_id)) {
168*4d7e907cSAndroid Build Coastguard Worker closeFactory();
169*4d7e907cSAndroid Build Coastguard Worker return false;
170*4d7e907cSAndroid Build Coastguard Worker }
171*4d7e907cSAndroid Build Coastguard Worker
172*4d7e907cSAndroid Build Coastguard Worker if (library != NULL) {
173*4d7e907cSAndroid Build Coastguard Worker *library = mLibrary;
174*4d7e907cSAndroid Build Coastguard Worker }
175*4d7e907cSAndroid Build Coastguard Worker if (factory != NULL) {
176*4d7e907cSAndroid Build Coastguard Worker *factory = mFactory;
177*4d7e907cSAndroid Build Coastguard Worker }
178*4d7e907cSAndroid Build Coastguard Worker return true;
179*4d7e907cSAndroid Build Coastguard Worker }
180*4d7e907cSAndroid Build Coastguard Worker
181*4d7e907cSAndroid Build Coastguard Worker template <class T>
queryPluginsFromPath(const String8 & path,vector<HidlCasPluginDescriptor> * results)182*4d7e907cSAndroid Build Coastguard Worker bool FactoryLoader<T>::queryPluginsFromPath(
183*4d7e907cSAndroid Build Coastguard Worker const String8 &path, vector<HidlCasPluginDescriptor>* results) {
184*4d7e907cSAndroid Build Coastguard Worker closeFactory();
185*4d7e907cSAndroid Build Coastguard Worker
186*4d7e907cSAndroid Build Coastguard Worker vector<CasPluginDescriptor> descriptors;
187*4d7e907cSAndroid Build Coastguard Worker if (!openFactory(path) || mFactory->queryPlugins(&descriptors) != OK) {
188*4d7e907cSAndroid Build Coastguard Worker closeFactory();
189*4d7e907cSAndroid Build Coastguard Worker return false;
190*4d7e907cSAndroid Build Coastguard Worker }
191*4d7e907cSAndroid Build Coastguard Worker
192*4d7e907cSAndroid Build Coastguard Worker for (auto it = descriptors.begin(); it != descriptors.end(); it++) {
193*4d7e907cSAndroid Build Coastguard Worker results->push_back( HidlCasPluginDescriptor {
194*4d7e907cSAndroid Build Coastguard Worker .caSystemId = it->CA_system_id,
195*4d7e907cSAndroid Build Coastguard Worker .name = it->name.c_str()});
196*4d7e907cSAndroid Build Coastguard Worker }
197*4d7e907cSAndroid Build Coastguard Worker return true;
198*4d7e907cSAndroid Build Coastguard Worker }
199*4d7e907cSAndroid Build Coastguard Worker
200*4d7e907cSAndroid Build Coastguard Worker template <class T>
openFactory(const String8 & path)201*4d7e907cSAndroid Build Coastguard Worker bool FactoryLoader<T>::openFactory(const String8 &path) {
202*4d7e907cSAndroid Build Coastguard Worker // get strong pointer to open shared library
203*4d7e907cSAndroid Build Coastguard Worker ssize_t index = mLibraryPathToOpenLibraryMap.indexOfKey(path);
204*4d7e907cSAndroid Build Coastguard Worker if (index >= 0) {
205*4d7e907cSAndroid Build Coastguard Worker mLibrary = mLibraryPathToOpenLibraryMap[index].promote();
206*4d7e907cSAndroid Build Coastguard Worker } else {
207*4d7e907cSAndroid Build Coastguard Worker index = mLibraryPathToOpenLibraryMap.add(path, NULL);
208*4d7e907cSAndroid Build Coastguard Worker }
209*4d7e907cSAndroid Build Coastguard Worker
210*4d7e907cSAndroid Build Coastguard Worker if (!mLibrary.get()) {
211*4d7e907cSAndroid Build Coastguard Worker mLibrary = new SharedLibrary(path);
212*4d7e907cSAndroid Build Coastguard Worker if (!*mLibrary) {
213*4d7e907cSAndroid Build Coastguard Worker return false;
214*4d7e907cSAndroid Build Coastguard Worker }
215*4d7e907cSAndroid Build Coastguard Worker
216*4d7e907cSAndroid Build Coastguard Worker mLibraryPathToOpenLibraryMap.replaceValueAt(index, mLibrary);
217*4d7e907cSAndroid Build Coastguard Worker }
218*4d7e907cSAndroid Build Coastguard Worker
219*4d7e907cSAndroid Build Coastguard Worker CreateFactoryFunc createFactory =
220*4d7e907cSAndroid Build Coastguard Worker (CreateFactoryFunc)mLibrary->lookup(mCreateFactoryFuncName);
221*4d7e907cSAndroid Build Coastguard Worker if (createFactory == NULL || (mFactory = createFactory()) == NULL) {
222*4d7e907cSAndroid Build Coastguard Worker return false;
223*4d7e907cSAndroid Build Coastguard Worker }
224*4d7e907cSAndroid Build Coastguard Worker return true;
225*4d7e907cSAndroid Build Coastguard Worker }
226*4d7e907cSAndroid Build Coastguard Worker
227*4d7e907cSAndroid Build Coastguard Worker template <class T>
closeFactory()228*4d7e907cSAndroid Build Coastguard Worker void FactoryLoader<T>::closeFactory() {
229*4d7e907cSAndroid Build Coastguard Worker delete mFactory;
230*4d7e907cSAndroid Build Coastguard Worker mFactory = NULL;
231*4d7e907cSAndroid Build Coastguard Worker mLibrary.clear();
232*4d7e907cSAndroid Build Coastguard Worker }
233*4d7e907cSAndroid Build Coastguard Worker
234*4d7e907cSAndroid Build Coastguard Worker } // namespace implementation
235*4d7e907cSAndroid Build Coastguard Worker } // namespace V1_0
236*4d7e907cSAndroid Build Coastguard Worker } // namespace cas
237*4d7e907cSAndroid Build Coastguard Worker } // namespace hardware
238*4d7e907cSAndroid Build Coastguard Worker } // namespace android
239*4d7e907cSAndroid Build Coastguard Worker
240*4d7e907cSAndroid Build Coastguard Worker #endif // ANDROID_HARDWARE_CAS_V1_0_FACTORY_LOADER_H_
241