Kernel: Cleanup NVMe Queue command submission
There is techically a race condition on thread sleep and checking done mask. This patch allows read to success even if this race condition is hit, although the full timeout has to be waited. This can be fixed in future with some sort of wait queues that can properly handle this race condition.
This commit is contained in:
parent
7f5c850744
commit
62f6128ba1
|
@ -67,24 +67,20 @@ namespace Kernel
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint64_t start_time = SystemTimer::get().ms_since_boot();
|
const uint64_t start_time = SystemTimer::get().ms_since_boot();
|
||||||
while (SystemTimer::get().ms_since_boot() < start_time + s_nvme_command_poll_timeout_ms)
|
while (!(m_done_mask & cid_mask) && SystemTimer::get().ms_since_boot() < start_time + s_nvme_command_poll_timeout_ms)
|
||||||
{
|
continue;
|
||||||
if (m_done_mask & cid_mask)
|
|
||||||
{
|
|
||||||
uint16_t status = m_status_codes[cid];
|
|
||||||
m_used_mask &= ~cid_mask;
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (SystemTimer::get().ms_since_boot() < start_time + s_nvme_command_timeout_ms)
|
// FIXME: Here is a possible race condition if done mask is set before
|
||||||
|
// scheduler has put the current thread blocking.
|
||||||
|
// EINTR should also be handled here.
|
||||||
|
while (!(m_done_mask & cid_mask) && SystemTimer::get().ms_since_boot() < start_time + s_nvme_command_timeout_ms)
|
||||||
|
Scheduler::get().block_current_thread(&m_semaphore, start_time + s_nvme_command_timeout_ms);
|
||||||
|
|
||||||
|
if (m_done_mask & cid_mask)
|
||||||
{
|
{
|
||||||
if (m_done_mask & cid_mask)
|
uint16_t status = m_status_codes[cid];
|
||||||
{
|
m_used_mask &= ~cid_mask;
|
||||||
uint16_t status = m_status_codes[cid];
|
return status;
|
||||||
m_used_mask &= ~cid_mask;
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
m_used_mask &= ~cid_mask;
|
m_used_mask &= ~cid_mask;
|
||||||
|
|
Loading…
Reference in New Issue