Fix MMIO handler race condition by rechecking page access protections under the global lock.

This commit is contained in:
Dr. Chat 2015-10-22 20:18:08 -05:00
parent 88be0a362c
commit 407d79cf3e
1 changed files with 29 additions and 27 deletions

View File

@ -116,33 +116,27 @@ uintptr_t MMIOHandler::AddPhysicalWriteWatch(uint32_t guest_address,
global_critical_region_.mutex().unlock(); global_critical_region_.mutex().unlock();
// Make the desired range read only under all address spaces. // Make the desired range read only under all address spaces.
xe::memory::Protect(physical_membase_ + entry->address, entry->length, memory::Protect(physical_membase_ + entry->address, entry->length,
xe::memory::PageAccess::kReadOnly, nullptr); xe::memory::PageAccess::kReadOnly, nullptr);
xe::memory::Protect(virtual_membase_ + 0xA0000000 + entry->address, memory::Protect(virtual_membase_ + 0xA0000000 + entry->address, entry->length,
entry->length, xe::memory::PageAccess::kReadOnly, xe::memory::PageAccess::kReadOnly, nullptr);
nullptr); memory::Protect(virtual_membase_ + 0xC0000000 + entry->address, entry->length,
xe::memory::Protect(virtual_membase_ + 0xC0000000 + entry->address, xe::memory::PageAccess::kReadOnly, nullptr);
entry->length, xe::memory::PageAccess::kReadOnly, memory::Protect(virtual_membase_ + 0xE0000000 + entry->address, entry->length,
nullptr); xe::memory::PageAccess::kReadOnly, nullptr);
xe::memory::Protect(virtual_membase_ + 0xE0000000 + entry->address,
entry->length, xe::memory::PageAccess::kReadOnly,
nullptr);
return reinterpret_cast<uintptr_t>(entry); return reinterpret_cast<uintptr_t>(entry);
} }
void MMIOHandler::ClearWriteWatch(WriteWatchEntry* entry) { void MMIOHandler::ClearWriteWatch(WriteWatchEntry* entry) {
xe::memory::Protect(physical_membase_ + entry->address, entry->length, memory::Protect(physical_membase_ + entry->address, entry->length,
xe::memory::PageAccess::kReadWrite, nullptr); xe::memory::PageAccess::kReadWrite, nullptr);
xe::memory::Protect(virtual_membase_ + 0xA0000000 + entry->address, memory::Protect(virtual_membase_ + 0xA0000000 + entry->address, entry->length,
entry->length, xe::memory::PageAccess::kReadWrite, xe::memory::PageAccess::kReadWrite, nullptr);
nullptr); memory::Protect(virtual_membase_ + 0xC0000000 + entry->address, entry->length,
xe::memory::Protect(virtual_membase_ + 0xC0000000 + entry->address, xe::memory::PageAccess::kReadWrite, nullptr);
entry->length, xe::memory::PageAccess::kReadWrite, memory::Protect(virtual_membase_ + 0xE0000000 + entry->address, entry->length,
nullptr); xe::memory::PageAccess::kReadWrite, nullptr);
xe::memory::Protect(virtual_membase_ + 0xE0000000 + entry->address,
entry->length, xe::memory::PageAccess::kReadWrite,
nullptr);
} }
void MMIOHandler::CancelWriteWatch(uintptr_t watch_handle) { void MMIOHandler::CancelWriteWatch(uintptr_t watch_handle) {
@ -170,17 +164,25 @@ bool MMIOHandler::CheckWriteWatch(X64Context* thread_context,
} }
std::list<WriteWatchEntry*> pending_invalidates; std::list<WriteWatchEntry*> pending_invalidates;
global_critical_region_.mutex().lock(); global_critical_region_.mutex().lock();
// Now that we hold the lock, recheck and see if the pages are still protected.
memory::PageAccess cur_access;
size_t page_length = memory::page_size();
memory::QueryProtect((void*)fault_address, page_length, cur_access);
if (cur_access != memory::PageAccess::kReadOnly &&
cur_access != memory::PageAccess::kNoAccess) {
// Another thread has cleared this write watch. Abort.
return true;
}
for (auto it = write_watches_.begin(); it != write_watches_.end();) { for (auto it = write_watches_.begin(); it != write_watches_.end();) {
auto entry = *it; auto entry = *it;
if (entry->address <= physical_address && if (entry->address <= physical_address &&
entry->address + entry->length > physical_address) { entry->address + entry->length > physical_address) {
// Hit! // Hit! Remove the writewatch.
pending_invalidates.push_back(entry); pending_invalidates.push_back(entry);
// TODO(benvanik): outside of lock?
ClearWriteWatch(entry); ClearWriteWatch(entry);
auto erase_it = it; it = write_watches_.erase(it);
++it;
write_watches_.erase(erase_it);
continue; continue;
} }
++it; ++it;