137 lines
3.4 KiB
C++
137 lines
3.4 KiB
C++
//
|
|
// detail/impl/win_static_mutex.ipp
|
|
// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
//
|
|
// Copyright (c) 2003-2021 Christopher M. Kohlhoff (chris at kohlhoff dot com)
|
|
//
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
//
|
|
|
|
#ifndef ASIO_DETAIL_IMPL_WIN_STATIC_MUTEX_IPP
|
|
#define ASIO_DETAIL_IMPL_WIN_STATIC_MUTEX_IPP
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER >= 1200)
|
|
# pragma once
|
|
#endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
|
|
|
|
#include "asio/detail/config.hpp"
|
|
|
|
#if defined(ASIO_WINDOWS)
|
|
|
|
#include <cstdio>
|
|
#include "asio/detail/throw_error.hpp"
|
|
#include "asio/detail/win_static_mutex.hpp"
|
|
#include "asio/error.hpp"
|
|
|
|
#include "asio/detail/push_options.hpp"
|
|
|
|
namespace asio {
|
|
namespace detail {
|
|
|
|
void win_static_mutex::init()
|
|
{
|
|
int error = do_init();
|
|
asio::error_code ec(error,
|
|
asio::error::get_system_category());
|
|
asio::detail::throw_error(ec, "static_mutex");
|
|
}
|
|
|
|
int win_static_mutex::do_init()
|
|
{
|
|
using namespace std; // For sprintf.
|
|
wchar_t mutex_name[128];
|
|
#if defined(ASIO_HAS_SECURE_RTL)
|
|
swprintf_s(
|
|
#else // defined(ASIO_HAS_SECURE_RTL)
|
|
_snwprintf(
|
|
#endif // defined(ASIO_HAS_SECURE_RTL)
|
|
mutex_name, 128, L"asio-58CCDC44-6264-4842-90C2-F3C545CB8AA7-%u-%p",
|
|
static_cast<unsigned int>(::GetCurrentProcessId()), this);
|
|
|
|
#if defined(ASIO_WINDOWS_APP)
|
|
HANDLE mutex = ::CreateMutexExW(0, mutex_name, CREATE_MUTEX_INITIAL_OWNER, 0);
|
|
#else // defined(ASIO_WINDOWS_APP)
|
|
HANDLE mutex = ::CreateMutexW(0, TRUE, mutex_name);
|
|
#endif // defined(ASIO_WINDOWS_APP)
|
|
DWORD last_error = ::GetLastError();
|
|
if (mutex == 0)
|
|
return ::GetLastError();
|
|
|
|
if (last_error == ERROR_ALREADY_EXISTS)
|
|
{
|
|
#if defined(ASIO_WINDOWS_APP)
|
|
::WaitForSingleObjectEx(mutex, INFINITE, false);
|
|
#else // defined(ASIO_WINDOWS_APP)
|
|
::WaitForSingleObject(mutex, INFINITE);
|
|
#endif // defined(ASIO_WINDOWS_APP)
|
|
}
|
|
|
|
if (initialised_)
|
|
{
|
|
::ReleaseMutex(mutex);
|
|
::CloseHandle(mutex);
|
|
return 0;
|
|
}
|
|
|
|
#if defined(__MINGW32__)
|
|
// Not sure if MinGW supports structured exception handling, so for now
|
|
// we'll just call the Windows API and hope.
|
|
# if defined(UNDER_CE)
|
|
::InitializeCriticalSection(&crit_section_);
|
|
# else
|
|
if (!::InitializeCriticalSectionAndSpinCount(&crit_section_, 0x80000000))
|
|
{
|
|
last_error = ::GetLastError();
|
|
::ReleaseMutex(mutex);
|
|
::CloseHandle(mutex);
|
|
return last_error;
|
|
}
|
|
# endif
|
|
#else
|
|
__try
|
|
{
|
|
# if defined(UNDER_CE)
|
|
::InitializeCriticalSection(&crit_section_);
|
|
# elif defined(ASIO_WINDOWS_APP)
|
|
if (!::InitializeCriticalSectionEx(&crit_section_, 0, 0))
|
|
{
|
|
last_error = ::GetLastError();
|
|
::ReleaseMutex(mutex);
|
|
::CloseHandle(mutex);
|
|
return last_error;
|
|
}
|
|
# else
|
|
if (!::InitializeCriticalSectionAndSpinCount(&crit_section_, 0x80000000))
|
|
{
|
|
last_error = ::GetLastError();
|
|
::ReleaseMutex(mutex);
|
|
::CloseHandle(mutex);
|
|
return last_error;
|
|
}
|
|
# endif
|
|
}
|
|
__except(GetExceptionCode() == STATUS_NO_MEMORY
|
|
? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH)
|
|
{
|
|
::ReleaseMutex(mutex);
|
|
::CloseHandle(mutex);
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
#endif
|
|
|
|
initialised_ = true;
|
|
::ReleaseMutex(mutex);
|
|
::CloseHandle(mutex);
|
|
return 0;
|
|
}
|
|
|
|
} // namespace detail
|
|
} // namespace asio
|
|
|
|
#include "asio/detail/pop_options.hpp"
|
|
|
|
#endif // defined(ASIO_WINDOWS)
|
|
|
|
#endif // ASIO_DETAIL_IMPL_WIN_STATIC_MUTEX_IPP
|