Add recovery code to NN ResilientPreparedModel and *Buffer am: 418c749746 am: ade46f92f8

Original change: https://android-review.googlesource.com/c/platform/hardware/interfaces/+/1544545

MUST ONLY BE SUBMITTED BY AUTOMERGER

Change-Id: Icc4cc8c1c524f442d5086444928861dca59c6868
This commit is contained in:
Michael Butler
2021-01-11 18:51:31 +00:00
committed by Automerger Merge Worker
5 changed files with 113 additions and 15 deletions

View File

@@ -42,7 +42,7 @@ class ResilientBuffer final : public nn::IBuffer {
nn::SharedBuffer buffer); nn::SharedBuffer buffer);
nn::SharedBuffer getBuffer() const; nn::SharedBuffer getBuffer() const;
nn::SharedBuffer recover(const nn::IBuffer* failingBuffer, bool blocking) const; nn::GeneralResult<nn::SharedBuffer> recover(const nn::IBuffer* failingBuffer) const;
nn::Request::MemoryDomainToken getToken() const override; nn::Request::MemoryDomainToken getToken() const override;

View File

@@ -43,8 +43,8 @@ class ResilientPreparedModel final : public nn::IPreparedModel {
nn::SharedPreparedModel preparedModel); nn::SharedPreparedModel preparedModel);
nn::SharedPreparedModel getPreparedModel() const; nn::SharedPreparedModel getPreparedModel() const;
nn::SharedPreparedModel recover(const nn::IPreparedModel* failingPreparedModel, nn::GeneralResult<nn::SharedPreparedModel> recover(
bool blocking) const; const nn::IPreparedModel* failingPreparedModel) const;
nn::ExecutionResult<std::pair<std::vector<nn::OutputShape>, nn::Timing>> execute( nn::ExecutionResult<std::pair<std::vector<nn::OutputShape>, nn::Timing>> execute(
const nn::Request& request, nn::MeasureTiming measure, const nn::Request& request, nn::MeasureTiming measure,

View File

@@ -20,6 +20,7 @@
#include <android-base/thread_annotations.h> #include <android-base/thread_annotations.h>
#include <nnapi/IBuffer.h> #include <nnapi/IBuffer.h>
#include <nnapi/Result.h> #include <nnapi/Result.h>
#include <nnapi/TypeUtils.h>
#include <nnapi/Types.h> #include <nnapi/Types.h>
#include <functional> #include <functional>
@@ -29,6 +30,34 @@
#include <vector> #include <vector>
namespace android::hardware::neuralnetworks::utils { namespace android::hardware::neuralnetworks::utils {
namespace {
template <typename FnType>
auto protect(const ResilientBuffer& resilientBuffer, const FnType& fn)
-> decltype(fn(*resilientBuffer.getBuffer())) {
auto buffer = resilientBuffer.getBuffer();
auto result = fn(*buffer);
// Immediately return if device is not dead.
if (result.has_value() || result.error().code != nn::ErrorStatus::DEAD_OBJECT) {
return result;
}
// Attempt recovery and return if it fails.
auto maybeBuffer = resilientBuffer.recover(buffer.get());
if (!maybeBuffer.has_value()) {
const auto& [resultErrorMessage, resultErrorCode] = result.error();
const auto& [recoveryErrorMessage, recoveryErrorCode] = maybeBuffer.error();
return nn::error(resultErrorCode)
<< resultErrorMessage << ", and failed to recover dead buffer with error "
<< recoveryErrorCode << ": " << recoveryErrorMessage;
}
buffer = std::move(maybeBuffer).value();
return fn(*buffer);
}
} // namespace
nn::GeneralResult<std::shared_ptr<const ResilientBuffer>> ResilientBuffer::create( nn::GeneralResult<std::shared_ptr<const ResilientBuffer>> ResilientBuffer::create(
Factory makeBuffer) { Factory makeBuffer) {
@@ -53,9 +82,16 @@ nn::SharedBuffer ResilientBuffer::getBuffer() const {
std::lock_guard guard(mMutex); std::lock_guard guard(mMutex);
return mBuffer; return mBuffer;
} }
nn::SharedBuffer ResilientBuffer::recover(const nn::IBuffer* /*failingBuffer*/, nn::GeneralResult<nn::SharedBuffer> ResilientBuffer::recover(
bool /*blocking*/) const { const nn::IBuffer* failingBuffer) const {
std::lock_guard guard(mMutex); std::lock_guard guard(mMutex);
// Another caller updated the failing prepared model.
if (mBuffer.get() != failingBuffer) {
return mBuffer;
}
mBuffer = NN_TRY(kMakeBuffer());
return mBuffer; return mBuffer;
} }
@@ -64,12 +100,16 @@ nn::Request::MemoryDomainToken ResilientBuffer::getToken() const {
} }
nn::GeneralResult<void> ResilientBuffer::copyTo(const nn::Memory& dst) const { nn::GeneralResult<void> ResilientBuffer::copyTo(const nn::Memory& dst) const {
return getBuffer()->copyTo(dst); const auto fn = [&dst](const nn::IBuffer& buffer) { return buffer.copyTo(dst); };
return protect(*this, fn);
} }
nn::GeneralResult<void> ResilientBuffer::copyFrom(const nn::Memory& src, nn::GeneralResult<void> ResilientBuffer::copyFrom(const nn::Memory& src,
const nn::Dimensions& dimensions) const { const nn::Dimensions& dimensions) const {
return getBuffer()->copyFrom(src, dimensions); const auto fn = [&src, &dimensions](const nn::IBuffer& buffer) {
return buffer.copyFrom(src, dimensions);
};
return protect(*this, fn);
} }
} // namespace android::hardware::neuralnetworks::utils } // namespace android::hardware::neuralnetworks::utils

View File

@@ -180,6 +180,7 @@ nn::GeneralResult<nn::SharedPreparedModel> ResilientDevice::prepareModel(
const nn::Model& model, nn::ExecutionPreference preference, nn::Priority priority, const nn::Model& model, nn::ExecutionPreference preference, nn::Priority priority,
nn::OptionalTimePoint deadline, const std::vector<nn::SharedHandle>& modelCache, nn::OptionalTimePoint deadline, const std::vector<nn::SharedHandle>& modelCache,
const std::vector<nn::SharedHandle>& dataCache, const nn::CacheToken& token) const { const std::vector<nn::SharedHandle>& dataCache, const nn::CacheToken& token) const {
#if 0
auto self = shared_from_this(); auto self = shared_from_this();
ResilientPreparedModel::Factory makePreparedModel = [device = std::move(self), model, ResilientPreparedModel::Factory makePreparedModel = [device = std::move(self), model,
preference, priority, deadline, modelCache, preference, priority, deadline, modelCache,
@@ -188,29 +189,41 @@ nn::GeneralResult<nn::SharedPreparedModel> ResilientDevice::prepareModel(
dataCache, token); dataCache, token);
}; };
return ResilientPreparedModel::create(std::move(makePreparedModel)); return ResilientPreparedModel::create(std::move(makePreparedModel));
#else
return prepareModelInternal(model, preference, priority, deadline, modelCache, dataCache,
token);
#endif
} }
nn::GeneralResult<nn::SharedPreparedModel> ResilientDevice::prepareModelFromCache( nn::GeneralResult<nn::SharedPreparedModel> ResilientDevice::prepareModelFromCache(
nn::OptionalTimePoint deadline, const std::vector<nn::SharedHandle>& modelCache, nn::OptionalTimePoint deadline, const std::vector<nn::SharedHandle>& modelCache,
const std::vector<nn::SharedHandle>& dataCache, const nn::CacheToken& token) const { const std::vector<nn::SharedHandle>& dataCache, const nn::CacheToken& token) const {
#if 0
auto self = shared_from_this(); auto self = shared_from_this();
ResilientPreparedModel::Factory makePreparedModel = [device = std::move(self), deadline, ResilientPreparedModel::Factory makePreparedModel = [device = std::move(self), deadline,
modelCache, dataCache, token] { modelCache, dataCache, token] {
return device->prepareModelFromCacheInternal(deadline, modelCache, dataCache, token); return device->prepareModelFromCacheInternal(deadline, modelCache, dataCache, token);
}; };
return ResilientPreparedModel::create(std::move(makePreparedModel)); return ResilientPreparedModel::create(std::move(makePreparedModel));
#else
return prepareModelFromCacheInternal(deadline, modelCache, dataCache, token);
#endif
} }
nn::GeneralResult<nn::SharedBuffer> ResilientDevice::allocate( nn::GeneralResult<nn::SharedBuffer> ResilientDevice::allocate(
const nn::BufferDesc& desc, const std::vector<nn::SharedPreparedModel>& preparedModels, const nn::BufferDesc& desc, const std::vector<nn::SharedPreparedModel>& preparedModels,
const std::vector<nn::BufferRole>& inputRoles, const std::vector<nn::BufferRole>& inputRoles,
const std::vector<nn::BufferRole>& outputRoles) const { const std::vector<nn::BufferRole>& outputRoles) const {
#if 0
auto self = shared_from_this(); auto self = shared_from_this();
ResilientBuffer::Factory makeBuffer = [device = std::move(self), desc, preparedModels, ResilientBuffer::Factory makeBuffer = [device = std::move(self), desc, preparedModels,
inputRoles, outputRoles] { inputRoles, outputRoles] {
return device->allocateInternal(desc, preparedModels, inputRoles, outputRoles); return device->allocateInternal(desc, preparedModels, inputRoles, outputRoles);
}; };
return ResilientBuffer::create(std::move(makeBuffer)); return ResilientBuffer::create(std::move(makeBuffer));
#else
return allocateInternal(desc, preparedModels, inputRoles, outputRoles);
#endif
} }
bool ResilientDevice::isValidInternal() const { bool ResilientDevice::isValidInternal() const {
@@ -225,8 +238,8 @@ nn::GeneralResult<nn::SharedPreparedModel> ResilientDevice::prepareModelInternal
if (!isValidInternal()) { if (!isValidInternal()) {
return std::make_shared<const InvalidPreparedModel>(); return std::make_shared<const InvalidPreparedModel>();
} }
const auto fn = [&model, preference, priority, deadline, &modelCache, &dataCache, const auto fn = [&model, preference, priority, &deadline, &modelCache, &dataCache,
token](const nn::IDevice& device) { &token](const nn::IDevice& device) {
return device.prepareModel(model, preference, priority, deadline, modelCache, dataCache, return device.prepareModel(model, preference, priority, deadline, modelCache, dataCache,
token); token);
}; };
@@ -239,7 +252,7 @@ nn::GeneralResult<nn::SharedPreparedModel> ResilientDevice::prepareModelFromCach
if (!isValidInternal()) { if (!isValidInternal()) {
return std::make_shared<const InvalidPreparedModel>(); return std::make_shared<const InvalidPreparedModel>();
} }
const auto fn = [deadline, &modelCache, &dataCache, token](const nn::IDevice& device) { const auto fn = [&deadline, &modelCache, &dataCache, &token](const nn::IDevice& device) {
return device.prepareModelFromCache(deadline, modelCache, dataCache, token); return device.prepareModelFromCache(deadline, modelCache, dataCache, token);
}; };
return protect(*this, fn, /*blocking=*/false); return protect(*this, fn, /*blocking=*/false);

View File

@@ -20,15 +20,45 @@
#include <android-base/thread_annotations.h> #include <android-base/thread_annotations.h>
#include <nnapi/IPreparedModel.h> #include <nnapi/IPreparedModel.h>
#include <nnapi/Result.h> #include <nnapi/Result.h>
#include <nnapi/TypeUtils.h>
#include <nnapi/Types.h> #include <nnapi/Types.h>
#include <functional> #include <functional>
#include <memory> #include <memory>
#include <mutex> #include <mutex>
#include <sstream>
#include <utility> #include <utility>
#include <vector> #include <vector>
namespace android::hardware::neuralnetworks::utils { namespace android::hardware::neuralnetworks::utils {
namespace {
template <typename FnType>
auto protect(const ResilientPreparedModel& resilientPreparedModel, const FnType& fn)
-> decltype(fn(*resilientPreparedModel.getPreparedModel())) {
auto preparedModel = resilientPreparedModel.getPreparedModel();
auto result = fn(*preparedModel);
// Immediately return if prepared model is not dead.
if (result.has_value() || result.error().code != nn::ErrorStatus::DEAD_OBJECT) {
return result;
}
// Attempt recovery and return if it fails.
auto maybePreparedModel = resilientPreparedModel.recover(preparedModel.get());
if (!maybePreparedModel.has_value()) {
const auto& [message, code] = maybePreparedModel.error();
std::ostringstream oss;
oss << ", and failed to recover dead prepared model with error " << code << ": " << message;
result.error().message += oss.str();
return result;
}
preparedModel = std::move(maybePreparedModel).value();
return fn(*preparedModel);
}
} // namespace
nn::GeneralResult<std::shared_ptr<const ResilientPreparedModel>> ResilientPreparedModel::create( nn::GeneralResult<std::shared_ptr<const ResilientPreparedModel>> ResilientPreparedModel::create(
Factory makePreparedModel) { Factory makePreparedModel) {
@@ -55,9 +85,16 @@ nn::SharedPreparedModel ResilientPreparedModel::getPreparedModel() const {
return mPreparedModel; return mPreparedModel;
} }
nn::SharedPreparedModel ResilientPreparedModel::recover( nn::GeneralResult<nn::SharedPreparedModel> ResilientPreparedModel::recover(
const nn::IPreparedModel* /*failingPreparedModel*/, bool /*blocking*/) const { const nn::IPreparedModel* failingPreparedModel) const {
std::lock_guard guard(mMutex); std::lock_guard guard(mMutex);
// Another caller updated the failing prepared model.
if (mPreparedModel.get() != failingPreparedModel) {
return mPreparedModel;
}
mPreparedModel = NN_TRY(kMakePreparedModel());
return mPreparedModel; return mPreparedModel;
} }
@@ -65,7 +102,11 @@ nn::ExecutionResult<std::pair<std::vector<nn::OutputShape>, nn::Timing>>
ResilientPreparedModel::execute(const nn::Request& request, nn::MeasureTiming measure, ResilientPreparedModel::execute(const nn::Request& request, nn::MeasureTiming measure,
const nn::OptionalTimePoint& deadline, const nn::OptionalTimePoint& deadline,
const nn::OptionalDuration& loopTimeoutDuration) const { const nn::OptionalDuration& loopTimeoutDuration) const {
return getPreparedModel()->execute(request, measure, deadline, loopTimeoutDuration); const auto fn = [&request, measure, &deadline,
&loopTimeoutDuration](const nn::IPreparedModel& preparedModel) {
return preparedModel.execute(request, measure, deadline, loopTimeoutDuration);
};
return protect(*this, fn);
} }
nn::GeneralResult<std::pair<nn::SyncFence, nn::ExecuteFencedInfoCallback>> nn::GeneralResult<std::pair<nn::SyncFence, nn::ExecuteFencedInfoCallback>>
@@ -75,8 +116,12 @@ ResilientPreparedModel::executeFenced(const nn::Request& request,
const nn::OptionalTimePoint& deadline, const nn::OptionalTimePoint& deadline,
const nn::OptionalDuration& loopTimeoutDuration, const nn::OptionalDuration& loopTimeoutDuration,
const nn::OptionalDuration& timeoutDurationAfterFence) const { const nn::OptionalDuration& timeoutDurationAfterFence) const {
return getPreparedModel()->executeFenced(request, waitFor, measure, deadline, const auto fn = [&request, &waitFor, measure, &deadline, &loopTimeoutDuration,
loopTimeoutDuration, timeoutDurationAfterFence); &timeoutDurationAfterFence](const nn::IPreparedModel& preparedModel) {
return preparedModel.executeFenced(request, waitFor, measure, deadline, loopTimeoutDuration,
timeoutDurationAfterFence);
};
return protect(*this, fn);
} }
std::any ResilientPreparedModel::getUnderlyingResource() const { std::any ResilientPreparedModel::getUnderlyingResource() const {