mirror of
https://github.com/Fennix-Project/Kernel.git
synced 2025-05-29 15:57:59 +00:00
Added timeout lock
This commit is contained in:
parent
98b797a95c
commit
88008ac470
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
#include <smp.hpp>
|
#include <smp.hpp>
|
||||||
|
#include <atomic.hpp>
|
||||||
|
|
||||||
#include "../kernel.h"
|
#include "../kernel.h"
|
||||||
|
|
||||||
@ -22,9 +23,7 @@ void LockClass::DeadLock(SpinLockData Lock)
|
|||||||
|
|
||||||
if (Config.UnlockDeadLock && this->DeadLocks > 10)
|
if (Config.UnlockDeadLock && this->DeadLocks > 10)
|
||||||
{
|
{
|
||||||
warn("Unlocking lock '%s' held by '%s'! %ld locks in queue. Core %ld is being held by %ld.",
|
warn("Unlocking lock '%s' to prevent deadlock. (this is enabled in the kernel config)", Lock.AttemptingToGet);
|
||||||
Lock.AttemptingToGet, Lock.CurrentHolder,
|
|
||||||
Lock.Count, CCore, Lock.Core);
|
|
||||||
this->DeadLocks = 0;
|
this->DeadLocks = 0;
|
||||||
this->Unlock();
|
this->Unlock();
|
||||||
}
|
}
|
||||||
@ -85,3 +84,52 @@ int LockClass::Unlock()
|
|||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void LockClass::TimeoutDeadLock(SpinLockData Lock, uint64_t Timeout)
|
||||||
|
{
|
||||||
|
CPUData *CoreData = GetCurrentCPU();
|
||||||
|
long CCore = 0xdead;
|
||||||
|
if (CoreData != nullptr)
|
||||||
|
CCore = CoreData->ID;
|
||||||
|
warn("Potential deadlock in lock '%s' held by '%s'! %ld locks in queue. Core %ld is being held by %ld. Timeout in %ld",
|
||||||
|
Lock.AttemptingToGet, Lock.CurrentHolder,
|
||||||
|
Lock.Count, CCore, Lock.Core,
|
||||||
|
Timeout);
|
||||||
|
|
||||||
|
// TODO: Print on screen too.
|
||||||
|
|
||||||
|
uint64_t Counter = TimeManager->GetCounter();
|
||||||
|
if (Timeout < Counter)
|
||||||
|
{
|
||||||
|
warn("Unlocking lock '%s' because of timeout. (%ld < %ld)", Lock.AttemptingToGet, Timeout, Counter);
|
||||||
|
this->Unlock();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (TaskManager)
|
||||||
|
TaskManager->Schedule();
|
||||||
|
}
|
||||||
|
|
||||||
|
int LockClass::TimeoutLock(const char *FunctionName, uint64_t Timeout)
|
||||||
|
{
|
||||||
|
LockData.AttemptingToGet = FunctionName;
|
||||||
|
Atomic<uint64_t> Target = 0;
|
||||||
|
Retry:
|
||||||
|
unsigned int i = 0;
|
||||||
|
while (__atomic_exchange_n(&IsLocked, true, __ATOMIC_ACQUIRE) && ++i < 0x10000000)
|
||||||
|
CPU::Pause();
|
||||||
|
if (i >= 0x10000000)
|
||||||
|
{
|
||||||
|
if (Target.Load() == 0)
|
||||||
|
Target = TimeManager->CalculateTarget(Timeout);
|
||||||
|
TimeoutDeadLock(LockData, Target.Load());
|
||||||
|
goto Retry;
|
||||||
|
}
|
||||||
|
LockData.Count++;
|
||||||
|
LockData.CurrentHolder = FunctionName;
|
||||||
|
CPUData *CoreData = GetCurrentCPU();
|
||||||
|
if (CoreData != nullptr)
|
||||||
|
LockData.Core = CoreData->ID;
|
||||||
|
__sync_synchronize();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
#include <cpu.hpp>
|
#include <cpu.hpp>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
|
|
||||||
/** @brief Please use this macro to create a new lock. */
|
/** @brief Please use this macro to create a new lock. */
|
||||||
class LockClass
|
class LockClass
|
||||||
{
|
{
|
||||||
@ -16,18 +17,23 @@ class LockClass
|
|||||||
size_t Count = 0;
|
size_t Count = 0;
|
||||||
long Core = 0;
|
long Core = 0;
|
||||||
};
|
};
|
||||||
void DeadLock(SpinLockData Lock);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
SpinLockData LockData;
|
SpinLockData LockData;
|
||||||
bool IsLocked = false;
|
bool IsLocked = false;
|
||||||
unsigned long DeadLocks = 0;
|
unsigned long DeadLocks = 0;
|
||||||
|
|
||||||
|
void DeadLock(SpinLockData Lock);
|
||||||
|
void TimeoutDeadLock(SpinLockData Lock, uint64_t Timeout);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SpinLockData *GetLockData() { return &LockData; }
|
SpinLockData *GetLockData() { return &LockData; }
|
||||||
int Lock(const char *FunctionName);
|
int Lock(const char *FunctionName);
|
||||||
int Unlock();
|
int Unlock();
|
||||||
|
|
||||||
|
int TimeoutLock(const char *FunctionName, uint64_t Timeout);
|
||||||
};
|
};
|
||||||
|
|
||||||
/** @brief Please use this macro to create a new smart lock. */
|
/** @brief Please use this macro to create a new smart lock. */
|
||||||
class SmartLockClass
|
class SmartLockClass
|
||||||
{
|
{
|
||||||
@ -42,6 +48,21 @@ public:
|
|||||||
}
|
}
|
||||||
~SmartLockClass() { this->LockPointer->Unlock(); }
|
~SmartLockClass() { this->LockPointer->Unlock(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class SmartTimeoutLockClass
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
LockClass *LockPointer = nullptr;
|
||||||
|
|
||||||
|
public:
|
||||||
|
SmartTimeoutLockClass(LockClass &Lock, const char *FunctionName, uint64_t Timeout)
|
||||||
|
{
|
||||||
|
this->LockPointer = &Lock;
|
||||||
|
this->LockPointer->TimeoutLock(FunctionName, Timeout);
|
||||||
|
}
|
||||||
|
~SmartTimeoutLockClass() { this->LockPointer->Unlock(); }
|
||||||
|
};
|
||||||
|
|
||||||
/** @brief Please use this macro to create a new smart critical section lock. */
|
/** @brief Please use this macro to create a new smart critical section lock. */
|
||||||
class SmartLockCriticalSectionClass
|
class SmartLockCriticalSectionClass
|
||||||
{
|
{
|
||||||
@ -65,6 +86,7 @@ public:
|
|||||||
CPU::Interrupts(CPU::Enable);
|
CPU::Interrupts(CPU::Enable);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/** @brief Please use this macro to create a new critical section. */
|
/** @brief Please use this macro to create a new critical section. */
|
||||||
class SmartCriticalSectionClass
|
class SmartCriticalSectionClass
|
||||||
{
|
{
|
||||||
@ -89,10 +111,16 @@ public:
|
|||||||
|
|
||||||
/** @brief Create a new lock (can be used with SmartCriticalSection). */
|
/** @brief Create a new lock (can be used with SmartCriticalSection). */
|
||||||
#define NewLock(Name) LockClass Name
|
#define NewLock(Name) LockClass Name
|
||||||
|
|
||||||
/** @brief Simple lock that is automatically released when the scope ends. */
|
/** @brief Simple lock that is automatically released when the scope ends. */
|
||||||
#define SmartLock(LockClassName) SmartLockClass CONCAT(lock##_, __COUNTER__)(LockClassName, __FUNCTION__)
|
#define SmartLock(LockClassName) SmartLockClass CONCAT(lock##_, __COUNTER__)(LockClassName, __FUNCTION__)
|
||||||
|
|
||||||
|
/** @brief Simple lock with timeout that is automatically released when the scope ends. */
|
||||||
|
#define SmartTimeoutLock(LockClassName, Timeout) SmartTimeoutLockClass CONCAT(lock##_, __COUNTER__)(LockClassName, __FUNCTION__, Timeout)
|
||||||
|
|
||||||
/** @brief Simple critical section that is automatically released when the scope ends and interrupts are restored if they were enabled. */
|
/** @brief Simple critical section that is automatically released when the scope ends and interrupts are restored if they were enabled. */
|
||||||
#define SmartCriticalSection(LockClassName) SmartLockCriticalSectionClass CONCAT(lock##_, __COUNTER__)(LockClassName, __FUNCTION__)
|
#define SmartCriticalSection(LockClassName) SmartLockCriticalSectionClass CONCAT(lock##_, __COUNTER__)(LockClassName, __FUNCTION__)
|
||||||
|
|
||||||
/** @brief Automatically disable interrupts and restore them when the scope ends. */
|
/** @brief Automatically disable interrupts and restore them when the scope ends. */
|
||||||
#define CriticalSection SmartCriticalSectionClass
|
#define CriticalSection SmartCriticalSectionClass
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user