1 /* 2 * Copyright 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #define LOG_TAG "AAudio" 18 //#define LOG_NDEBUG 0 19 #include <utils/Log.h> 20 21 #include <stdint.h> 22 #include <stdio.h> 23 24 #include <sys/mman.h> 25 #include <aaudio/AAudio.h> 26 27 #include <binder/Parcelable.h> 28 #include <utility/AAudioUtilities.h> 29 30 #include "binding/SharedMemoryParcelable.h" 31 32 using android::NO_ERROR; 33 using android::status_t; 34 using android::Parcel; 35 using android::Parcelable; 36 37 using namespace aaudio; 38 39 SharedMemoryParcelable::SharedMemoryParcelable() {} 40 SharedMemoryParcelable::~SharedMemoryParcelable() {}; 41 42 void SharedMemoryParcelable::setup(int fd, int32_t sizeInBytes) { 43 mFd = fd; 44 mSizeInBytes = sizeInBytes; 45 46 } 47 48 status_t SharedMemoryParcelable::writeToParcel(Parcel* parcel) const { 49 status_t status = parcel->writeInt32(mSizeInBytes); 50 if (status != NO_ERROR) return status; 51 if (mSizeInBytes > 0) { 52 status = parcel->writeDupFileDescriptor(mFd); 53 ALOGE_IF(status != NO_ERROR, "SharedMemoryParcelable writeDupFileDescriptor failed : %d", 54 status); 55 } 56 return status; 57 } 58 59 status_t SharedMemoryParcelable::readFromParcel(const Parcel* parcel) { 60 status_t status = parcel->readInt32(&mSizeInBytes); 61 if (status != NO_ERROR) { 62 return status; 63 } 64 if (mSizeInBytes > 0) { 65 // Keep the original FD until you are done with the mFd. 66 // If you close it in here then it will prevent mFd from working. 67 mOriginalFd = parcel->readFileDescriptor(); 68 ALOGV("SharedMemoryParcelable::readFromParcel() LEAK? mOriginalFd = %d\n", mOriginalFd); 69 mFd = fcntl(mOriginalFd, F_DUPFD_CLOEXEC, 0); 70 ALOGV("SharedMemoryParcelable::readFromParcel() LEAK? mFd = %d\n", mFd); 71 if (mFd == -1) { 72 status = -errno; 73 ALOGE("SharedMemoryParcelable readFromParcel fcntl() failed : %d", status); 74 } 75 } 76 return status; 77 } 78 79 aaudio_result_t SharedMemoryParcelable::close() { 80 if (mResolvedAddress != MMAP_UNRESOLVED_ADDRESS) { 81 int err = munmap(mResolvedAddress, mSizeInBytes); 82 if (err < 0) { 83 ALOGE("SharedMemoryParcelable::close() munmap() failed %d", err); 84 return AAudioConvert_androidToAAudioResult(err); 85 } 86 mResolvedAddress = MMAP_UNRESOLVED_ADDRESS; 87 } 88 if (mFd != -1) { 89 ALOGV("SharedMemoryParcelable::close() LEAK? mFd = %d\n", mFd); 90 ::close(mFd); 91 mFd = -1; 92 } 93 if (mOriginalFd != -1) { 94 ALOGV("SharedMemoryParcelable::close() LEAK? mOriginalFd = %d\n", mOriginalFd); 95 ::close(mOriginalFd); 96 mOriginalFd = -1; 97 } 98 return AAUDIO_OK; 99 } 100 101 aaudio_result_t SharedMemoryParcelable::resolve(int32_t offsetInBytes, int32_t sizeInBytes, 102 void **regionAddressPtr) { 103 104 if (offsetInBytes < 0) { 105 ALOGE("SharedMemoryParcelable illegal offsetInBytes = %d", offsetInBytes); 106 return AAUDIO_ERROR_OUT_OF_RANGE; 107 } else if ((offsetInBytes + sizeInBytes) > mSizeInBytes) { 108 ALOGE("SharedMemoryParcelable out of range, offsetInBytes = %d, " 109 "sizeInBytes = %d, mSizeInBytes = %d", 110 offsetInBytes, sizeInBytes, mSizeInBytes); 111 return AAUDIO_ERROR_OUT_OF_RANGE; 112 } 113 if (mResolvedAddress == MMAP_UNRESOLVED_ADDRESS) { 114 mResolvedAddress = (uint8_t *) mmap(0, mSizeInBytes, PROT_READ|PROT_WRITE, 115 MAP_SHARED, mFd, 0); 116 if (mResolvedAddress == MMAP_UNRESOLVED_ADDRESS) { 117 ALOGE("SharedMemoryParcelable mmap failed for fd = %d, errno = %s", 118 mFd, strerror(errno)); 119 return AAUDIO_ERROR_INTERNAL; 120 } 121 } 122 *regionAddressPtr = mResolvedAddress + offsetInBytes; 123 ALOGV("SharedMemoryParcelable mResolvedAddress = %p", mResolvedAddress); 124 ALOGV("SharedMemoryParcelable offset by %d, *regionAddressPtr = %p", 125 offsetInBytes, *regionAddressPtr); 126 return AAUDIO_OK; 127 } 128 129 int32_t SharedMemoryParcelable::getSizeInBytes() { 130 return mSizeInBytes; 131 } 132 133 aaudio_result_t SharedMemoryParcelable::validate() { 134 if (mSizeInBytes < 0 || mSizeInBytes >= MAX_MMAP_SIZE_BYTES) { 135 ALOGE("SharedMemoryParcelable invalid mSizeInBytes = %d", mSizeInBytes); 136 return AAUDIO_ERROR_OUT_OF_RANGE; 137 } 138 if (mSizeInBytes > 0) { 139 if (mFd == -1) { 140 ALOGE("SharedMemoryParcelable uninitialized mFd = %d", mFd); 141 return AAUDIO_ERROR_INTERNAL; 142 } 143 } 144 return AAUDIO_OK; 145 } 146 147 void SharedMemoryParcelable::dump() { 148 ALOGD("SharedMemoryParcelable mFd = %d", mFd); 149 ALOGD("SharedMemoryParcelable mSizeInBytes = %d", mSizeInBytes); 150 ALOGD("SharedMemoryParcelable mResolvedAddress = %p", mResolvedAddress); 151 } 152