14514f5e3Sopenharmony_ci/*
24514f5e3Sopenharmony_ci * Copyright (c) 2024 Huawei Device Co., Ltd.
34514f5e3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
44514f5e3Sopenharmony_ci * you may not use this file except in compliance with the License.
54514f5e3Sopenharmony_ci * You may obtain a copy of the License at
64514f5e3Sopenharmony_ci *
74514f5e3Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
84514f5e3Sopenharmony_ci *
94514f5e3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
104514f5e3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
114514f5e3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
124514f5e3Sopenharmony_ci * See the License for the specific language governing permissions and
134514f5e3Sopenharmony_ci * limitations under the License.
144514f5e3Sopenharmony_ci */
154514f5e3Sopenharmony_ci
164514f5e3Sopenharmony_ci#include "ecmascript/mutator_lock.h"
174514f5e3Sopenharmony_ci#include "ecmascript/js_thread.h"
184514f5e3Sopenharmony_ci
194514f5e3Sopenharmony_cinamespace panda::ecmascript {
204514f5e3Sopenharmony_ci#ifndef NDEBUG
214514f5e3Sopenharmony_civoid MutatorLock::ReadLock()
224514f5e3Sopenharmony_ci{
234514f5e3Sopenharmony_ci    ASSERT(!HasLock());
244514f5e3Sopenharmony_ci    RWLock::ReadLock();
254514f5e3Sopenharmony_ci    SetState(MutatorLockState::RDLOCK);
264514f5e3Sopenharmony_ci}
274514f5e3Sopenharmony_ci
284514f5e3Sopenharmony_civoid MutatorLock::WriteLock()
294514f5e3Sopenharmony_ci{
304514f5e3Sopenharmony_ci    ASSERT(!HasLock());
314514f5e3Sopenharmony_ci    RWLock::WriteLock();
324514f5e3Sopenharmony_ci    SetState(MutatorLockState::WRLOCK);
334514f5e3Sopenharmony_ci}
344514f5e3Sopenharmony_ci
354514f5e3Sopenharmony_cibool MutatorLock::TryReadLock()
364514f5e3Sopenharmony_ci{
374514f5e3Sopenharmony_ci    ASSERT(!HasLock());
384514f5e3Sopenharmony_ci    bool res = RWLock::TryReadLock();
394514f5e3Sopenharmony_ci    if (res) {
404514f5e3Sopenharmony_ci        SetState(MutatorLockState::RDLOCK);
414514f5e3Sopenharmony_ci    }
424514f5e3Sopenharmony_ci    return res;
434514f5e3Sopenharmony_ci}
444514f5e3Sopenharmony_ci
454514f5e3Sopenharmony_cibool MutatorLock::TryWriteLock()
464514f5e3Sopenharmony_ci{
474514f5e3Sopenharmony_ci    ASSERT(!HasLock());
484514f5e3Sopenharmony_ci    bool res = RWLock::TryWriteLock();
494514f5e3Sopenharmony_ci    if (res) {
504514f5e3Sopenharmony_ci        SetState(MutatorLockState::WRLOCK);
514514f5e3Sopenharmony_ci    }
524514f5e3Sopenharmony_ci    return res;
534514f5e3Sopenharmony_ci}
544514f5e3Sopenharmony_ci
554514f5e3Sopenharmony_civoid MutatorLock::Unlock()
564514f5e3Sopenharmony_ci{
574514f5e3Sopenharmony_ci    ASSERT(HasLock());
584514f5e3Sopenharmony_ci    RWLock::Unlock();
594514f5e3Sopenharmony_ci    SetState(MutatorLockState::UNLOCKED);
604514f5e3Sopenharmony_ci}
614514f5e3Sopenharmony_ci
624514f5e3Sopenharmony_cibool MutatorLock::HasLock() const
634514f5e3Sopenharmony_ci{
644514f5e3Sopenharmony_ci    return GetState() != MutatorLockState::UNLOCKED;
654514f5e3Sopenharmony_ci}
664514f5e3Sopenharmony_ci
674514f5e3Sopenharmony_ciMutatorLock::MutatorLockState MutatorLock::GetState() const
684514f5e3Sopenharmony_ci{
694514f5e3Sopenharmony_ci    return JSThread::GetCurrent()->GetMutatorLockState();
704514f5e3Sopenharmony_ci}
714514f5e3Sopenharmony_ci
724514f5e3Sopenharmony_civoid MutatorLock::SetState(MutatorLock::MutatorLockState newState)
734514f5e3Sopenharmony_ci{
744514f5e3Sopenharmony_ci    JSThread::GetCurrent()->SetMutatorLockState(newState);
754514f5e3Sopenharmony_ci}
764514f5e3Sopenharmony_ci#endif
774514f5e3Sopenharmony_ci
784514f5e3Sopenharmony_civoid SuspendBarrier::Wait()
794514f5e3Sopenharmony_ci{
804514f5e3Sopenharmony_ci    while (true) {
814514f5e3Sopenharmony_ci        int32_t curCount = passBarrierCount_.load(std::memory_order_relaxed);
824514f5e3Sopenharmony_ci        if (LIKELY(curCount > 0)) {
834514f5e3Sopenharmony_ci#if defined(PANDA_TARGET_OHOS)
844514f5e3Sopenharmony_ci            sched_yield();
854514f5e3Sopenharmony_ci#endif
864514f5e3Sopenharmony_ci        } else {
874514f5e3Sopenharmony_ci            // Use seq_cst to synchronize memory.
884514f5e3Sopenharmony_ci            curCount = passBarrierCount_.load(std::memory_order_seq_cst);
894514f5e3Sopenharmony_ci            ASSERT(curCount == 0);
904514f5e3Sopenharmony_ci            break;
914514f5e3Sopenharmony_ci        }
924514f5e3Sopenharmony_ci    }
934514f5e3Sopenharmony_ci}
944514f5e3Sopenharmony_ci
954514f5e3Sopenharmony_civoid SuspendBarrier::PassCount(int32_t delta)
964514f5e3Sopenharmony_ci{
974514f5e3Sopenharmony_ci    bool done = false;
984514f5e3Sopenharmony_ci    do {
994514f5e3Sopenharmony_ci        int32_t curCount = passBarrierCount_.load(std::memory_order_relaxed);
1004514f5e3Sopenharmony_ci        // Reduce value by 1.
1014514f5e3Sopenharmony_ci        done = passBarrierCount_.compare_exchange_strong(curCount, curCount + delta);
1024514f5e3Sopenharmony_ci    } while (!done);
1034514f5e3Sopenharmony_ci}
1044514f5e3Sopenharmony_ci}  // namespace panda::ecmascript