Home | History | Annotate | Download | only in binder
      1 /*
      2  * Copyright (C) 2015 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 #include <binder/Status.h>
     18 
     19 namespace android {
     20 namespace binder {
     21 
     22 Status Status::ok() {
     23     return Status();
     24 }
     25 
     26 Status Status::fromExceptionCode(int32_t exceptionCode) {
     27     if (exceptionCode == EX_TRANSACTION_FAILED) {
     28         return Status(exceptionCode, FAILED_TRANSACTION);
     29     }
     30     return Status(exceptionCode, OK);
     31 }
     32 
     33 Status Status::fromExceptionCode(int32_t exceptionCode,
     34                                  const String8& message) {
     35     if (exceptionCode == EX_TRANSACTION_FAILED) {
     36         return Status(exceptionCode, FAILED_TRANSACTION, message);
     37     }
     38     return Status(exceptionCode, OK, message);
     39 }
     40 
     41 Status Status::fromExceptionCode(int32_t exceptionCode,
     42                                  const char* message) {
     43     return fromExceptionCode(exceptionCode, String8(message));
     44 }
     45 
     46 Status Status::fromServiceSpecificError(int32_t serviceSpecificErrorCode) {
     47     return Status(EX_SERVICE_SPECIFIC, serviceSpecificErrorCode);
     48 }
     49 
     50 Status Status::fromServiceSpecificError(int32_t serviceSpecificErrorCode,
     51                                         const String8& message) {
     52     return Status(EX_SERVICE_SPECIFIC, serviceSpecificErrorCode, message);
     53 }
     54 
     55 Status Status::fromServiceSpecificError(int32_t serviceSpecificErrorCode,
     56                                         const char* message) {
     57     return fromServiceSpecificError(serviceSpecificErrorCode, String8(message));
     58 }
     59 
     60 Status Status::fromStatusT(status_t status) {
     61     Status ret;
     62     ret.setFromStatusT(status);
     63     return ret;
     64 }
     65 
     66 std::string Status::exceptionToString(int32_t exceptionCode) {
     67     switch (exceptionCode) {
     68         #define EXCEPTION_TO_CASE(EXCEPTION) case EXCEPTION: return #EXCEPTION;
     69         EXCEPTION_TO_CASE(EX_NONE)
     70         EXCEPTION_TO_CASE(EX_SECURITY)
     71         EXCEPTION_TO_CASE(EX_BAD_PARCELABLE)
     72         EXCEPTION_TO_CASE(EX_ILLEGAL_ARGUMENT)
     73         EXCEPTION_TO_CASE(EX_NULL_POINTER)
     74         EXCEPTION_TO_CASE(EX_ILLEGAL_STATE)
     75         EXCEPTION_TO_CASE(EX_NETWORK_MAIN_THREAD)
     76         EXCEPTION_TO_CASE(EX_UNSUPPORTED_OPERATION)
     77         EXCEPTION_TO_CASE(EX_SERVICE_SPECIFIC)
     78         EXCEPTION_TO_CASE(EX_PARCELABLE)
     79         EXCEPTION_TO_CASE(EX_HAS_REPLY_HEADER)
     80         EXCEPTION_TO_CASE(EX_TRANSACTION_FAILED)
     81         #undef EXCEPTION_TO_CASE
     82         default: return std::to_string(exceptionCode);
     83     }
     84 }
     85 
     86 Status::Status(int32_t exceptionCode, int32_t errorCode)
     87     : mException(exceptionCode),
     88       mErrorCode(errorCode) {}
     89 
     90 Status::Status(int32_t exceptionCode, int32_t errorCode, const String8& message)
     91     : mException(exceptionCode),
     92       mErrorCode(errorCode),
     93       mMessage(message) {}
     94 
     95 status_t Status::readFromParcel(const Parcel& parcel) {
     96     status_t status = parcel.readInt32(&mException);
     97     if (status != OK) {
     98         setFromStatusT(status);
     99         return status;
    100     }
    101 
    102     // Skip over fat response headers.  Not used (or propagated) in native code.
    103     if (mException == EX_HAS_REPLY_HEADER) {
    104         // Note that the header size includes the 4 byte size field.
    105         const size_t header_start = parcel.dataPosition();
    106         // Get available size before reading more
    107         const size_t header_avail = parcel.dataAvail();
    108 
    109         int32_t header_size;
    110         status = parcel.readInt32(&header_size);
    111         if (status != OK) {
    112             setFromStatusT(status);
    113             return status;
    114         }
    115 
    116         if (header_size < 0 || static_cast<size_t>(header_size) > header_avail) {
    117             android_errorWriteLog(0x534e4554, "132650049");
    118             setFromStatusT(UNKNOWN_ERROR);
    119             return UNKNOWN_ERROR;
    120         }
    121 
    122         parcel.setDataPosition(header_start + header_size);
    123         // And fat response headers are currently only used when there are no
    124         // exceptions, so act like there was no error.
    125         mException = EX_NONE;
    126     }
    127 
    128     if (mException == EX_NONE) {
    129         return status;
    130     }
    131 
    132     // The remote threw an exception.  Get the message back.
    133     String16 message;
    134     status = parcel.readString16(&message);
    135     if (status != OK) {
    136         setFromStatusT(status);
    137         return status;
    138     }
    139     mMessage = String8(message);
    140 
    141     // Skip over the remote stack trace data
    142     int32_t remote_stack_trace_header_size;
    143     status = parcel.readInt32(&remote_stack_trace_header_size);
    144     if (status != OK) {
    145         setFromStatusT(status);
    146         return status;
    147     }
    148     if (remote_stack_trace_header_size < 0 ||
    149         static_cast<size_t>(remote_stack_trace_header_size) > parcel.dataAvail()) {
    150 
    151         android_errorWriteLog(0x534e4554, "132650049");
    152         setFromStatusT(UNKNOWN_ERROR);
    153         return UNKNOWN_ERROR;
    154     }
    155     parcel.setDataPosition(parcel.dataPosition() + remote_stack_trace_header_size);
    156 
    157     if (mException == EX_SERVICE_SPECIFIC) {
    158         status = parcel.readInt32(&mErrorCode);
    159     } else if (mException == EX_PARCELABLE) {
    160         // Skip over the blob of Parcelable data
    161         const size_t header_start = parcel.dataPosition();
    162         // Get available size before reading more
    163         const size_t header_avail = parcel.dataAvail();
    164 
    165         int32_t header_size;
    166         status = parcel.readInt32(&header_size);
    167         if (status != OK) {
    168             setFromStatusT(status);
    169             return status;
    170         }
    171 
    172         if (header_size < 0 || static_cast<size_t>(header_size) > header_avail) {
    173             android_errorWriteLog(0x534e4554, "132650049");
    174             setFromStatusT(UNKNOWN_ERROR);
    175             return UNKNOWN_ERROR;
    176         }
    177 
    178         parcel.setDataPosition(header_start + header_size);
    179     }
    180     if (status != OK) {
    181         setFromStatusT(status);
    182         return status;
    183     }
    184 
    185     return status;
    186 }
    187 
    188 status_t Status::writeToParcel(Parcel* parcel) const {
    189     // Something really bad has happened, and we're not going to even
    190     // try returning rich error data.
    191     if (mException == EX_TRANSACTION_FAILED) {
    192         return mErrorCode;
    193     }
    194 
    195     status_t status = parcel->writeInt32(mException);
    196     if (status != OK) { return status; }
    197     if (mException == EX_NONE) {
    198         // We have no more information to write.
    199         return status;
    200     }
    201     status = parcel->writeString16(String16(mMessage));
    202     status = parcel->writeInt32(0); // Empty remote stack trace header
    203     if (mException == EX_SERVICE_SPECIFIC) {
    204         status = parcel->writeInt32(mErrorCode);
    205     } else if (mException == EX_PARCELABLE) {
    206         // Sending Parcelable blobs currently not supported
    207         status = parcel->writeInt32(0);
    208     }
    209     return status;
    210 }
    211 
    212 void Status::setException(int32_t ex, const String8& message) {
    213     mException = ex;
    214     mErrorCode = ex == EX_TRANSACTION_FAILED ? FAILED_TRANSACTION : NO_ERROR;
    215     mMessage.setTo(message);
    216 }
    217 
    218 void Status::setServiceSpecificError(int32_t errorCode, const String8& message) {
    219     setException(EX_SERVICE_SPECIFIC, message);
    220     mErrorCode = errorCode;
    221 }
    222 
    223 void Status::setFromStatusT(status_t status) {
    224     mException = (status == NO_ERROR) ? EX_NONE : EX_TRANSACTION_FAILED;
    225     mErrorCode = status;
    226     mMessage.clear();
    227 }
    228 
    229 String8 Status::toString8() const {
    230     String8 ret;
    231     if (mException == EX_NONE) {
    232         ret.append("No error");
    233     } else {
    234         ret.appendFormat("Status(%d, %s): '", mException, exceptionToString(mException).c_str());
    235         if (mException == EX_SERVICE_SPECIFIC ||
    236             mException == EX_TRANSACTION_FAILED) {
    237             ret.appendFormat("%d: ", mErrorCode);
    238         }
    239         ret.append(String8(mMessage));
    240         ret.append("'");
    241     }
    242     return ret;
    243 }
    244 
    245 std::stringstream& operator<< (std::stringstream& stream, const Status& s) {
    246     stream << s.toString8().string();
    247     return stream;
    248 }
    249 
    250 }  // namespace binder
    251 }  // namespace android
    252