19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2008 The Android Open Source Project 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License"); 59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License. 69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at 79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * http://www.apache.org/licenses/LICENSE-2.0 99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software 119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS, 129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and 149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License. 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#define LOG_TAG "MemoryFile" 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/Log.h> 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <cutils/ashmem.h> 219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <android_runtime/AndroidRuntime.h> 229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "JNIHelp.h" 239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <unistd.h> 249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/mman.h> 259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android { 289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 29761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringertstatic jobject android_os_MemoryFile_open(JNIEnv* env, jobject clazz, jstring name, jint length) 309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* namestr = (name ? env->GetStringUTFChars(name, NULL) : NULL); 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int result = ashmem_create_region(namestr, length); 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (name) 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project env->ReleaseStringUTFChars(name, namestr); 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 38761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert if (result < 0) { 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/io/IOException", "ashmem_create_region failed"); 40963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert return NULL; 41761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert } 42761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert 43761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert return jniCreateFileDescriptor(env, result); 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 46761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringertstatic jint android_os_MemoryFile_mmap(JNIEnv* env, jobject clazz, jobject fileDescriptor, 47963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert jint length, jint prot) 489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 49761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert int fd = jniGetFDFromFileDescriptor(env, fileDescriptor); 50963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert jint result = (jint)mmap(NULL, length, prot, MAP_SHARED, fd, 0); 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!result) 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/io/IOException", "mmap failed"); 539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return result; 549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 56761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringertstatic void android_os_MemoryFile_munmap(JNIEnv* env, jobject clazz, jint addr, jint length) 579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 58761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert int result = munmap((void *)addr, length); 59761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert if (result < 0) 60761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert jniThrowException(env, "java/io/IOException", "munmap failed"); 61761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert} 62761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert 63761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringertstatic void android_os_MemoryFile_close(JNIEnv* env, jobject clazz, jobject fileDescriptor) 64761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert{ 65761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert int fd = jniGetFDFromFileDescriptor(env, fileDescriptor); 66761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert if (fd >= 0) { 67761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert jniSetFileDescriptorOfFD(env, fileDescriptor, -1); 68761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert close(fd); 69761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert } 709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jint android_os_MemoryFile_read(JNIEnv* env, jobject clazz, 73761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert jobject fileDescriptor, jint address, jbyteArray buffer, jint srcOffset, jint destOffset, 749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint count, jboolean unpinned) 759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 76761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert int fd = jniGetFDFromFileDescriptor(env, fileDescriptor); 779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unpinned && ashmem_pin_region(fd, 0, 0) == ASHMEM_WAS_PURGED) { 789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ashmem_unpin_region(fd, 0, 0); 799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/io/IOException", "ashmem region was purged"); 809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 83ad984f19339008f7eb8687aecc1779be70bc1f65Bjorn Bringert env->SetByteArrayRegion(buffer, destOffset, count, (const jbyte *)address + srcOffset); 849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unpinned) { 869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ashmem_unpin_region(fd, 0, 0); 879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return count; 899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jint android_os_MemoryFile_write(JNIEnv* env, jobject clazz, 92761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert jobject fileDescriptor, jint address, jbyteArray buffer, jint srcOffset, jint destOffset, 939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jint count, jboolean unpinned) 949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 95761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert int fd = jniGetFDFromFileDescriptor(env, fileDescriptor); 969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unpinned && ashmem_pin_region(fd, 0, 0) == ASHMEM_WAS_PURGED) { 979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ashmem_unpin_region(fd, 0, 0); 989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/io/IOException", "ashmem region was purged"); 999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 102ad984f19339008f7eb8687aecc1779be70bc1f65Bjorn Bringert env->GetByteArrayRegion(buffer, srcOffset, count, (jbyte *)address + destOffset); 1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unpinned) { 1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ashmem_unpin_region(fd, 0, 0); 1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return count; 1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 110761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringertstatic void android_os_MemoryFile_pin(JNIEnv* env, jobject clazz, jobject fileDescriptor, jboolean pin) 1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 112761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert int fd = jniGetFDFromFileDescriptor(env, fileDescriptor); 1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int result = (pin ? ashmem_pin_region(fd, 0, 0) : ashmem_unpin_region(fd, 0, 0)); 1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (result < 0) { 1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jniThrowException(env, "java/io/IOException", NULL); 1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1197bcbd511731e13b9f2778e6aa6c633417d266f5eMarco Nelissenstatic jint android_os_MemoryFile_get_size(JNIEnv* env, jobject clazz, 120963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert jobject fileDescriptor) { 121963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert int fd = jniGetFDFromFileDescriptor(env, fileDescriptor); 122963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert // Use ASHMEM_GET_SIZE to find out if the fd refers to an ashmem region. 123963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert // ASHMEM_GET_SIZE should succeed for all ashmem regions, and the kernel 124963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert // should return ENOTTY for all other valid file descriptors 125963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert int result = ashmem_get_size_region(fd); 126963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert if (result < 0) { 127963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert if (errno == ENOTTY) { 128963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert // ENOTTY means that the ioctl does not apply to this object, 129963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert // i.e., it is not an ashmem region. 130ec100900e63a8374ac010e7131d9c7e54c5e6984Marco Nelissen return (jint) -1; 131963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert } 132963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert // Some other error, throw exception 133963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert jniThrowIOException(env, errno); 134ec100900e63a8374ac010e7131d9c7e54c5e6984Marco Nelissen return (jint) -1; 135963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert } 136ec100900e63a8374ac010e7131d9c7e54c5e6984Marco Nelissen return (jint) result; 137963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert} 138963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert 1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic const JNINativeMethod methods[] = { 140761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert {"native_open", "(Ljava/lang/String;I)Ljava/io/FileDescriptor;", (void*)android_os_MemoryFile_open}, 141963cd006c45716b034f656bf7e7179e6476f7e4dBjorn Bringert {"native_mmap", "(Ljava/io/FileDescriptor;II)I", (void*)android_os_MemoryFile_mmap}, 142761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert {"native_munmap", "(II)V", (void*)android_os_MemoryFile_munmap}, 143761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert {"native_close", "(Ljava/io/FileDescriptor;)V", (void*)android_os_MemoryFile_close}, 144761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert {"native_read", "(Ljava/io/FileDescriptor;I[BIIIZ)I", (void*)android_os_MemoryFile_read}, 145761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert {"native_write", "(Ljava/io/FileDescriptor;I[BIIIZ)V", (void*)android_os_MemoryFile_write}, 146761e0918d30b6a3f292625b44b86dffd1538bc78Bjorn Bringert {"native_pin", "(Ljava/io/FileDescriptor;Z)V", (void*)android_os_MemoryFile_pin}, 1477bcbd511731e13b9f2778e6aa6c633417d266f5eMarco Nelissen {"native_get_size", "(Ljava/io/FileDescriptor;)I", 1487bcbd511731e13b9f2778e6aa6c633417d266f5eMarco Nelissen (void*)android_os_MemoryFile_get_size} 1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; 1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectint register_android_os_MemoryFile(JNIEnv* env) 1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return AndroidRuntime::registerNativeMethods( 154dd66bcbf9d6ef0c50a18d9c4b1b39ce7ef7afcc4Elliott Hughes env, "android/os/MemoryFile", 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methods, NELEM(methods)); 1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 159