blob: 90d59321ca9b856716add2b2690af517afceeccc [file] [log] [blame]
/*
* Copyright (C) 2020 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.
*/
#ifndef ANDROID_FRAMEWORKS_ML_NN_RUNTIME_EVENT_H
#define ANDROID_FRAMEWORKS_ML_NN_RUNTIME_EVENT_H
#include <android-base/logging.h>
#include <nnapi/IPreparedModel.h>
#include <memory>
#include <utility>
#include "ExecutionCallback.h"
namespace android::nn {
class IEvent {
public:
virtual ~IEvent() = default;
virtual void wait() const = 0;
virtual ErrorStatus getStatus() const = 0;
virtual int getSyncFenceFd(bool shouldDup) const = 0;
};
// The CallbackEvent wraps ExecutionCallback
class CallbackEvent : public IEvent {
public:
CallbackEvent(std::shared_ptr<ExecutionCallback> callback)
: kExecutionCallback(std::move(callback)) {
CHECK(kExecutionCallback != nullptr);
}
void wait() const override { kExecutionCallback->wait(); }
ErrorStatus getStatus() const override { return kExecutionCallback->getStatus(); }
// Always return -1 as this is not backed by a sync fence.
int getSyncFenceFd(bool /*should_dup*/) const override { return -1; }
private:
const std::shared_ptr<ExecutionCallback> kExecutionCallback;
};
// The SyncFenceEvent wraps sync fence and ExecuteFencedInfoCallback
class SyncFenceEvent : public IEvent {
public:
SyncFenceEvent(int sync_fence_fd, const ExecuteFencedInfoCallback& callback)
: kFencedExecutionCallback(callback) {
if (sync_fence_fd > 0) {
// Dup the provided file descriptor
mSyncFenceFd = dup(sync_fence_fd);
CHECK(mSyncFenceFd > 0);
}
}
// Close the fd the event owns.
~SyncFenceEvent() { close(mSyncFenceFd); }
// Use syncWait to wait for the sync fence until the status change.
void wait() const override { syncWait(mSyncFenceFd, -1); }
// Get the status of the event.
// In case of syncWait error, query the dispatch callback for detailed
// error status.
ErrorStatus getStatus() const override {
auto error = ErrorStatus::NONE;
if (mSyncFenceFd > 0 && syncWait(mSyncFenceFd, -1) != FenceState::SIGNALED) {
error = ErrorStatus::GENERAL_FAILURE;
// If there is a callback available, use the callback to get the error code.
if (kFencedExecutionCallback != nullptr) {
auto result = kFencedExecutionCallback();
if (!result.has_value()) {
LOG(ERROR) << "Fenced execution callback failed: " << result.error().message;
error = result.error().code;
CHECK_NE(error, ErrorStatus::NONE);
}
}
}
return error;
}
// Return the sync fence fd.
// If shouldDup is true, the caller must close the fd returned:
// - When being used internally within NNAPI runtime, set shouldDup to false.
// - When being used to return a fd to application code, set shouldDup to
// true.
int getSyncFenceFd(bool shouldDup) const override {
if (shouldDup) {
return dup(mSyncFenceFd);
}
return mSyncFenceFd;
}
private:
// TODO(b/148423931): used android::base::unique_fd instead.
int mSyncFenceFd = -1;
const ExecuteFencedInfoCallback kFencedExecutionCallback;
};
} // namespace android::nn
#endif // ANDROID_FRAMEWORKS_ML_NN_RUNTIME_EVENT_H