/* * 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__) #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 = NULL; #endif #if !defined(_WIN32) void add_sysprop_change_callback(sysprop_change_callback cb, int priority) { pthread_mutex_lock(&gSyspropMutex); if (gSyspropList == NULL) { 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__) 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__) // 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 != NULL) { 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 }