2021-10-02 10:28:49 +00:00
|
|
|
/***
|
|
|
|
Copyright (C) 2021 J Reece Wilson (a/k/a "Reece"). All rights reserved.
|
|
|
|
|
|
|
|
File: LSMutex.Linux.cpp
|
|
|
|
Date: 2021-10-1
|
2022-04-05 01:19:37 +00:00
|
|
|
Date: 2022-4-4
|
2021-10-02 10:28:49 +00:00
|
|
|
Author: Reece
|
|
|
|
***/
|
|
|
|
#include <Source/RuntimeInternal.hpp>
|
|
|
|
#include "LSMutex.hpp"
|
2022-04-05 02:36:39 +00:00
|
|
|
#include "LSFromFdNonblocking.hpp"
|
2021-10-02 10:28:49 +00:00
|
|
|
|
2022-06-11 23:52:46 +00:00
|
|
|
namespace Aurora::IO::Loop
|
2021-10-02 10:28:49 +00:00
|
|
|
{
|
2022-04-05 01:19:37 +00:00
|
|
|
LSMutex::LSMutex()
|
|
|
|
{
|
2022-04-07 01:20:46 +00:00
|
|
|
Init();
|
|
|
|
}
|
|
|
|
|
2022-04-15 10:01:43 +00:00
|
|
|
LSMutex::LSMutex(int handle)
|
|
|
|
{
|
|
|
|
this->handle = handle;
|
|
|
|
}
|
|
|
|
|
2022-04-07 01:20:46 +00:00
|
|
|
LSMutex::~LSMutex()
|
|
|
|
{
|
2022-08-07 04:18:34 +00:00
|
|
|
if ((!this->bOwns) &&
|
|
|
|
(!this->bNoAutoRel /*must the remote grug clean up the IPC mutex? lets not hit a double unlock assert...*/))
|
|
|
|
{
|
|
|
|
SysPushErrorIO("Mutex owned by calling process during destruction... Forcefully unlocking...");
|
|
|
|
Unlock();
|
|
|
|
}
|
|
|
|
|
2022-04-07 01:20:46 +00:00
|
|
|
if ((this->handle != 0) &&
|
|
|
|
(this->handle != -1))
|
|
|
|
{
|
2022-04-15 10:01:43 +00:00
|
|
|
::close(AuExchange(this->handle, -1));
|
2022-04-07 01:20:46 +00:00
|
|
|
}
|
2022-04-05 01:19:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool LSMutex::OnTrigger(AuUInt handle)
|
|
|
|
{
|
2022-04-19 21:50:34 +00:00
|
|
|
return IsSignaledNonblocking();
|
2022-04-05 01:19:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LSMutex::Init()
|
|
|
|
{
|
2022-12-16 03:48:04 +00:00
|
|
|
handle = ::eventfd(1, EFD_NONBLOCK | EFD_CLOEXEC );
|
2022-04-05 01:19:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool LSMutex::Unlock()
|
|
|
|
{
|
|
|
|
AuUInt64 plsNoOverflow {1};
|
2022-08-07 04:18:34 +00:00
|
|
|
bool ok = ::write(this->handle, &plsNoOverflow, sizeof(plsNoOverflow)) == 8;
|
|
|
|
if (ok)
|
|
|
|
{
|
|
|
|
this->bOwns = false;
|
|
|
|
}
|
|
|
|
return ok;
|
2022-04-05 01:19:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool LSMutex::IsSignaled()
|
2022-04-05 02:36:39 +00:00
|
|
|
{
|
|
|
|
return IsSignaledFromNonblockingImpl(this, this, &LSMutex::IsSignaledNonblocking);
|
|
|
|
}
|
|
|
|
|
2022-04-12 21:26:15 +00:00
|
|
|
bool LSMutex::WaitOn(AuUInt32 timeout)
|
|
|
|
{
|
|
|
|
return LSHandle::WaitOn(timeout);
|
|
|
|
}
|
|
|
|
|
2022-04-05 02:36:39 +00:00
|
|
|
bool LSMutex::IsSignaledNonblocking()
|
2022-04-05 01:19:37 +00:00
|
|
|
{
|
|
|
|
AuUInt64 oldSemaphoreValue {};
|
2022-08-07 04:18:34 +00:00
|
|
|
auto read = ::read(this->handle, &oldSemaphoreValue, sizeof(oldSemaphoreValue));
|
|
|
|
auto ok = read == 8;
|
2022-04-05 01:19:37 +00:00
|
|
|
SysAssertDbg(!ok || oldSemaphoreValue == 1, "Double unlock caught");
|
2022-08-07 04:18:34 +00:00
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
// TODO: Might hook here
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this->bOwns = true;
|
|
|
|
}
|
2022-04-05 01:19:37 +00:00
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
ELoopSource LSMutex::GetType()
|
|
|
|
{
|
|
|
|
return ELoopSource::eSourceMutex;
|
|
|
|
}
|
|
|
|
|
2023-10-21 03:31:00 +00:00
|
|
|
#if 0
|
2022-04-05 01:19:37 +00:00
|
|
|
AUKN_SYM AuSPtr<ILSMutex> NewLSMutex()
|
|
|
|
{
|
2022-12-16 03:48:04 +00:00
|
|
|
auto pMutex = AuMakeShared<LSMutex>();
|
|
|
|
if (!pMutex)
|
2022-04-05 01:19:37 +00:00
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-12-16 03:48:04 +00:00
|
|
|
if (!pMutex->HasValidHandle())
|
2022-04-05 01:19:37 +00:00
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-12-16 03:48:04 +00:00
|
|
|
return pMutex;
|
2022-04-05 01:19:37 +00:00
|
|
|
}
|
2023-10-21 03:31:00 +00:00
|
|
|
#endif
|
2021-10-02 10:28:49 +00:00
|
|
|
}
|