/*
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#define LOG_TAG "HidlStatus"
#include <android-base/logging.h>
#include <hidl/Status.h>
#include <utils/CallStack.h>
#include <unordered_map>
namespace android {
namespace hardware {
static std::string statusToString(status_t s) {
const std::unordered_map<status_t, std::string> statusStrings{{
#define STATUS_TO_STRING_PAIR(STATUS) {STATUS, #STATUS}
STATUS_TO_STRING_PAIR(OK),
STATUS_TO_STRING_PAIR(UNKNOWN_ERROR),
STATUS_TO_STRING_PAIR(NO_MEMORY),
STATUS_TO_STRING_PAIR(INVALID_OPERATION),
STATUS_TO_STRING_PAIR(BAD_VALUE),
STATUS_TO_STRING_PAIR(BAD_TYPE),
STATUS_TO_STRING_PAIR(NAME_NOT_FOUND),
STATUS_TO_STRING_PAIR(PERMISSION_DENIED),
STATUS_TO_STRING_PAIR(NO_INIT),
STATUS_TO_STRING_PAIR(ALREADY_EXISTS),
STATUS_TO_STRING_PAIR(DEAD_OBJECT),
STATUS_TO_STRING_PAIR(FAILED_TRANSACTION),
STATUS_TO_STRING_PAIR(BAD_INDEX),
STATUS_TO_STRING_PAIR(NOT_ENOUGH_DATA),
STATUS_TO_STRING_PAIR(WOULD_BLOCK),
STATUS_TO_STRING_PAIR(TIMED_OUT),
STATUS_TO_STRING_PAIR(UNKNOWN_TRANSACTION),
STATUS_TO_STRING_PAIR(FDS_NOT_ALLOWED),
STATUS_TO_STRING_PAIR(UNEXPECTED_NULL)
}};
auto it = statusStrings.find(s);
if (it != statusStrings.end()) {
return it->second;
}
std::string str = std::to_string(s);
char *err = strerror(-s);
if (err != nullptr) {
str.append(1, ' ').append(err);
}
return str;
}
static std::string exceptionToString(int32_t ex) {
const std::unordered_map<int32_t, std::string> exceptionStrings{{
#define EXCEPTION_TO_STRING_PAIR(EXCEPTION) {Status::Exception::EXCEPTION, #EXCEPTION}
EXCEPTION_TO_STRING_PAIR(EX_NONE),
EXCEPTION_TO_STRING_PAIR(EX_SECURITY),
EXCEPTION_TO_STRING_PAIR(EX_BAD_PARCELABLE),
EXCEPTION_TO_STRING_PAIR(EX_ILLEGAL_ARGUMENT),
EXCEPTION_TO_STRING_PAIR(EX_NULL_POINTER),
EXCEPTION_TO_STRING_PAIR(EX_ILLEGAL_STATE),
EXCEPTION_TO_STRING_PAIR(EX_NETWORK_MAIN_THREAD),
EXCEPTION_TO_STRING_PAIR(EX_UNSUPPORTED_OPERATION),
EXCEPTION_TO_STRING_PAIR(EX_HAS_REPLY_HEADER),
EXCEPTION_TO_STRING_PAIR(EX_TRANSACTION_FAILED)
}};
auto it = exceptionStrings.find(ex);
return it == exceptionStrings.end() ? std::to_string(ex) : it->second;
}
Status Status::ok() {
return Status();
}
Status Status::fromExceptionCode(int32_t exceptionCode) {
if (exceptionCode == EX_TRANSACTION_FAILED) {
return Status(exceptionCode, FAILED_TRANSACTION);
}
return Status(exceptionCode, OK);
}
Status Status::fromExceptionCode(int32_t exceptionCode,
const char *message) {
if (exceptionCode == EX_TRANSACTION_FAILED) {
return Status(exceptionCode, FAILED_TRANSACTION, message);
}
return Status(exceptionCode, OK, message);
}
Status Status::fromStatusT(status_t status) {
Status ret;
ret.setFromStatusT(status);
return ret;
}
Status::Status(int32_t exceptionCode, int32_t errorCode)
: mException(exceptionCode),
mErrorCode(errorCode) {}
Status::Status(int32_t exceptionCode, int32_t errorCode, const char *message)
: mException(exceptionCode),
mErrorCode(errorCode),
mMessage(message) {}
void Status::setException(int32_t ex, const char *message) {
mException = ex;
mErrorCode = ex == EX_TRANSACTION_FAILED ? FAILED_TRANSACTION : NO_ERROR;
mMessage = message;
}
void Status::setFromStatusT(status_t status) {
mException = (status == NO_ERROR) ? EX_NONE : EX_TRANSACTION_FAILED;
mErrorCode = status;
mMessage.clear();
}
std::string Status::description() const {
std::ostringstream oss;
oss << (*this);
return oss.str();
}
std::ostream& operator<< (std::ostream& stream, const Status& s) {
if (s.exceptionCode() == Status::EX_NONE) {
stream << "No error";
} else {
stream << "Status(" << exceptionToString(s.exceptionCode()) << "): '";
if (s.exceptionCode() == Status::EX_TRANSACTION_FAILED) {
stream << statusToString(s.transactionError()) << ": ";
}
stream << s.exceptionMessage() << "'";
}
return stream;
}
static HidlReturnRestriction gReturnRestriction = HidlReturnRestriction::NONE;
void setProcessHidlReturnRestriction(HidlReturnRestriction restriction) {
gReturnRestriction = restriction;
}
namespace details {
void return_status::assertOk() const {
if (!isOk()) {
LOG(FATAL) << "Attempted to retrieve value from failed HIDL call: " << description();
}
}
return_status::~return_status() {
// mCheckedStatus must be checked before isOk since isOk modifies mCheckedStatus
if (mCheckedStatus) return;
if (!isOk()) {
LOG(FATAL) << "Failed HIDL return status not checked: " << description();
}
if (gReturnRestriction == HidlReturnRestriction::NONE) {
return;
}
if (gReturnRestriction == HidlReturnRestriction::ERROR_IF_UNCHECKED) {
LOG(ERROR) << "Failed to check status of HIDL Return.";
CallStack::logStack("unchecked HIDL return", CallStack::getCurrent(10).get(), ANDROID_LOG_ERROR);
} else {
LOG(FATAL) << "Failed to check status of HIDL Return.";
}
}
return_status& return_status::operator=(return_status&& other) noexcept {
if (!mCheckedStatus && !isOk()) {
LOG(FATAL) << "Failed HIDL return status not checked: " << description();
}
std::swap(mStatus, other.mStatus);
std::swap(mCheckedStatus, other.mCheckedStatus);
return *this;
}
} // namespace details
} // namespace hardware
} // namespace android