blob: c664c8d53cf52d33e630c28cab7c8ed6b9ffdaf0 [file] [log] [blame]
// Copyright 2020 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/updater/prefs_impl.h"
#include <windows.h>
#include <memory>
#include "base/logging.h"
#include "base/time/time.h"
#include "base/win/scoped_handle.h"
#include "chrome/updater/win/constants.h"
#include "chrome/updater/win/util.h"
namespace updater {
class ScopedPrefsLockImpl {
public:
ScopedPrefsLockImpl() = default;
ScopedPrefsLockImpl(const ScopedPrefsLockImpl&) = delete;
ScopedPrefsLockImpl& operator=(const ScopedPrefsLockImpl&) = delete;
~ScopedPrefsLockImpl();
bool Initialize(bool is_machine, base::TimeDelta timeout);
private:
base::win::ScopedHandle mutex_;
};
ScopedPrefsLock::ScopedPrefsLock(std::unique_ptr<ScopedPrefsLockImpl> impl)
: impl_(std::move(impl)) {}
ScopedPrefsLock::~ScopedPrefsLock() = default;
std::unique_ptr<ScopedPrefsLock> AcquireGlobalPrefsLock(
base::TimeDelta timeout) {
auto lock = std::make_unique<ScopedPrefsLockImpl>();
// TODO(crbug.com/1096654): need to pass is_machine instead of 'false' here.
DVLOG(2) << "Trying to acquire the lock.";
if (!lock->Initialize(false, timeout))
return nullptr;
DVLOG(2) << "Lock acquired.";
return std::make_unique<ScopedPrefsLock>(std::move(lock));
}
bool ScopedPrefsLockImpl::Initialize(bool is_machine, base::TimeDelta timeout) {
NamedObjectAttributes lock_attr;
GetNamedObjectAttributes(kPrefsAccessMutex, is_machine, &lock_attr);
mutex_.Set(::CreateMutex(&lock_attr.sa, false, lock_attr.name.c_str()));
if (!mutex_.IsValid())
return false;
DWORD ret = ::WaitForSingleObject(mutex_.Get(), timeout.InMilliseconds());
return ret == WAIT_OBJECT_0 || ret == WAIT_ABANDONED;
}
ScopedPrefsLockImpl::~ScopedPrefsLockImpl() {
if (mutex_.IsValid()) {
::ReleaseMutex(mutex_.Get());
DVLOG(2) << "Lock released.";
}
}
} // namespace updater