mmc_bus: fix possible deadlock when initialization fails

Should fix the remaining part of #16778

Change-Id: Ia148f4bde095d33752df788b8f7911ee9cf44c61
Reviewed-on: https://review.haiku-os.org/c/haiku/+/3773
Reviewed-by: Adrien Destugues <pulkomandy@gmail.com>
This commit is contained in:
Adrien Destugues 2021-03-12 14:31:32 +01:00 committed by Adrien Destugues
parent 8897bed702
commit a59f6b7a8c
2 changed files with 18 additions and 7 deletions

View File

@ -49,16 +49,20 @@ MMCBus::~MMCBus()
{ {
CALLED(); CALLED();
// stop worker thread // Tell the worker thread we want to stop
fStatus = B_SHUTTING_DOWN; fStatus = B_SHUTTING_DOWN;
// Delete the semaphores (this will unlock the worker thread if it was
// waiting on them)
delete_sem(fScanSemaphore);
delete_sem(fLockSemaphore);
// Wait for the worker thread to terminate
status_t result; status_t result;
if (fWorkerThread != 0) if (fWorkerThread != 0)
wait_for_thread(fWorkerThread, &result); wait_for_thread(fWorkerThread, &result);
// TODO power off cards, stop clock, etc if needed.
delete_sem(fLockSemaphore); // TODO power off cards, stop clock, etc if needed.
delete_sem(fScanSemaphore);
} }
@ -159,6 +163,13 @@ MMCBus::_WorkerThread(void* cookie)
do { do {
bus->_AcquireScanSemaphore(); bus->_AcquireScanSemaphore();
// Check if we need to exit early (possible if the parent device did
// not manage initialize itself correctly)
if (bus->fStatus == B_SHUTTING_DOWN) {
release_sem(bus->fLockSemaphore);
return B_OK;
}
TRACE("Reset the bus...\n"); TRACE("Reset the bus...\n");
result = bus->ExecuteCommand(0, SD_GO_IDLE_STATE, 0, NULL); result = bus->ExecuteCommand(0, SD_GO_IDLE_STATE, 0, NULL);
TRACE("CMD0 result: %s\n", strerror(result)); TRACE("CMD0 result: %s\n", strerror(result));

View File

@ -663,13 +663,13 @@ init_bus(device_node* node, void** bus_cookie)
sPCIx86Module = NULL; sPCIx86Module = NULL;
ERROR("PCIx86Module not loaded\n"); ERROR("PCIx86Module not loaded\n");
// FIXME try probing FDT as well // FIXME try probing FDT as well
return -1; return B_NO_MEMORY;
} }
uint8_t bar, slot; uint8_t bar, slot;
if (gDeviceManager->get_attr_uint8(node, SLOT_NUMBER, &slot, false) < B_OK if (gDeviceManager->get_attr_uint8(node, SLOT_NUMBER, &slot, false) < B_OK
|| gDeviceManager->get_attr_uint8(node, BAR_INDEX, &bar, false) < B_OK) || gDeviceManager->get_attr_uint8(node, BAR_INDEX, &bar, false) < B_OK)
return -1; return B_BAD_TYPE;
// Ignore invalid bars // Ignore invalid bars
TRACE("Register SD bus at slot %d, using bar %d\n", slot + 1, bar); TRACE("Register SD bus at slot %d, using bar %d\n", slot + 1, bar);
@ -679,7 +679,7 @@ init_bus(device_node* node, void** bus_cookie)
if (pciInfo.u.h0.base_register_sizes[bar] == 0) { if (pciInfo.u.h0.base_register_sizes[bar] == 0) {
ERROR("No registers to map\n"); ERROR("No registers to map\n");
return -1; return B_IO_ERROR;
} }
int msiCount = sPCIx86Module->get_msi_count(pciInfo.bus, int msiCount = sPCIx86Module->get_msi_count(pciInfo.bus,