/* * 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" // // Miscellaneous utility functions. // #include <utils/misc.h> #include <utils/Log.h> #include <sys/stat.h> #include <string.h> #include <errno.h> #include <assert.h> #include <stdio.h> #if defined(HAVE_PTHREADS) # include <pthread.h> #endif #include <utils/Vector.h> using namespace android; namespace android { /* * Like strdup(), but uses C++ "new" operator instead of malloc. */ char* strdupNew(const char* str) { char* newStr; int len; if (str == NULL) return NULL; len = strlen(str); newStr = new char[len+1]; memcpy(newStr, str, len+1); return newStr; } /* * Concatenate an argument vector. */ char* concatArgv(int argc, const char* const argv[]) { char* newStr = NULL; int len, totalLen, posn, idx; /* * First, figure out the total length. */ totalLen = idx = 0; while (1) { if (idx == argc || argv[idx] == NULL) break; if (idx) totalLen++; // leave a space between args totalLen += strlen(argv[idx]); idx++; } /* * Alloc the string. */ newStr = new char[totalLen +1]; if (newStr == NULL) return NULL; /* * Finally, allocate the string and copy data over. */ idx = posn = 0; while (1) { if (idx == argc || argv[idx] == NULL) break; if (idx) newStr[posn++] = ' '; len = strlen(argv[idx]); memcpy(&newStr[posn], argv[idx], len); posn += len; idx++; } assert(posn == totalLen); newStr[posn] = '\0'; return newStr; } /* * Count the #of args in an argument vector. Don't count the final NULL. */ int countArgv(const char* const argv[]) { int count = 0; while (argv[count] != NULL) count++; return count; } #include <stdio.h> /* * Get a file's type. */ FileType getFileType(const char* fileName) { struct stat sb; if (stat(fileName, &sb) < 0) { if (errno == ENOENT || errno == ENOTDIR) return kFileTypeNonexistent; else { fprintf(stderr, "getFileType got errno=%d on '%s'\n", errno, fileName); return kFileTypeUnknown; } } else { if (S_ISREG(sb.st_mode)) return kFileTypeRegular; else if (S_ISDIR(sb.st_mode)) return kFileTypeDirectory; else if (S_ISCHR(sb.st_mode)) return kFileTypeCharDev; else if (S_ISBLK(sb.st_mode)) return kFileTypeBlockDev; else if (S_ISFIFO(sb.st_mode)) return kFileTypeFifo; #ifdef HAVE_SYMLINKS else if (S_ISLNK(sb.st_mode)) return kFileTypeSymlink; else if (S_ISSOCK(sb.st_mode)) return kFileTypeSocket; #endif else return kFileTypeUnknown; } } /* * Get a file's modification date. */ time_t getFileModDate(const char* fileName) { struct stat sb; if (stat(fileName, &sb) < 0) return (time_t) -1; return sb.st_mtime; } /* * Round up to the next highest power of 2. * * Found on http://graphics.stanford.edu/~seander/bithacks.html. */ unsigned int roundUpPower2(unsigned int val) { val--; val |= val >> 1; val |= val >> 2; val |= val >> 4; val |= val >> 8; val |= val >> 16; val++; return val; } struct sysprop_change_callback_info { sysprop_change_callback callback; int priority; }; #if defined(HAVE_PTHREADS) static pthread_mutex_t gSyspropMutex = PTHREAD_MUTEX_INITIALIZER; static Vector<sysprop_change_callback_info>* gSyspropList = NULL; #endif void add_sysprop_change_callback(sysprop_change_callback cb, int priority) { #if defined(HAVE_PTHREADS) 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); #endif } void report_sysprop_change() { #if defined(HAVE_PTHREADS) 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 } }; // namespace android