android_util_Process.cpp revision d34fac541a5a541b4207ee9cdbcc5005b5fa848c
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* //device/libs/android_runtime/android_util_Process.cpp 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** Copyright 2006, The Android Open Source Project 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** 569a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** Licensed under the Apache License, Version 2.0 (the "License"); 669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** you may not use this file except in compliance with the License. 769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** You may obtain a copy of the License at 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** 969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** http://www.apache.org/licenses/LICENSE-2.0 109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** 1169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** Unless required by applicable law or agreed to in writing, software 1269a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** distributed under the License is distributed on an "AS IS" BASIS, 1369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 1469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes** See the License for the specific language governing permissions and 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** limitations under the License. 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project*/ 179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#define LOG_TAG "Process" 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 20cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen// To make sure cpu_set_t is included from sched.h 21cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen#define _GNU_SOURCE 1 229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Log.h> 230795272aa226f4e965968a03daddc53ce30b7cdaMathias Agopian#include <binder/IPCThreadState.h> 240795272aa226f4e965968a03daddc53ce30b7cdaMathias Agopian#include <binder/IServiceManager.h> 25f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten#include <cutils/sched_policy.h> 269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String8.h> 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Vector.h> 280769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross#include <processgroup/processgroup.h> 299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 30ed6b9dff563c5e22f040ff37e12c0d771e0478aeAndreas Gampe#include "core_jni_helpers.h" 319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "android_util_Binder.h" 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "JNIHelp.h" 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <dirent.h> 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <fcntl.h> 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <grp.h> 38c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn#include <inttypes.h> 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <pwd.h> 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <signal.h> 41c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn#include <sys/errno.h> 42c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn#include <sys/resource.h> 43c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn#include <sys/stat.h> 44c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn#include <sys/types.h> 456af763bec7c3f4d50fee8dd0046409bb8a7fe8f6Glenn Kasten#include <unistd.h> 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 47160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#define GUARD_THREAD_PRIORITY 0 48a5109a878eeff22e32ee5ce1b1cd15e8daad5234San Mehat 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectusing namespace android; 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 510f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampestatic const bool kDebugPolicy = false; 520f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampestatic const bool kDebugProc = false; 530f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe 54160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#if GUARD_THREAD_PRIORITY 55160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher TateMutex gKeyCreateMutex; 56160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tatestatic pthread_key_t gBgKey = -1; 57160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#endif 58160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate 59f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten// For both of these, err should be in the errno range (positive), not a status_t (negative) 604c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunkstatic void signalExceptionForError(JNIEnv* env, int err, int tid) { 619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project switch (err) { 629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case EINVAL: 634c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk jniThrowExceptionFmt(env, "java/lang/IllegalArgumentException", 644c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk "Invalid argument: %d", tid); 659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case ESRCH: 674c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk jniThrowExceptionFmt(env, "java/lang/IllegalArgumentException", 684c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk "Given thread %d does not exist", tid); 699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case EPERM: 714c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk jniThrowExceptionFmt(env, "java/lang/SecurityException", 724c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk "No permission to modify given thread %d", tid); 739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project default: 759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/lang/RuntimeException", "Unknown error"); 769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 804c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunkstatic void signalExceptionForPriorityError(JNIEnv* env, int err, int tid) { 81e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat switch (err) { 824c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk case EACCES: 834c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk jniThrowExceptionFmt(env, "java/lang/SecurityException", 844c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk "No permission to set the priority of %d", tid); 85e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat break; 864c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk default: 874c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForError(env, err, tid); 88e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat break; 894c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk } 904c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk 914c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk} 924c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk 934c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunkstatic void signalExceptionForGroupError(JNIEnv* env, int err, int tid) { 944c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk switch (err) { 95e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat case EACCES: 964c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk jniThrowExceptionFmt(env, "java/lang/SecurityException", 974c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk "No permission to set the group of %d", tid); 98e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat break; 99e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat default: 1004c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForError(env, err, tid); 101e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat break; 102e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat } 103e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat} 104e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat 1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectjint android_os_Process_getUidForName(JNIEnv* env, jobject clazz, jstring name) 1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (name == NULL) { 10869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jchar* str16 = env->GetStringCritical(name, 0); 1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String8 name8; 1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (str16) { 1156698749dd4d4d6513b26aa9071af290b956b68a7Dan Albert name8 = String8(reinterpret_cast<const char16_t*>(str16), 1166698749dd4d4d6513b26aa9071af290b956b68a7Dan Albert env->GetStringLength(name)); 1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringCritical(name, str16); 1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const size_t N = name8.size(); 1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (N > 0) { 1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* str = name8.string(); 1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (size_t i=0; i<N; i++) { 1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (str[i] < '0' || str[i] > '9') { 1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project struct passwd* pwd = getpwnam(str); 1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (pwd == NULL) { 1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return pwd->pw_uid; 1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return atoi(str); 1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectjint android_os_Process_getGidForName(JNIEnv* env, jobject clazz, jstring name) 1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (name == NULL) { 14069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jchar* str16 = env->GetStringCritical(name, 0); 1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String8 name8; 1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (str16) { 1476698749dd4d4d6513b26aa9071af290b956b68a7Dan Albert name8 = String8(reinterpret_cast<const char16_t*>(str16), 1486698749dd4d4d6513b26aa9071af290b956b68a7Dan Albert env->GetStringLength(name)); 1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringCritical(name, str16); 1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const size_t N = name8.size(); 1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (N > 0) { 1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* str = name8.string(); 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (size_t i=0; i<N; i++) { 1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (str[i] < '0' || str[i] > '9') { 1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project struct group* grp = getgrnam(str); 1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (grp == NULL) { 1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return grp->gr_gid; 1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return atoi(str); 1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 169f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kastenvoid android_os_Process_setThreadGroup(JNIEnv* env, jobject clazz, int tid, jint grp) 170e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat{ 171c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn ALOGV("%s tid=%d grp=%" PRId32, __func__, tid, grp); 172f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten SchedPolicy sp = (SchedPolicy) grp; 173f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten int res = set_sched_policy(tid, sp); 174887f355f99ff83d568ef2885a4fdcaae475583dfDianne Hackborn if (res != NO_ERROR) { 1754c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, -res, tid); 176e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat } 177e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat} 178e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat 17969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughesvoid android_os_Process_setProcessGroup(JNIEnv* env, jobject clazz, int pid, jint grp) 1803e458241d9930465a20a861ecb42744355d48e48San Mehat{ 181c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn ALOGV("%s pid=%d grp=%" PRId32, __func__, pid, grp); 1823e458241d9930465a20a861ecb42744355d48e48San Mehat DIR *d; 1833e458241d9930465a20a861ecb42744355d48e48San Mehat char proc_path[255]; 1843e458241d9930465a20a861ecb42744355d48e48San Mehat struct dirent *de; 1853e458241d9930465a20a861ecb42744355d48e48San Mehat 186f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten if ((grp == SP_FOREGROUND) || (grp > SP_MAX)) { 1874c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, EINVAL, pid); 1883e458241d9930465a20a861ecb42744355d48e48San Mehat return; 1893e458241d9930465a20a861ecb42744355d48e48San Mehat } 1903e458241d9930465a20a861ecb42744355d48e48San Mehat 191f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten bool isDefault = false; 192f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten if (grp < 0) { 193f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten grp = SP_FOREGROUND; 194f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten isDefault = true; 195f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten } 196f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten SchedPolicy sp = (SchedPolicy) grp; 197f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten 1980f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe if (kDebugPolicy) { 1990f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe char cmdline[32]; 2000f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe int fd; 201a5109a878eeff22e32ee5ce1b1cd15e8daad5234San Mehat 2020f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe strcpy(cmdline, "unknown"); 203a5109a878eeff22e32ee5ce1b1cd15e8daad5234San Mehat 2040f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe sprintf(proc_path, "/proc/%d/cmdline", pid); 2050f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe fd = open(proc_path, O_RDONLY); 2060f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe if (fd >= 0) { 2070f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe int rc = read(fd, cmdline, sizeof(cmdline)-1); 2080f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe cmdline[rc] = 0; 2090f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe close(fd); 2100f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe } 21169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 2120f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe if (sp == SP_BACKGROUND) { 2130f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe ALOGD("setProcessGroup: vvv pid %d (%s)", pid, cmdline); 2140f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe } else { 2150f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe ALOGD("setProcessGroup: ^^^ pid %d (%s)", pid, cmdline); 2160f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe } 217a5109a878eeff22e32ee5ce1b1cd15e8daad5234San Mehat } 2180f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe 2193e458241d9930465a20a861ecb42744355d48e48San Mehat sprintf(proc_path, "/proc/%d/task", pid); 2203e458241d9930465a20a861ecb42744355d48e48San Mehat if (!(d = opendir(proc_path))) { 2211fd0ec738b0a2b97cc28701aa37b1a9869afc684San Mehat // If the process exited on us, don't generate an exception 2221fd0ec738b0a2b97cc28701aa37b1a9869afc684San Mehat if (errno != ENOENT) 2234c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, errno, pid); 2243e458241d9930465a20a861ecb42744355d48e48San Mehat return; 2253e458241d9930465a20a861ecb42744355d48e48San Mehat } 2263e458241d9930465a20a861ecb42744355d48e48San Mehat 2273e458241d9930465a20a861ecb42744355d48e48San Mehat while ((de = readdir(d))) { 2287e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat int t_pid; 2297e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat int t_pri; 2307e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat 2313e458241d9930465a20a861ecb42744355d48e48San Mehat if (de->d_name[0] == '.') 2323e458241d9930465a20a861ecb42744355d48e48San Mehat continue; 2337e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat t_pid = atoi(de->d_name); 2347e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat 2357e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat if (!t_pid) { 2363762c311729fe9f3af085c14c5c1fb471d994c03Steve Block ALOGE("Error getting pid for '%s'\n", de->d_name); 2377e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat continue; 2387e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat } 2391fd0ec738b0a2b97cc28701aa37b1a9869afc684San Mehat 24007b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten t_pri = getpriority(PRIO_PROCESS, t_pid); 24107b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten 24207b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten if (t_pri <= ANDROID_PRIORITY_AUDIO) { 24320375fe124b4bc54a2380396fd798f93fcdb3135Tim Murray int scheduler = sched_getscheduler(t_pid) & ~SCHED_RESET_ON_FORK; 24407b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten if ((scheduler == SCHED_FIFO) || (scheduler == SCHED_RR)) { 2459e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray // This task wants to stay in its current audio group so it can keep its budget 2469e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray // don't update its cpuset or cgroup 24707b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten continue; 24807b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten } 24907b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten } 2507e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat 25107b0465095bd9ab3412caefa4fcacbdc3825c64bGlenn Kasten if (isDefault) { 252f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten if (t_pri >= ANDROID_PRIORITY_BACKGROUND) { 253f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten // This task wants to stay at background 2549e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray // update its cpuset so it doesn't only run on bg core(s) 255d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen if (cpusets_enabled()) { 256d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen int err = set_cpuset_policy(t_pid, sp); 257d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen if (err != NO_ERROR) { 258d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen signalExceptionForGroupError(env, -err, t_pid); 259d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen break; 260d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen } 2619e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray } 262f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten continue; 263f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten } 2647e63789a0e0689d940609b1daceebc1bc08dcbefSan Mehat } 2659e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray int err; 266d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen 267d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen if (cpusets_enabled()) { 268d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen // set both cpuset and cgroup for general threads 269d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen err = set_cpuset_policy(t_pid, sp); 270d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen if (err != NO_ERROR) { 271d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen signalExceptionForGroupError(env, -err, t_pid); 272d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen break; 273d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen } 2749e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray } 27569a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 2769e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray err = set_sched_policy(t_pid, sp); 277f1b56449f58963e4f0473d5e26961f68c31759f4Glenn Kasten if (err != NO_ERROR) { 2784c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, -err, t_pid); 2790a42b811aea490a9a605b75f0320101f6eafd283San Mehat break; 280242d65bf9faf1d2bc3468490e510551140e23462San Mehat } 2819e41c7479ccaedb89a89a58079570ca0ee0e3727Tim Murray 2823e458241d9930465a20a861ecb42744355d48e48San Mehat } 2833e458241d9930465a20a861ecb42744355d48e48San Mehat closedir(d); 2843e458241d9930465a20a861ecb42744355d48e48San Mehat} 2853e458241d9930465a20a861ecb42744355d48e48San Mehat 2869e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkeyjint android_os_Process_getProcessGroup(JNIEnv* env, jobject clazz, jint pid) 2879e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey{ 2889e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey SchedPolicy sp; 2899e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey if (get_sched_policy(pid, &sp) != 0) { 2904c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, errno, pid); 2919e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey } 2929e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey return (int) sp; 2939e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey} 2949e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey 295cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen/** Sample CPUset list format: 296cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen * 0-3,4,6-8 297cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen */ 298cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenenstatic void parse_cpuset_cpus(char *cpus, cpu_set_t *cpu_set) { 299cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen unsigned int start, end, matched, i; 300cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen char *cpu_range = strtok(cpus, ","); 301cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen while (cpu_range != NULL) { 302cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen start = end = 0; 303cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen matched = sscanf(cpu_range, "%u-%u", &start, &end); 304cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen cpu_range = strtok(NULL, ","); 305cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (start >= CPU_SETSIZE) { 306cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen ALOGE("parse_cpuset_cpus: ignoring CPU number larger than %d.", CPU_SETSIZE); 307cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen continue; 308cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } else if (end >= CPU_SETSIZE) { 309cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen ALOGE("parse_cpuset_cpus: ignoring CPU numbers larger than %d.", CPU_SETSIZE); 310cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen end = CPU_SETSIZE - 1; 311cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 312cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (matched == 1) { 313cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen CPU_SET(start, cpu_set); 314cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } else if (matched == 2) { 315cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen for (i = start; i <= end; i++) { 316cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen CPU_SET(i, cpu_set); 317cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 318cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } else { 319cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen ALOGE("Failed to match cpus"); 320cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 321cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 322cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen return; 323cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen} 324cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 325cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen/** 326cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen * Stores the CPUs assigned to the cpuset corresponding to the 327cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen * SchedPolicy in the passed in cpu_set. 328cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen */ 329cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenenstatic void get_cpuset_cores_for_policy(SchedPolicy policy, cpu_set_t *cpu_set) 330cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen{ 331cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen FILE *file; 332cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen const char *filename; 333cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 334cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen CPU_ZERO(cpu_set); 335cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 336cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen switch (policy) { 337cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen case SP_BACKGROUND: 338cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen filename = "/dev/cpuset/background/cpus"; 339cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen break; 340cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen case SP_FOREGROUND: 341cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen case SP_AUDIO_APP: 342cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen case SP_AUDIO_SYS: 343cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen filename = "/dev/cpuset/foreground/cpus"; 344cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen break; 345cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen case SP_TOP_APP: 346cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen filename = "/dev/cpuset/top-app/cpus"; 347cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen break; 348cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen default: 349cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen filename = NULL; 350cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 351cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 352cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (!filename) return; 353cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 354cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen file = fopen(filename, "re"); 355cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (file != NULL) { 356cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen // Parse cpus string 357cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen char *line = NULL; 358cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen size_t len = 0; 359cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen ssize_t num_read = getline(&line, &len, file); 360cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen fclose (file); 361cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (num_read > 0) { 362cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen parse_cpuset_cpus(line, cpu_set); 363cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } else { 364cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen ALOGE("Failed to read %s", filename); 365cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 366cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen free(line); 367cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 368cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen return; 369cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen} 370cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 371cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 372cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen/** 373cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen * Determine CPU cores exclusively assigned to the 374cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen * cpuset corresponding to the SchedPolicy and store 375cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen * them in the passed in cpu_set_t 376cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen */ 377cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenenvoid get_exclusive_cpuset_cores(SchedPolicy policy, cpu_set_t *cpu_set) { 378d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen if (cpusets_enabled()) { 379d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen int i; 380d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen cpu_set_t tmp_set; 381d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen get_cpuset_cores_for_policy(policy, cpu_set); 382d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen for (i = 0; i < SP_CNT; i++) { 383d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen if ((SchedPolicy) i == policy) continue; 384d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen get_cpuset_cores_for_policy((SchedPolicy)i, &tmp_set); 385d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen // First get cores exclusive to one set or the other 386d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen CPU_XOR(&tmp_set, cpu_set, &tmp_set); 387d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen // Then get the ones only in cpu_set 388d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen CPU_AND(cpu_set, cpu_set, &tmp_set); 389d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen } 390d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen } else { 391d34fac541a5a541b4207ee9cdbcc5005b5fa848cIsaac Chen CPU_ZERO(cpu_set); 392cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 393cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen return; 394cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen} 395cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 396cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn CoenenjintArray android_os_Process_getExclusiveCores(JNIEnv* env, jobject clazz) { 397cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen SchedPolicy sp; 398cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen cpu_set_t cpu_set; 399cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen jintArray cpus; 400cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen int pid = getpid(); 401cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (get_sched_policy(pid, &sp) != 0) { 4024c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, errno, pid); 403cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen return NULL; 404cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 405cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen get_exclusive_cpuset_cores(sp, &cpu_set); 406cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen int num_cpus = CPU_COUNT(&cpu_set); 407cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen cpus = env->NewIntArray(num_cpus); 408cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (cpus == NULL) { 409cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen jniThrowException(env, "java/lang/OutOfMemoryError", NULL); 410cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen return NULL; 411cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 412cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 413cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen jint* cpu_elements = env->GetIntArrayElements(cpus, 0); 414cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen int count = 0; 415cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen for (int i = 0; i < CPU_SETSIZE && count < num_cpus; i++) { 416cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen if (CPU_ISSET(i, &cpu_set)) { 417cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen cpu_elements[count++] = i; 418cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 419cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen } 420cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 421cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen env->ReleaseIntArrayElements(cpus, cpu_elements, 0); 422cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen return cpus; 423cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen} 424cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen 425160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tatestatic void android_os_Process_setCanSelfBackground(JNIEnv* env, jobject clazz, jboolean bgOk) { 426160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate // Establishes the calling thread as illegal to put into the background. 427160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate // Typically used only for the system process's main looper. 428160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#if GUARD_THREAD_PRIORITY 42906451fe081d5ae79121a6f301475c7042f7f3a5dElliott Hughes ALOGV("Process.setCanSelfBackground(%d) : tid=%d", bgOk, gettid()); 430160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate { 431160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate Mutex::Autolock _l(gKeyCreateMutex); 432160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate if (gBgKey == -1) { 433160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate pthread_key_create(&gBgKey, NULL); 434160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate } 435160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate } 436160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate 437160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate // inverted: not-okay, we set a sentinel value 438160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate pthread_setspecific(gBgKey, (void*)(bgOk ? 0 : 0xbaad)); 439160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#endif 440160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate} 441160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate 4421b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharanjint android_os_Process_getThreadScheduler(JNIEnv* env, jclass clazz, 4431b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan jint tid) 4441b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan{ 4451b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan int policy = 0; 4461b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan// linux has sched_getscheduler(), others don't. 4471b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan#if defined(__linux__) 4481b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan errno = 0; 4491b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan policy = sched_getscheduler(tid); 4501b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan if (errno != 0) { 4514c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForPriorityError(env, errno, tid); 4521b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan } 4531b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan#else 4544c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForPriorityError(env, ENOSYS, tid); 4551b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan#endif 4561b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan return policy; 4571b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan} 4581b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan 4596793ac943afeb16642f477c43ddfd27e498db37bGlenn Kastenvoid android_os_Process_setThreadScheduler(JNIEnv* env, jclass clazz, 4606793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten jint tid, jint policy, jint pri) 4616793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten{ 46265b4a68669227a57696e0e7ed1e4ef8da2705d9eYabin Cui// linux has sched_setscheduler(), others don't. 46365b4a68669227a57696e0e7ed1e4ef8da2705d9eYabin Cui#if defined(__linux__) 4646793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten struct sched_param param; 4656793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten param.sched_priority = pri; 4666793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten int rc = sched_setscheduler(tid, policy, ¶m); 4676793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten if (rc) { 4684c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForPriorityError(env, errno, tid); 4696793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten } 470cc767191cfb675f744e0165608b0a4196aba2b37Glenn Kasten#else 4714c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForPriorityError(env, ENOSYS, tid); 472cc767191cfb675f744e0165608b0a4196aba2b37Glenn Kasten#endif 4736793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten} 4746793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten 4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid android_os_Process_setThreadPriority(JNIEnv* env, jobject clazz, 4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint pid, jint pri) 4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 478160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#if GUARD_THREAD_PRIORITY 479160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate // if we're putting the current thread into the background, check the TLS 480160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate // to make sure this thread isn't guarded. If it is, raise an exception. 481160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate if (pri >= ANDROID_PRIORITY_BACKGROUND) { 48206451fe081d5ae79121a6f301475c7042f7f3a5dElliott Hughes if (pid == gettid()) { 483160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate void* bgOk = pthread_getspecific(gBgKey); 484160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate if (bgOk == ((void*)0xbaad)) { 4853762c311729fe9f3af085c14c5c1fb471d994c03Steve Block ALOGE("Thread marked fg-only put self in background!"); 486160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate jniThrowException(env, "java/lang/SecurityException", "May not put this thread into background"); 487160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate return; 488160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate } 489160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate } 490160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate } 491160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate#endif 492160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate 493887f355f99ff83d568ef2885a4fdcaae475583dfDianne Hackborn int rc = androidSetThreadPriority(pid, pri); 494887f355f99ff83d568ef2885a4fdcaae475583dfDianne Hackborn if (rc != 0) { 495887f355f99ff83d568ef2885a4fdcaae475583dfDianne Hackborn if (rc == INVALID_OPERATION) { 4964c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForPriorityError(env, errno, pid); 497887f355f99ff83d568ef2885a4fdcaae475583dfDianne Hackborn } else { 4984c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForGroupError(env, errno, pid); 499887f355f99ff83d568ef2885a4fdcaae475583dfDianne Hackborn } 5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 50169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 502c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn //ALOGI("Setting priority of %" PRId32 ": %" PRId32 ", getpriority returns %d\n", 5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // pid, pri, getpriority(PRIO_PROCESS, pid)); 5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid android_os_Process_setCallingThreadPriority(JNIEnv* env, jobject clazz, 5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint pri) 5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 50906451fe081d5ae79121a6f301475c7042f7f3a5dElliott Hughes android_os_Process_setThreadPriority(env, clazz, gettid(), pri); 5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectjint android_os_Process_getThreadPriority(JNIEnv* env, jobject clazz, 5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint pid) 5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project errno = 0; 5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint pri = getpriority(PRIO_PROCESS, pid); 5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (errno != 0) { 5184c0c4df7cfa0b4228ef547aa940674393cb2ab77Ruben Brunk signalExceptionForPriorityError(env, errno, pid); 5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 520c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn //ALOGI("Returning priority of %" PRId32 ": %" PRId32 "\n", pid, pri); 5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return pri; 5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5245534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchandjboolean android_os_Process_setSwappiness(JNIEnv *env, jobject clazz, 5255534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand jint pid, jboolean is_increased) 5265534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand{ 5275534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand char text[64]; 5285534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand 5295534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand if (is_increased) { 5305534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand strcpy(text, "/sys/fs/cgroup/memory/sw/tasks"); 5315534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand } else { 5325534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand strcpy(text, "/sys/fs/cgroup/memory/tasks"); 5335534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand } 5345534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand 5355534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand struct stat st; 5365534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand if (stat(text, &st) || !S_ISREG(st.st_mode)) { 5375534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand return false; 5385534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand } 5395534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand 5405534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand int fd = open(text, O_WRONLY); 5415534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand if (fd >= 0) { 542c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn sprintf(text, "%" PRId32, pid); 5435534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand write(fd, text, strlen(text)); 5445534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand close(fd); 5455534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand } 5465534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand 5475534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand return true; 5485534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand} 5495534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand 5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid android_os_Process_setArgV0(JNIEnv* env, jobject clazz, jstring name) 5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (name == NULL) { 55369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 55669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jchar* str = env->GetStringCritical(name, 0); 5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String8 name8; 5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (str) { 5606698749dd4d4d6513b26aa9071af290b956b68a7Dan Albert name8 = String8(reinterpret_cast<const char16_t*>(str), 5616698749dd4d4d6513b26aa9071af290b956b68a7Dan Albert env->GetStringLength(name)); 5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringCritical(name, str); 5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 565342c7dcd022746c25dc202230a6a9390c403062fDmitriy Filchenko if (!name8.isEmpty()) { 566f5b6e5590e0a2c1be10bc0bd0a3b141256fdf7fdDmitriy Filchenko AndroidRuntime::getRuntime()->setArgv0(name8.string(), true /* setProcName */); 5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectjint android_os_Process_setUid(JNIEnv* env, jobject clazz, jint uid) 5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return setuid(uid) == 0 ? 0 : errno; 5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectjint android_os_Process_setGid(JNIEnv* env, jobject clazz, jint uid) 5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return setgid(uid) == 0 ? 0 : errno; 5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic int pid_compare(const void* v1, const void* v2) 5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 582c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn //ALOGI("Compare %" PRId32 " vs %" PRId32 "\n", *((const jint*)v1), *((const jint*)v2)); 5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return *((const jint*)v1) - *((const jint*)v2); 5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 586c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughesstatic jlong getFreeMemoryImpl(const char* const sums[], const size_t sumsLen[], size_t num) 5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int fd = open("/proc/meminfo", O_RDONLY); 58969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fd < 0) { 5918564c8da817a845353d213acd8636b76f567b234Steve Block ALOGW("Unable to open /proc/meminfo"); 5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 59469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char buffer[256]; 5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const int len = read(fd, buffer, sizeof(buffer)-1); 5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project close(fd); 59869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (len < 0) { 6008564c8da817a845353d213acd8636b76f567b234Steve Block ALOGW("Unable to read /proc/meminfo"); 6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project buffer[len] = 0; 6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 605c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes size_t numFound = 0; 6060bca96bcbfe559f9330a01f723c5c9cba51ec05aMarco Nelissen jlong mem = 0; 60769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* p = buffer; 60959325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn while (*p && numFound < num) { 6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int i = 0; 6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (sums[i]) { 6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (strncmp(p, sums[i], sumsLen[i]) == 0) { 6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p += sumsLen[i]; 6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (*p == ' ') p++; 6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* num = p; 6169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (*p >= '0' && *p <= '9') p++; 6179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (*p != 0) { 6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *p = 0; 6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p++; 6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (*p == 0) p--; 6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6220bca96bcbfe559f9330a01f723c5c9cba51ec05aMarco Nelissen mem += atoll(num) * 1024; 6239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project numFound++; 6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 6279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p++; 6299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 63069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return numFound > 0 ? mem : -1; 6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 63459325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackbornstatic jlong android_os_Process_getFreeMemory(JNIEnv* env, jobject clazz) 63559325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn{ 63659325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn static const char* const sums[] = { "MemFree:", "Cached:", NULL }; 637c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes static const size_t sumsLen[] = { strlen("MemFree:"), strlen("Cached:"), 0 }; 63859325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn return getFreeMemoryImpl(sums, sumsLen, 2); 63959325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn} 64059325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn 64159325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackbornstatic jlong android_os_Process_getTotalMemory(JNIEnv* env, jobject clazz) 64259325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn{ 64359325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn static const char* const sums[] = { "MemTotal:", NULL }; 644c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes static const size_t sumsLen[] = { strlen("MemTotal:"), 0 }; 64559325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn return getFreeMemoryImpl(sums, sumsLen, 1); 64659325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn} 64759325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn 6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid android_os_Process_readProcLines(JNIEnv* env, jobject clazz, jstring fileStr, 6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jobjectArray reqFields, jlongArray outFields) 6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 6516215d3ff4b5dfa52a5d8b9a42e343051f31066a5Steve Block //ALOGI("getMemInfo: %p %p", reqFields, outFields); 65269a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fileStr == NULL || reqFields == NULL || outFields == NULL) { 65469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 65769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* file8 = env->GetStringUTFChars(fileStr, NULL); 6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (file8 == NULL) { 6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String8 file(file8); 6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringUTFChars(fileStr, file8); 66469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jsize count = env->GetArrayLength(reqFields); 6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count > env->GetArrayLength(outFields)) { 6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/lang/IllegalArgumentException", "Array lengths differ"); 6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 67069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Vector<String8> fields; 6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int i; 67369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i=0; i<count; i++) { 6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jobject obj = env->GetObjectArrayElement(reqFields, i); 6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (obj != NULL) { 6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* str8 = env->GetStringUTFChars((jstring)obj, NULL); 6786215d3ff4b5dfa52a5d8b9a42e343051f31066a5Steve Block //ALOGI("String at %d: %p = %s", i, obj, str8); 6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (str8 == NULL) { 68069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, "Element in reqFields"); 6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fields.add(String8(str8)); 6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringUTFChars((jstring)obj, str8); 6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 68669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, "Element in reqFields"); 6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 69069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jlong* sizesArray = env->GetLongArrayElements(outFields, 0); 6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sizesArray == NULL) { 6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 69569a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 696c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn //ALOGI("Clearing %" PRId32 " sizes", count); 6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i=0; i<count; i++) { 6989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sizesArray[i] = 0; 6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 70069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int fd = open(file.string(), O_RDONLY); 70269a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fd >= 0) { 7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const size_t BUFFER_SIZE = 2048; 7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* buffer = (char*)malloc(BUFFER_SIZE); 7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int len = read(fd, buffer, BUFFER_SIZE-1); 7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project close(fd); 70869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (len < 0) { 7108564c8da817a845353d213acd8636b76f567b234Steve Block ALOGW("Unable to read %s", file.string()); 7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project len = 0; 7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project buffer[len] = 0; 71469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int foundCount = 0; 71669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* p = buffer; 7189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (*p && foundCount < count) { 7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project bool skipToEol = true; 7206215d3ff4b5dfa52a5d8b9a42e343051f31066a5Steve Block //ALOGI("Parsing at: %s", p); 7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i=0; i<count; i++) { 7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const String8& field = fields[i]; 7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (strncmp(p, field.string(), field.length()) == 0) { 7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p += field.length(); 725add868cebaf62cffe96e79764ea0b7f2320a03ebAmith Yamasani while (*p == ' ' || *p == '\t') p++; 7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* num = p; 7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (*p >= '0' && *p <= '9') p++; 7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project skipToEol = *p != '\n'; 7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (*p != 0) { 7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *p = 0; 7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p++; 7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* end; 7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sizesArray[i] = strtoll(num, &end, 10); 735c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn //ALOGI("Field %s = %" PRId64, field.string(), sizesArray[i]); 7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project foundCount++; 7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (skipToEol) { 7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (*p && *p != '\n') { 7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p++; 7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (*p == '\n') { 7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p++; 7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 74969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project free(buffer); 7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 7528564c8da817a845353d213acd8636b76f567b234Steve Block ALOGW("Unable to open %s", file.string()); 7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 75469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7556215d3ff4b5dfa52a5d8b9a42e343051f31066a5Steve Block //ALOGI("Done!"); 7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseLongArrayElements(outFields, sizesArray, 0); 7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectjintArray android_os_Process_getPids(JNIEnv* env, jobject clazz, 7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jstring file, jintArray lastArray) 7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (file == NULL) { 76369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return NULL; 7659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 76669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* file8 = env->GetStringUTFChars(file, NULL); 7689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (file8 == NULL) { 7699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/lang/OutOfMemoryError", NULL); 7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return NULL; 7719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 77269a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project DIR* dirp = opendir(file8); 77469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringUTFChars(file, file8); 77669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if(dirp == NULL) { 7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return NULL; 7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 78069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jsize curCount = 0; 7829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint* curData = NULL; 7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (lastArray != NULL) { 7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curCount = env->GetArrayLength(lastArray); 7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curData = env->GetIntArrayElements(lastArray, 0); 7869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 78769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint curPos = 0; 78969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project struct dirent* entry; 7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while ((entry=readdir(dirp)) != NULL) { 7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* p = entry->d_name; 7939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (*p) { 7949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (*p < '0' || *p > '9') break; 7959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project p++; 7969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (*p != 0) continue; 79869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* end; 8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int pid = strtol(entry->d_name, &end, 10); 8016215d3ff4b5dfa52a5d8b9a42e343051f31066a5Steve Block //ALOGI("File %s pid=%d\n", entry->d_name, pid); 8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (curPos >= curCount) { 8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jsize newCount = (curCount == 0) ? 10 : (curCount*2); 8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jintArray newArray = env->NewIntArray(newCount); 8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (newArray == NULL) { 8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project closedir(dirp); 8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/lang/OutOfMemoryError", NULL); 8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return NULL; 8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint* newData = env->GetIntArrayElements(newArray, 0); 8119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (curData != NULL) { 8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project memcpy(newData, curData, sizeof(jint)*curCount); 8139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseIntArrayElements(lastArray, curData, 0); 8149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project lastArray = newArray; 8169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curCount = newCount; 8179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curData = newData; 8189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 81969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curData[curPos] = pid; 8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curPos++; 8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 82369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project closedir(dirp); 82569a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (curData != NULL && curPos > 0) { 8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project qsort(curData, curPos, sizeof(jint), pid_compare); 8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 82969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (curPos < curCount) { 8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curData[curPos] = -1; 8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project curPos++; 8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 83469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (curData != NULL) { 8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseIntArrayElements(lastArray, curData, 0); 8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 83869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return lastArray; 8409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 8419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectenum { 8439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_TERM_MASK = 0xff, 8449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_ZERO_TERM = 0, 8459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_SPACE_TERM = ' ', 8469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_COMBINE = 0x100, 8479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_PARENS = 0x200, 84813ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn PROC_QUOTES = 0x400, 8494de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn PROC_CHAR = 0x800, 8509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_OUT_STRING = 0x1000, 8519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_OUT_LONG = 0x2000, 8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project PROC_OUT_FLOAT = 0x4000, 8539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; 8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 855c64edde69d18498fb2954f71a546357b07ab996aEvan Millarjboolean android_os_Process_parseProcLineArray(JNIEnv* env, jobject clazz, 85669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes char* buffer, jint startIndex, jint endIndex, jintArray format, 857c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jobjectArray outStrings, jlongArray outLongs, jfloatArray outFloats) 8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 85969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jsize NF = env->GetArrayLength(format); 8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jsize NS = outStrings ? env->GetArrayLength(outStrings) : 0; 8629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jsize NL = outLongs ? env->GetArrayLength(outLongs) : 0; 8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jsize NR = outFloats ? env->GetArrayLength(outFloats) : 0; 86469a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint* formatData = env->GetIntArrayElements(format, 0); 8669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jlong* longsData = outLongs ? 8679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->GetLongArrayElements(outLongs, 0) : NULL; 8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jfloat* floatsData = outFloats ? 8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->GetFloatArrayElements(outFloats, 0) : NULL; 8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (formatData == NULL || (NL > 0 && longsData == NULL) 8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project || (NR > 0 && floatsData == NULL)) { 8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (formatData != NULL) { 8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseIntArrayElements(format, formatData, 0); 8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (longsData != NULL) { 8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseLongArrayElements(outLongs, longsData, 0); 8779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (floatsData != NULL) { 8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseFloatArrayElements(outFloats, floatsData, 0); 8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/lang/OutOfMemoryError", NULL); 8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return JNI_FALSE; 8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 885c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jsize i = startIndex; 8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jsize di = 0; 88769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jboolean res = JNI_TRUE; 88969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (jsize fi=0; fi<NF; fi++) { 89113ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn jint mode = formatData[fi]; 8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&PROC_PARENS) != 0) { 8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 8943b1e22e2898e27fe486ac4137692e0f20166e13eBernhard Rosenkränzer } else if ((mode&PROC_QUOTES) != 0) { 89513ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn if (buffer[i] == '"') { 89613ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn i++; 89713ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn } else { 89813ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn mode &= ~PROC_QUOTES; 89913ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn } 9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char term = (char)(mode&PROC_TERM_MASK); 9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const jsize start = i; 903c64edde69d18498fb2954f71a546357b07ab996aEvan Millar if (i >= endIndex) { 9040f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe if (kDebugProc) { 9050f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe ALOGW("Ran off end of data @%d", i); 9060f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe } 9079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project res = JNI_FALSE; 9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 9099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 91069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jsize end = -1; 9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&PROC_PARENS) != 0) { 913c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes while (i < endIndex && buffer[i] != ')') { 9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end = i; 9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 91813ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn } else if ((mode&PROC_QUOTES) != 0) { 91913ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn while (buffer[i] != '"' && i < endIndex) { 92013ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn i++; 92113ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn } 92213ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn end = i; 92313ac041b9f21043bc7c848a743be618bfd7a67e9Dianne Hackborn i++; 9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 925c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes while (i < endIndex && buffer[i] != term) { 9269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 9279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end < 0) { 9299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end = i; 9309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 93169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 932c64edde69d18498fb2954f71a546357b07ab996aEvan Millar if (i < endIndex) { 9339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 9349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&PROC_COMBINE) != 0) { 935c367d48c55e5a3fa0df14fd62889e4bb6b63cb01Elliott Hughes while (i < endIndex && buffer[i] == term) { 9369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project i++; 9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 94069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 941c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn //ALOGI("Field %" PRId32 ": %" PRId32 "-%" PRId32 " dest=%" PRId32 " mode=0x%" PRIx32 "\n", i, start, end, di, mode); 94269a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&(PROC_OUT_FLOAT|PROC_OUT_LONG|PROC_OUT_STRING)) != 0) { 9449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char c = buffer[end]; 9459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project buffer[end] = 0; 9469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&PROC_OUT_FLOAT) != 0 && di < NR) { 9479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char* end; 9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project floatsData[di] = strtof(buffer+start, &end); 9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&PROC_OUT_LONG) != 0 && di < NL) { 9514de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn if ((mode&PROC_CHAR) != 0) { 9524de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn // Caller wants single first character returned as one long. 9534de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn longsData[di] = buffer[start]; 9544de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn } else { 9554de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn char* end; 9564de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn longsData[di] = strtoll(buffer+start, &end, 10); 9574de5a3ac6655f76b67af38712ae5aeb6d7c15938Dianne Hackborn } 9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if ((mode&PROC_OUT_STRING) != 0 && di < NS) { 9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jstring str = env->NewStringUTF(buffer+start); 9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->SetObjectArrayElement(outStrings, di, str); 9629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project buffer[end] = c; 9649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project di++; 9659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 96769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseIntArrayElements(format, formatData, 0); 9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (longsData != NULL) { 9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseLongArrayElements(outLongs, longsData, 0); 9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (floatsData != NULL) { 9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseFloatArrayElements(outFloats, floatsData, 0); 9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 97569a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return res; 9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 979c64edde69d18498fb2954f71a546357b07ab996aEvan Millarjboolean android_os_Process_parseProcLine(JNIEnv* env, jobject clazz, 98069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jbyteArray buffer, jint startIndex, jint endIndex, jintArray format, 981c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jobjectArray outStrings, jlongArray outLongs, jfloatArray outFloats) 982c64edde69d18498fb2954f71a546357b07ab996aEvan Millar{ 983c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jbyte* bufferArray = env->GetByteArrayElements(buffer, NULL); 984c64edde69d18498fb2954f71a546357b07ab996aEvan Millar 98569a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jboolean result = android_os_Process_parseProcLineArray(env, clazz, 98669a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes (char*) bufferArray, startIndex, endIndex, format, outStrings, 987c64edde69d18498fb2954f71a546357b07ab996aEvan Millar outLongs, outFloats); 98869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 989c64edde69d18498fb2954f71a546357b07ab996aEvan Millar env->ReleaseByteArrayElements(buffer, bufferArray, 0); 99069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 991c64edde69d18498fb2954f71a546357b07ab996aEvan Millar return result; 992c64edde69d18498fb2954f71a546357b07ab996aEvan Millar} 993c64edde69d18498fb2954f71a546357b07ab996aEvan Millar 994c64edde69d18498fb2954f71a546357b07ab996aEvan Millarjboolean android_os_Process_readProcFile(JNIEnv* env, jobject clazz, 995c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jstring file, jintArray format, jobjectArray outStrings, 996c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jlongArray outLongs, jfloatArray outFloats) 997c64edde69d18498fb2954f71a546357b07ab996aEvan Millar{ 998c64edde69d18498fb2954f71a546357b07ab996aEvan Millar if (file == NULL || format == NULL) { 99969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 1000c64edde69d18498fb2954f71a546357b07ab996aEvan Millar return JNI_FALSE; 1001c64edde69d18498fb2954f71a546357b07ab996aEvan Millar } 1002c64edde69d18498fb2954f71a546357b07ab996aEvan Millar 1003c64edde69d18498fb2954f71a546357b07ab996aEvan Millar const char* file8 = env->GetStringUTFChars(file, NULL); 1004c64edde69d18498fb2954f71a546357b07ab996aEvan Millar if (file8 == NULL) { 1005c64edde69d18498fb2954f71a546357b07ab996aEvan Millar jniThrowException(env, "java/lang/OutOfMemoryError", NULL); 1006c64edde69d18498fb2954f71a546357b07ab996aEvan Millar return JNI_FALSE; 1007c64edde69d18498fb2954f71a546357b07ab996aEvan Millar } 1008c64edde69d18498fb2954f71a546357b07ab996aEvan Millar int fd = open(file8, O_RDONLY); 100969a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 1010c64edde69d18498fb2954f71a546357b07ab996aEvan Millar if (fd < 0) { 10110f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe if (kDebugProc) { 10120f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe ALOGW("Unable to open process file: %s\n", file8); 10130f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe } 1014306af678a5f1938629e1182b46cc43f2da6e7774Dianne Hackborn env->ReleaseStringUTFChars(file, file8); 1015c64edde69d18498fb2954f71a546357b07ab996aEvan Millar return JNI_FALSE; 1016c64edde69d18498fb2954f71a546357b07ab996aEvan Millar } 1017306af678a5f1938629e1182b46cc43f2da6e7774Dianne Hackborn env->ReleaseStringUTFChars(file, file8); 101869a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 1019c64edde69d18498fb2954f71a546357b07ab996aEvan Millar char buffer[256]; 1020c64edde69d18498fb2954f71a546357b07ab996aEvan Millar const int len = read(fd, buffer, sizeof(buffer)-1); 1021c64edde69d18498fb2954f71a546357b07ab996aEvan Millar close(fd); 102269a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 1023c64edde69d18498fb2954f71a546357b07ab996aEvan Millar if (len < 0) { 10240f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe if (kDebugProc) { 10250f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe ALOGW("Unable to open process file: %s fd=%d\n", file8, fd); 10260f0b4919667f418b249c497f5ad3e83fdf4437e5Andreas Gampe } 1027c64edde69d18498fb2954f71a546357b07ab996aEvan Millar return JNI_FALSE; 1028c64edde69d18498fb2954f71a546357b07ab996aEvan Millar } 1029c64edde69d18498fb2954f71a546357b07ab996aEvan Millar buffer[len] = 0; 103069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 103169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes return android_os_Process_parseProcLineArray(env, clazz, buffer, 0, len, 1032c64edde69d18498fb2954f71a546357b07ab996aEvan Millar format, outStrings, outLongs, outFloats); 103369a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 1034c64edde69d18498fb2954f71a546357b07ab996aEvan Millar} 1035c64edde69d18498fb2954f71a546357b07ab996aEvan Millar 10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid android_os_Process_setApplicationObject(JNIEnv* env, jobject clazz, 10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jobject binderObject) 10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (binderObject == NULL) { 104069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes jniThrowNullPointerException(env, NULL); 10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sp<IBinder> binder = ibinderForJavaObject(env, binderObject); 10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid android_os_Process_sendSignal(JNIEnv* env, jobject clazz, jint pid, jint sig) 10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (pid > 0) { 1050c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn ALOGI("Sending signal. PID: %" PRId32 " SIG: %" PRId32, pid, sig); 10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project kill(pid, sig); 10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1055906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackbornvoid android_os_Process_sendSignalQuiet(JNIEnv* env, jobject clazz, jint pid, jint sig) 1056906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn{ 1057906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn if (pid > 0) { 1058906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn kill(pid, sig); 1059906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn } 1060906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn} 1061906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn 10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jlong android_os_Process_getElapsedCpuTime(JNIEnv* env, jobject clazz) 10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project struct timespec ts; 10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int res = clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &ts); 106769a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 10689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (res != 0) { 10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return (jlong) 0; 107069a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes } 107169a017bc1d1649350f830dfada5c6ed5eac0b770Elliott Hughes 10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project nsecs_t when = seconds_to_nanoseconds(ts.tv_sec) + ts.tv_nsec; 10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return (jlong) nanoseconds_to_milliseconds(when); 10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jlong android_os_Process_getPss(JNIEnv* env, jobject clazz, jint pid) 10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char filename[64]; 10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1080c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn snprintf(filename, sizeof(filename), "/proc/%" PRId32 "/smaps", pid); 10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project FILE * file = fopen(filename, "r"); 10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!file) { 10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return (jlong) -1; 10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Tally up all of the Pss from the various maps 10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char line[256]; 10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jlong pss = 0; 10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (fgets(line, sizeof(line), file)) { 10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jlong v; 1092c6a410164e1fd728cf7873493baacca7bc29548dMark Salyzyn if (sscanf(line, "Pss: %" SCNd64 " kB", &v) == 1) { 10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project pss += v; 10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fclose(file); 10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Return the Pss value in bytes, not kilobytes 11009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return pss * 1024; 11019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 11029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1103f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne HackbornjintArray android_os_Process_getPidsForCommands(JNIEnv* env, jobject clazz, 1104f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jobjectArray commandNames) 1105f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn{ 1106f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (commandNames == NULL) { 1107f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jniThrowNullPointerException(env, NULL); 1108f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn return NULL; 1109f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1110f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1111f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn Vector<String8> commands; 1112f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1113f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jsize count = env->GetArrayLength(commandNames); 1114f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1115f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn for (int i=0; i<count; i++) { 1116f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jobject obj = env->GetObjectArrayElement(commandNames, i); 1117f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (obj != NULL) { 1118f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn const char* str8 = env->GetStringUTFChars((jstring)obj, NULL); 1119f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (str8 == NULL) { 1120f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jniThrowNullPointerException(env, "Element in commandNames"); 1121f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn return NULL; 1122f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1123f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn commands.add(String8(str8)); 1124f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn env->ReleaseStringUTFChars((jstring)obj, str8); 1125f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } else { 1126f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jniThrowNullPointerException(env, "Element in commandNames"); 1127f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn return NULL; 1128f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1129f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1130f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1131f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn Vector<jint> pids; 1132f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1133f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn DIR *proc = opendir("/proc"); 1134f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (proc == NULL) { 1135f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn fprintf(stderr, "/proc: %s\n", strerror(errno)); 1136f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn return NULL; 1137f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1138f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1139f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn struct dirent *d; 1140f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn while ((d = readdir(proc))) { 1141f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn int pid = atoi(d->d_name); 1142f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (pid <= 0) continue; 1143f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1144f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn char path[PATH_MAX]; 1145f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn char data[PATH_MAX]; 1146f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn snprintf(path, sizeof(path), "/proc/%d/cmdline", pid); 1147f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1148f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn int fd = open(path, O_RDONLY); 1149f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (fd < 0) { 1150f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn continue; 1151f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1152f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn const int len = read(fd, data, sizeof(data)-1); 1153f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn close(fd); 1154f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1155f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (len < 0) { 1156f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn continue; 1157f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1158f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn data[len] = 0; 1159f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1160f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn for (int i=0; i<len; i++) { 1161f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (data[i] == ' ') { 1162f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn data[i] = 0; 1163f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn break; 1164f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1165f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1166f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1167f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn for (size_t i=0; i<commands.size(); i++) { 1168f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (commands[i] == data) { 1169f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn pids.add(pid); 1170f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn break; 1171f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1172f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1173f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1174f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1175f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn closedir(proc); 1176f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1177f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jintArray pidArray = env->NewIntArray(pids.size()); 1178f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (pidArray == NULL) { 1179f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn jniThrowException(env, "java/lang/OutOfMemoryError", NULL); 1180f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn return NULL; 1181f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1182f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1183f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn if (pids.size() > 0) { 1184f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn env->SetIntArrayRegion(pidArray, 0, pids.size(), pids.array()); 1185f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn } 1186f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 1187f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn return pidArray; 1188f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn} 1189f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn 11900769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Crossjint android_os_Process_killProcessGroup(JNIEnv* env, jobject clazz, jint uid, jint pid) 11910769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross{ 11920769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross return killProcessGroup(uid, pid, SIGKILL); 11930769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross} 11940769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross 11950769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Crossvoid android_os_Process_removeAllProcessGroups(JNIEnv* env, jobject clazz) 11960769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross{ 11970769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross return removeAllProcessGroups(); 11980769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross} 11990769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross 12009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic const JNINativeMethod methods[] = { 12019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"getUidForName", "(Ljava/lang/String;)I", (void*)android_os_Process_getUidForName}, 12029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"getGidForName", "(Ljava/lang/String;)I", (void*)android_os_Process_getGidForName}, 12039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"setThreadPriority", "(II)V", (void*)android_os_Process_setThreadPriority}, 12046793ac943afeb16642f477c43ddfd27e498db37bGlenn Kasten {"setThreadScheduler", "(III)V", (void*)android_os_Process_setThreadScheduler}, 1205160edb3645f8b7012bab70ae6e6e8c4a5733082bChristopher Tate {"setCanSelfBackground", "(Z)V", (void*)android_os_Process_setCanSelfBackground}, 12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"setThreadPriority", "(I)V", (void*)android_os_Process_setCallingThreadPriority}, 12079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"getThreadPriority", "(I)I", (void*)android_os_Process_getThreadPriority}, 12081b15d13243d7e7f672713f21f0c46d88c3c067a2Srinath Sridharan {"getThreadScheduler", "(I)I", (void*)android_os_Process_getThreadScheduler}, 1209e9d376b801b7890b1ef5006ed55de4208e64bb63San Mehat {"setThreadGroup", "(II)V", (void*)android_os_Process_setThreadGroup}, 12109e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey {"setProcessGroup", "(II)V", (void*)android_os_Process_setProcessGroup}, 12119e57c414f39e1a31349bc215635fdcfaf1902ceeJeff Sharkey {"getProcessGroup", "(I)I", (void*)android_os_Process_getProcessGroup}, 1212cd4bdf3eb91c1cc2757d32a86fc90745d0dff990Martijn Coenen {"getExclusiveCores", "()[I", (void*)android_os_Process_getExclusiveCores}, 12135534ba91ac0a0c9af822af62bcf92e2c5a8d6ec8Rom Lemarchand {"setSwappiness", "(IZ)Z", (void*)android_os_Process_setSwappiness}, 12149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"setArgV0", "(Ljava/lang/String;)V", (void*)android_os_Process_setArgV0}, 12159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"setUid", "(I)I", (void*)android_os_Process_setUid}, 12169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"setGid", "(I)I", (void*)android_os_Process_setGid}, 12179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"sendSignal", "(II)V", (void*)android_os_Process_sendSignal}, 1218906497c574d45d8dfd295b16dece0d0bc32c0895Dianne Hackborn {"sendSignalQuiet", "(II)V", (void*)android_os_Process_sendSignalQuiet}, 12190bca96bcbfe559f9330a01f723c5c9cba51ec05aMarco Nelissen {"getFreeMemory", "()J", (void*)android_os_Process_getFreeMemory}, 122059325eb31f25704bb88c348160bb69e7c1aa3b48Dianne Hackborn {"getTotalMemory", "()J", (void*)android_os_Process_getTotalMemory}, 12219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"readProcLines", "(Ljava/lang/String;[Ljava/lang/String;[J)V", (void*)android_os_Process_readProcLines}, 12229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"getPids", "(Ljava/lang/String;[I)[I", (void*)android_os_Process_getPids}, 12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"readProcFile", "(Ljava/lang/String;[I[Ljava/lang/String;[J[F)Z", (void*)android_os_Process_readProcFile}, 1224c64edde69d18498fb2954f71a546357b07ab996aEvan Millar {"parseProcLine", "([BII[I[Ljava/lang/String;[J[F)Z", (void*)android_os_Process_parseProcLine}, 12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"getElapsedCpuTime", "()J", (void*)android_os_Process_getElapsedCpuTime}, 12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project {"getPss", "(I)J", (void*)android_os_Process_getPss}, 1227f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn {"getPidsForCommands", "([Ljava/lang/String;)[I", (void*)android_os_Process_getPidsForCommands}, 12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //{"setApplicationObject", "(Landroid/os/IBinder;)V", (void*)android_os_Process_setApplicationObject}, 12290769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross {"killProcessGroup", "(II)I", (void*)android_os_Process_killProcessGroup}, 12300769e550011d8f8a19e333efe1706ef0e6cc6a5fColin Cross {"removeAllProcessGroups", "()V", (void*)android_os_Process_removeAllProcessGroups}, 12319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; 12329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectint register_android_os_Process(JNIEnv* env) 12349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 1235ed6b9dff563c5e22f040ff37e12c0d771e0478aeAndreas Gampe return RegisterMethodsOrDie(env, "android/os/Process", methods, NELEM(methods)); 12369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1237