Solve the static problem of user mode locking under high concurrency.

This commit is contained in:
zhushengle 2020-09-28 13:23:55 +08:00
parent aa407d3721
commit c884ada07d
1 changed files with 22 additions and 29 deletions

View File

@ -465,41 +465,29 @@ STATIC INT32 OsFindAndInsertToHash(FutexNode *node)
return ret; return ret;
} }
STATIC INT32 OsFutexWaitParmaCheck(const UINT32 *userVaddr, UINT32 flags, UINT32 val, UINT32 absTime) STATIC INT32 OsFutexWaitParamCheck(const UINT32 *userVaddr, UINT32 flags, UINT32 absTime)
{ {
UINTPTR futexKey = (UINTPTR)userVaddr; UINTPTR futexKey = (UINTPTR)userVaddr;
UINT32 lockVal;
INT32 ret;
if (OS_INT_ACTIVE) { if (OS_INT_ACTIVE) {
return LOS_EINTR; return LOS_EINTR;
} }
if (flags) { if (flags) {
PRINT_ERR("Futex wait parma check failed! error flags: 0x%x\n", flags); PRINT_ERR("Futex wait param check failed! error flags: 0x%x\n", flags);
return LOS_EINVAL; return LOS_EINVAL;
} }
if ((futexKey % sizeof(INT32)) || (futexKey < OS_FUTEX_KEY_BASE) || (futexKey >= OS_FUTEX_KEY_MAX)) { if ((futexKey % sizeof(INT32)) || (futexKey < OS_FUTEX_KEY_BASE) || (futexKey >= OS_FUTEX_KEY_MAX)) {
PRINT_ERR("Futex wait parma check failed! error futex key: 0x%x\n", futexKey); PRINT_ERR("Futex wait param check failed! error futex key: 0x%x\n", futexKey);
return LOS_EINVAL; return LOS_EINVAL;
} }
if (!absTime) { if (!absTime) {
PRINT_ERR("Futex wait parma check failed! error absTime: %u\n", absTime); PRINT_ERR("Futex wait param check failed! error absTime: %u\n", absTime);
return LOS_EINVAL; return LOS_EINVAL;
} }
ret = LOS_ArchCopyFromUser(&lockVal, userVaddr, sizeof(UINT32));
if (ret) {
PRINT_ERR("Futex wait parma check failed! copy from user failed!\n");
return LOS_EINVAL;
}
if (lockVal != val) {
return LOS_EBADF;
}
return LOS_OK; return LOS_OK;
} }
@ -542,10 +530,10 @@ STATIC INT32 OsFutexInserTaskToHash(LosTaskCB **taskCB, FutexNode **node, const
return LOS_OK; return LOS_OK;
} }
STATIC INT32 OsFutexWaitTask(const UINT32 timeOut, const UINT32 *userVaddr) STATIC INT32 OsFutexWaitTask(const UINT32 timeOut, const UINT32 *userVaddr, UINT32 val)
{ {
INT32 futexRet; INT32 futexRet;
UINT32 intSave; UINT32 intSave, lockVal;
LosTaskCB *taskCB = NULL; LosTaskCB *taskCB = NULL;
FutexNode *node = NULL; FutexNode *node = NULL;
UINTPTR futexKey = (UINTPTR)userVaddr; UINTPTR futexKey = (UINTPTR)userVaddr;
@ -556,7 +544,19 @@ STATIC INT32 OsFutexWaitTask(const UINT32 timeOut, const UINT32 *userVaddr)
return LOS_EINVAL; return LOS_EINVAL;
} }
if (LOS_ArchCopyFromUser(&lockVal, userVaddr, sizeof(UINT32))) {
PRINT_ERR("Futex wait param check failed! copy from user failed!\n");
futexRet = LOS_EINVAL;
goto EXIT_ERR;
}
if (lockVal != val) {
futexRet = LOS_EBADF;
goto EXIT_ERR;
}
if (OsFutexInserTaskToHash(&taskCB, &node, futexKey)) { if (OsFutexInserTaskToHash(&taskCB, &node, futexKey)) {
futexRet = LOS_NOK;
goto EXIT_ERR; goto EXIT_ERR;
} }
SCHEDULER_LOCK(intSave); SCHEDULER_LOCK(intSave);
@ -564,10 +564,6 @@ STATIC INT32 OsFutexWaitTask(const UINT32 timeOut, const UINT32 *userVaddr)
OsPercpuGet()->taskLockCnt++; OsPercpuGet()->taskLockCnt++;
LOS_SpinUnlock(&g_taskSpin); LOS_SpinUnlock(&g_taskSpin);
#ifdef LOS_FUTEX_DEBUG
OsFutexHashShow();
#endif
futexRet = OsFutexUnlock(&hashNode->listLock); futexRet = OsFutexUnlock(&hashNode->listLock);
if (futexRet) { if (futexRet) {
OsPercpuGet()->taskLockCnt--; OsPercpuGet()->taskLockCnt--;
@ -594,20 +590,17 @@ STATIC INT32 OsFutexWaitTask(const UINT32 timeOut, const UINT32 *userVaddr)
return LOS_OK; return LOS_OK;
EXIT_ERR: EXIT_ERR:
futexRet = OsFutexUnlock(&hashNode->listLock); (VOID)OsFutexUnlock(&hashNode->listLock);
EXIT_UNLOCK_ERR: EXIT_UNLOCK_ERR:
if (futexRet) {
return futexRet; return futexRet;
} }
return LOS_NOK;
}
INT32 OsFutexWait(const UINT32 *userVaddr, UINT32 flags, UINT32 val, UINT32 absTime) INT32 OsFutexWait(const UINT32 *userVaddr, UINT32 flags, UINT32 val, UINT32 absTime)
{ {
INT32 ret; INT32 ret;
UINT32 timeOut = LOS_WAIT_FOREVER; UINT32 timeOut = LOS_WAIT_FOREVER;
ret = OsFutexWaitParmaCheck(userVaddr, flags, val, absTime); ret = OsFutexWaitParamCheck(userVaddr, flags, absTime);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -615,7 +608,7 @@ INT32 OsFutexWait(const UINT32 *userVaddr, UINT32 flags, UINT32 val, UINT32 absT
timeOut = OsFutexGetTick(absTime); timeOut = OsFutexGetTick(absTime);
} }
return OsFutexWaitTask(timeOut, userVaddr); return OsFutexWaitTask(timeOut, userVaddr, val);
} }
/* Check to see if the task to be awakened has timed out /* Check to see if the task to be awakened has timed out