| /* | 
 |  * Copyright (C) 2005 The Android Open Source Project | 
 |  * | 
 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 |  * you may not use this file except in compliance with the License. | 
 |  * You may obtain a copy of the License at | 
 |  * | 
 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 |  * | 
 |  * Unless required by applicable law or agreed to in writing, software | 
 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 |  * See the License for the specific language governing permissions and | 
 |  * limitations under the License. | 
 |  */ | 
 |  | 
 | #define LOG_TAG "misc" | 
 |  | 
 | #include <utils/misc.h> | 
 |  | 
 | #include <pthread.h> | 
 |  | 
 | #include <utils/Log.h> | 
 | #include <utils/Vector.h> | 
 |  | 
 | #if defined(__ANDROID__) && !defined(__ANDROID_RECOVERY__) | 
 | #include <dlfcn.h> | 
 | #include <vndksupport/linker.h> | 
 | #endif | 
 |  | 
 | extern "C" void do_report_sysprop_change(); | 
 |  | 
 | using namespace android; | 
 |  | 
 | namespace android { | 
 |  | 
 | struct sysprop_change_callback_info { | 
 |     sysprop_change_callback callback; | 
 |     int priority; | 
 | }; | 
 |  | 
 | #if !defined(_WIN32) | 
 | static pthread_mutex_t gSyspropMutex = PTHREAD_MUTEX_INITIALIZER; | 
 | static Vector<sysprop_change_callback_info>* gSyspropList = nullptr; | 
 | #endif | 
 |  | 
 | #if !defined(_WIN32) | 
 | void add_sysprop_change_callback(sysprop_change_callback cb, int priority) { | 
 |     pthread_mutex_lock(&gSyspropMutex); | 
 |     if (gSyspropList == nullptr) { | 
 |         gSyspropList = new Vector<sysprop_change_callback_info>(); | 
 |     } | 
 |     sysprop_change_callback_info info; | 
 |     info.callback = cb; | 
 |     info.priority = priority; | 
 |     bool added = false; | 
 |     for (size_t i=0; i<gSyspropList->size(); i++) { | 
 |         if (priority >= gSyspropList->itemAt(i).priority) { | 
 |             gSyspropList->insertAt(info, i); | 
 |             added = true; | 
 |             break; | 
 |         } | 
 |     } | 
 |     if (!added) { | 
 |         gSyspropList->add(info); | 
 |     } | 
 |     pthread_mutex_unlock(&gSyspropMutex); | 
 | } | 
 | #else | 
 | void add_sysprop_change_callback(sysprop_change_callback, int) {} | 
 | #endif | 
 |  | 
 | #if defined(__ANDROID__) && !defined(__ANDROID_RECOVERY__) | 
 | void (*get_report_sysprop_change_func())() { | 
 |     void (*func)() = nullptr; | 
 |     void* handle = android_load_sphal_library("libutils.so", RTLD_NOW); | 
 |     if (handle != nullptr) { | 
 |         func = reinterpret_cast<decltype(func)>(dlsym(handle, "do_report_sysprop_change")); | 
 |     } | 
 |  | 
 |     return func; | 
 | } | 
 | #endif | 
 |  | 
 | void report_sysprop_change() { | 
 |     do_report_sysprop_change(); | 
 |  | 
 | #if defined(__ANDROID__) && !defined(__ANDROID_RECOVERY__) | 
 |     // libutils.so is double loaded; from the default namespace and from the | 
 |     // 'sphal' namespace. Redirect the sysprop change event to the other instance | 
 |     // of libutils.so loaded in the 'sphal' namespace so that listeners attached | 
 |     // to that instance is also notified with this event. | 
 |     static auto func = get_report_sysprop_change_func(); | 
 |     if (func != nullptr) { | 
 |         (*func)(); | 
 |     } | 
 | #endif | 
 | } | 
 |  | 
 | };  // namespace android | 
 |  | 
 | void do_report_sysprop_change() { | 
 | #if !defined(_WIN32) | 
 |     pthread_mutex_lock(&gSyspropMutex); | 
 |     Vector<sysprop_change_callback_info> listeners; | 
 |     if (gSyspropList != nullptr) { | 
 |         listeners = *gSyspropList; | 
 |     } | 
 |     pthread_mutex_unlock(&gSyspropMutex); | 
 |  | 
 |     //ALOGI("Reporting sysprop change to %d listeners", listeners.size()); | 
 |     for (size_t i=0; i<listeners.size(); i++) { | 
 |         listeners[i].callback(); | 
 |     } | 
 | #endif | 
 | } |