1bc2ed2b3Sopenharmony_ci/*
2bc2ed2b3Sopenharmony_ci * Copyright (C) 2022 Huawei Device Co., Ltd.
3bc2ed2b3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4bc2ed2b3Sopenharmony_ci * you may not use this file except in compliance with the License.
5bc2ed2b3Sopenharmony_ci * You may obtain a copy of the License at
6bc2ed2b3Sopenharmony_ci *
7bc2ed2b3Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8bc2ed2b3Sopenharmony_ci *
9bc2ed2b3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10bc2ed2b3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11bc2ed2b3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12bc2ed2b3Sopenharmony_ci * See the License for the specific language governing permissions and
13bc2ed2b3Sopenharmony_ci * limitations under the License.
14bc2ed2b3Sopenharmony_ci */
15bc2ed2b3Sopenharmony_ci#include "nfc_watch_dog.h"
16bc2ed2b3Sopenharmony_ci#include <chrono>
17bc2ed2b3Sopenharmony_ci#include "loghelper.h"
18bc2ed2b3Sopenharmony_ci#include "external_deps_proxy.h"
19bc2ed2b3Sopenharmony_ci
20bc2ed2b3Sopenharmony_cinamespace OHOS {
21bc2ed2b3Sopenharmony_cinamespace NFC {
22bc2ed2b3Sopenharmony_ciNfcWatchDog::NfcWatchDog(const std::string& threadName, int timeout, std::weak_ptr<NCI::INciNfccInterface> nfccProxy)
23bc2ed2b3Sopenharmony_ci    : threadName_(threadName), timeout_(timeout), canceled_(false), thread_(nullptr), nciNfccProxy_(nfccProxy)
24bc2ed2b3Sopenharmony_ci{
25bc2ed2b3Sopenharmony_ci}
26bc2ed2b3Sopenharmony_ci
27bc2ed2b3Sopenharmony_ciNfcWatchDog::~NfcWatchDog()
28bc2ed2b3Sopenharmony_ci{
29bc2ed2b3Sopenharmony_ci    if (thread_ && thread_->joinable()) {
30bc2ed2b3Sopenharmony_ci        conditionVariable_.notify_one();
31bc2ed2b3Sopenharmony_ci        thread_->join();
32bc2ed2b3Sopenharmony_ci    }
33bc2ed2b3Sopenharmony_ci}
34bc2ed2b3Sopenharmony_ci
35bc2ed2b3Sopenharmony_civoid NfcWatchDog::MainLoop()
36bc2ed2b3Sopenharmony_ci{
37bc2ed2b3Sopenharmony_ci    std::unique_lock<std::mutex> lock(mutex_);
38bc2ed2b3Sopenharmony_ci    InfoLog("Watchdog [%{public}s] starts to run.", threadName_.c_str());
39bc2ed2b3Sopenharmony_ci    conditionVariable_.wait_for(lock, std::chrono::milliseconds(timeout_), [this] { return canceled_; });
40bc2ed2b3Sopenharmony_ci    if (canceled_) {
41bc2ed2b3Sopenharmony_ci        return;
42bc2ed2b3Sopenharmony_ci    }
43bc2ed2b3Sopenharmony_ci    // If Routing Wake Lock is held, Routing Wake Lock release. Watchdog triggered, release lock before aborting.
44bc2ed2b3Sopenharmony_ci    if (nciNfccProxy_.expired()) {
45bc2ed2b3Sopenharmony_ci        return;
46bc2ed2b3Sopenharmony_ci    }
47bc2ed2b3Sopenharmony_ci    InfoLog("Watchdog triggered, aborting.");
48bc2ed2b3Sopenharmony_ci    NfcFailedParams err;
49bc2ed2b3Sopenharmony_ci    if (threadName_.compare("DoTurnOn") == 0) {
50bc2ed2b3Sopenharmony_ci        ExternalDepsProxy::GetInstance().BuildFailedParams(err, MainErrorCode::NFC_OPEN_FAILED,
51bc2ed2b3Sopenharmony_ci            SubErrorCode::PROCESS_ABORT);
52bc2ed2b3Sopenharmony_ci    } else if (threadName_.compare("DoTurnOff") == 0) {
53bc2ed2b3Sopenharmony_ci        ExternalDepsProxy::GetInstance().BuildFailedParams(err, MainErrorCode::NFC_CLOSE_FAILED,
54bc2ed2b3Sopenharmony_ci            SubErrorCode::PROCESS_ABORT);
55bc2ed2b3Sopenharmony_ci    } else {
56bc2ed2b3Sopenharmony_ci        ExternalDepsProxy::GetInstance().BuildFailedParams(err, MainErrorCode::NFC_GENERAL_ERR,
57bc2ed2b3Sopenharmony_ci            SubErrorCode::PROCESS_ABORT);
58bc2ed2b3Sopenharmony_ci    }
59bc2ed2b3Sopenharmony_ci    ExternalDepsProxy::GetInstance().WriteNfcFailedHiSysEvent(&err);
60bc2ed2b3Sopenharmony_ci    nciNfccProxy_.lock()->Abort();
61bc2ed2b3Sopenharmony_ci}
62bc2ed2b3Sopenharmony_ci
63bc2ed2b3Sopenharmony_civoid NfcWatchDog::Run()
64bc2ed2b3Sopenharmony_ci{
65bc2ed2b3Sopenharmony_ci    thread_ = std::make_unique<std::thread>([this]() { this->MainLoop(); });
66bc2ed2b3Sopenharmony_ci}
67bc2ed2b3Sopenharmony_ci
68bc2ed2b3Sopenharmony_civoid NfcWatchDog::Cancel()
69bc2ed2b3Sopenharmony_ci{
70bc2ed2b3Sopenharmony_ci    std::unique_lock<std::mutex> lock(mutex_);
71bc2ed2b3Sopenharmony_ci    canceled_ = true;
72bc2ed2b3Sopenharmony_ci    conditionVariable_.notify_one();
73bc2ed2b3Sopenharmony_ci}
74bc2ed2b3Sopenharmony_ci}  // namespace NFC
75bc2ed2b3Sopenharmony_ci}  // namespace OHOS
76