17f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong/*
27f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * Copyright (C) 2016 The Android Open Source Project
37f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong *
47f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * Licensed under the Apache License, Version 2.0 (the "License");
57f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * you may not use this file except in compliance with the License.
67f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * You may obtain a copy of the License at
77f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong *
87f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong *      http://www.apache.org/licenses/LICENSE-2.0
97f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong *
107f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * Unless required by applicable law or agreed to in writing, software
117f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * distributed under the License is distributed on an "AS IS" BASIS,
127f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
137f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * See the License for the specific language governing permissions and
147f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong * limitations under the License.
157f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong */
167f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
177f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong#define LOG_TAG "HidlSupport"
187f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
197f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong#include <hidl/HidlBinderSupport.h>
207f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
21777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong// C includes
22777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong#include <unistd.h>
23777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong
24777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong// C++ includes
25777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong#include <fstream>
26777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong#include <sstream>
27777bef976bf218a2290b86d52c45dcff84a2f698Yifan Hong
287f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongnamespace android {
297f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongnamespace hardware {
307f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
31108d09d4eae07cd1d70a76187a929dc61548a229Steven Morelandhidl_binder_death_recipient::hidl_binder_death_recipient(const sp<hidl_death_recipient> &recipient,
32108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland        uint64_t cookie, const sp<::android::hidl::base::V1_0::IBase> &base) :
33108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland    mRecipient(recipient), mCookie(cookie), mBase(base) {
34108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland}
35108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland
36108d09d4eae07cd1d70a76187a929dc61548a229Steven Morelandvoid hidl_binder_death_recipient::binderDied(const wp<IBinder>& /*who*/) {
37108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland    sp<hidl_death_recipient> recipient = mRecipient.promote();
3825508806b87fc4d428590f06e0b6d361a254355cMartijn Coenen    if (recipient != nullptr && mBase != nullptr) {
39108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland        recipient->serviceDied(mCookie, mBase);
40108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland    }
4125508806b87fc4d428590f06e0b6d361a254355cMartijn Coenen    mBase = nullptr;
42108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland}
43108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland
44108d09d4eae07cd1d70a76187a929dc61548a229Steven Morelandwp<hidl_death_recipient> hidl_binder_death_recipient::getRecipient() {
45108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland    return mRecipient;
46108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland}
47108d09d4eae07cd1d70a76187a929dc61548a229Steven Moreland
483079100878a2595be644d866e67c12a9de620fd5Martijn Coenenconst size_t hidl_memory::kOffsetOfHandle = offsetof(hidl_memory, mHandle);
493079100878a2595be644d866e67c12a9de620fd5Martijn Coenenconst size_t hidl_memory::kOffsetOfName = offsetof(hidl_memory, mName);
506d4acb12519597610322097f86850d9e404d251aHridya Valsarajustatic_assert(hidl_memory::kOffsetOfHandle == 0, "wrong offset");
516d4acb12519597610322097f86850d9e404d251aHridya Valsarajustatic_assert(hidl_memory::kOffsetOfName == 24, "wrong offset");
523079100878a2595be644d866e67c12a9de620fd5Martijn Coenen
539d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenenstatus_t readEmbeddedFromParcel(const hidl_memory& memory,
543079100878a2595be644d866e67c12a9de620fd5Martijn Coenen        const Parcel &parcel, size_t parentHandle, size_t parentOffset) {
551f535a27054e7553e968cac664981073f0074009Steven Moreland    const native_handle_t *handle;
561f535a27054e7553e968cac664981073f0074009Steven Moreland    ::android::status_t _hidl_err = parcel.readNullableEmbeddedNativeHandle(
573079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            parentHandle,
581f535a27054e7553e968cac664981073f0074009Steven Moreland            parentOffset + hidl_memory::kOffsetOfHandle,
591f535a27054e7553e968cac664981073f0074009Steven Moreland            &handle);
603079100878a2595be644d866e67c12a9de620fd5Martijn Coenen
611f535a27054e7553e968cac664981073f0074009Steven Moreland    if (_hidl_err == ::android::OK) {
621f535a27054e7553e968cac664981073f0074009Steven Moreland        _hidl_err = readEmbeddedFromParcel(
639d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen                memory.name(),
641f535a27054e7553e968cac664981073f0074009Steven Moreland                parcel,
651f535a27054e7553e968cac664981073f0074009Steven Moreland                parentHandle,
661f535a27054e7553e968cac664981073f0074009Steven Moreland                parentOffset + hidl_memory::kOffsetOfName);
673079100878a2595be644d866e67c12a9de620fd5Martijn Coenen    }
683079100878a2595be644d866e67c12a9de620fd5Martijn Coenen
693079100878a2595be644d866e67c12a9de620fd5Martijn Coenen    return _hidl_err;
703079100878a2595be644d866e67c12a9de620fd5Martijn Coenen}
713079100878a2595be644d866e67c12a9de620fd5Martijn Coenen
723079100878a2595be644d866e67c12a9de620fd5Martijn Coenenstatus_t writeEmbeddedToParcel(const hidl_memory &memory,
733079100878a2595be644d866e67c12a9de620fd5Martijn Coenen        Parcel *parcel, size_t parentHandle, size_t parentOffset) {
743079100878a2595be644d866e67c12a9de620fd5Martijn Coenen    status_t _hidl_err = parcel->writeEmbeddedNativeHandle(
753079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            memory.handle(),
763079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            parentHandle,
773079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            parentOffset + hidl_memory::kOffsetOfHandle);
783079100878a2595be644d866e67c12a9de620fd5Martijn Coenen
793079100878a2595be644d866e67c12a9de620fd5Martijn Coenen    if (_hidl_err == ::android::OK) {
803079100878a2595be644d866e67c12a9de620fd5Martijn Coenen        _hidl_err = writeEmbeddedToParcel(
813079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            memory.name(),
823079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            parcel,
833079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            parentHandle,
843079100878a2595be644d866e67c12a9de620fd5Martijn Coenen            parentOffset + hidl_memory::kOffsetOfName);
853079100878a2595be644d866e67c12a9de620fd5Martijn Coenen    }
863079100878a2595be644d866e67c12a9de620fd5Martijn Coenen
873079100878a2595be644d866e67c12a9de620fd5Martijn Coenen    return _hidl_err;
883079100878a2595be644d866e67c12a9de620fd5Martijn Coenen}
897f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongconst size_t hidl_string::kOffsetOfBuffer = offsetof(hidl_string, mBuffer);
906d4acb12519597610322097f86850d9e404d251aHridya Valsarajustatic_assert(hidl_string::kOffsetOfBuffer == 0, "wrong offset");
917f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
929d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenenstatus_t readEmbeddedFromParcel(const hidl_string &string ,
937f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        const Parcel &parcel, size_t parentHandle, size_t parentOffset) {
941f535a27054e7553e968cac664981073f0074009Steven Moreland    const void *out;
959d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen
969d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    status_t status = parcel.readEmbeddedBuffer(
979d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen            string.size() + 1,
987f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            nullptr /* buffer_handle */,
997f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            parentHandle,
1001f535a27054e7553e968cac664981073f0074009Steven Moreland            parentOffset + hidl_string::kOffsetOfBuffer,
1011f535a27054e7553e968cac664981073f0074009Steven Moreland            &out);
1029d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen
1039d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    if (status != OK) {
1049d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen        return status;
1059d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    }
1069d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen
1079d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    // Always safe to access out[string.size()] because we read size+1 bytes
1089d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    if (static_cast<const char *>(out)[string.size()] != '\0') {
1099d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen        ALOGE("Received unterminated hidl_string buffer.");
1109d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen        return BAD_VALUE;
1119d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    }
1129d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen
1139d3eb357a2a3a64baa4824d550b5563b3dc0b15aMartijn Coenen    return OK;
1147f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}
1157f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1167f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongstatus_t writeEmbeddedToParcel(const hidl_string &string,
1177f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        Parcel *parcel, size_t parentHandle, size_t parentOffset) {
1187f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    return parcel->writeEmbeddedBuffer(
1197f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            string.c_str(),
1207f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            string.size() + 1,
1217f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            nullptr /* handle */,
1227f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            parentHandle,
1237f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            parentOffset + hidl_string::kOffsetOfBuffer);
1247f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}
1257f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1267f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongandroid::status_t writeToParcel(const hidl_version &version, android::hardware::Parcel& parcel) {
1277f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    return parcel.writeUint32(static_cast<uint32_t>(version.get_major()) << 16 | version.get_minor());
1287f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}
1297f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1307f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Honghidl_version* readFromParcel(const android::hardware::Parcel& parcel) {
1317f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    uint32_t version;
1327f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    android::status_t status = parcel.readUint32(&version);
1337f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (status != OK) {
1347f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return nullptr;
1357f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    } else {
1367f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return new hidl_version(version >> 16, version & 0xFFFF);
1377f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1387f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}
1397f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1407f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongstatus_t readFromParcel(Status *s, const Parcel& parcel) {
1417f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    int32_t exception;
1427f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    status_t status = parcel.readInt32(&exception);
1437f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (status != OK) {
1447f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        s->setFromStatusT(status);
1457f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return status;
1467f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1477f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1487f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    // Skip over fat response headers.  Not used (or propagated) in native code.
1497f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (exception == Status::EX_HAS_REPLY_HEADER) {
1507f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        // Note that the header size includes the 4 byte size field.
1517f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        const int32_t header_start = parcel.dataPosition();
1527f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        int32_t header_size;
1537f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        status = parcel.readInt32(&header_size);
1547f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        if (status != OK) {
1557f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            s->setFromStatusT(status);
1567f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong            return status;
1577f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        }
1587f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        parcel.setDataPosition(header_start + header_size);
1597f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        // And fat response headers are currently only used when there are no
1607f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        // exceptions, so act like there was no error.
1617f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        exception = Status::EX_NONE;
1627f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1637f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1647f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (exception == Status::EX_NONE) {
1657f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        *s = Status::ok();
1667f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return status;
1677f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1687f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1697f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    // The remote threw an exception.  Get the message back.
1707f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    String16 message;
1717f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    status = parcel.readString16(&message);
1727f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (status != OK) {
1737f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        s->setFromStatusT(status);
1747f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return status;
1757f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1767f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
17772db40f4a612d04187ae2fc1f9a3766933264bdfSteven Moreland    s->setException(exception, String8(message));
1787f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1797f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    return status;
1807f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}
1817f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1827f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hongstatus_t writeToParcel(const Status &s, Parcel* parcel) {
1837f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    // Something really bad has happened, and we're not going to even
1847f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    // try returning rich error data.
1857f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (s.exceptionCode() == Status::EX_TRANSACTION_FAILED) {
1867f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return s.transactionError();
1877f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1887f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1897f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    status_t status = parcel->writeInt32(s.exceptionCode());
1907f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (status != OK) { return status; }
1917f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    if (s.exceptionCode() == Status::EX_NONE) {
1927f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        // We have no more information to write.
1937f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong        return status;
1947f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    }
1957f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    status = parcel->writeString16(String16(s.exceptionMessage()));
1967f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong    return status;
1977f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}
1987f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong
1996cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Morelandvoid configureBinderRpcThreadpool(size_t maxThreads, bool callerWillJoin) {
2006cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Moreland    ProcessState::self()->setThreadPoolConfiguration(maxThreads, callerWillJoin /*callerJoinsPool*/);
2016cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Moreland}
2026cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Moreland
2036cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Morelandvoid joinBinderRpcThreadpool() {
2046cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Moreland    IPCThreadState::self()->joinThreadPool();
2056cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Moreland}
2066cf8fa2d989bf57aed9a62fd9f54261176a9f860Steven Moreland
2077f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}  // namespace hardware
2087f97f44562b057c3c780c0a05c101b677f9b0f96Yifan Hong}  // namespace android
209