- MMS server: fixed - possible deadlock in obtainFile-service/file upload task (LIB61850-351)

pull/410/head
Michael Zillgith 3 years ago
parent a3b04b7bc4
commit 7c06680cba

@ -219,7 +219,7 @@ LIB61850_INTERNAL MmsObtainFileTask
MmsServer_getObtainFileTask(MmsServer self); MmsServer_getObtainFileTask(MmsServer self);
LIB61850_INTERNAL void LIB61850_INTERNAL void
mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task); mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task, int taskState);
#endif #endif
LIB61850_INTERNAL ByteBuffer* LIB61850_INTERNAL ByteBuffer*

@ -411,8 +411,29 @@ createObtainFileResponse(uint32_t invokeId, ByteBuffer* response)
} }
void void
mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task) mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task, int taskState)
{ {
/* call locks in certain order (lock IsoConnection -> reserverTransmitBuffer, task->taskLock) to prevent potential deadlock */
ByteBuffer* message = NULL;
if (taskState == MMS_FILE_UPLOAD_STATE_SEND_FILE_READ ||
taskState == MMS_FILE_UPLOAD_STATE_SEND_FILE_CLOSE ||
taskState == MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_ERROR_SOURCE ||
taskState == MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_ERROR_DESTINATION ||
taskState == MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_RESPONSE)
{
IsoConnection_lock(task->connection->isoConnection);
message = MmsServer_reserveTransmitBuffer(self);
}
#if (CONFIG_MMS_THREADLESS_STACK != 1)
Semaphore_wait(task->taskLock);
#endif
if (task->state == taskState) {
switch (task->state) { switch (task->state) {
case MMS_FILE_UPLOAD_STATE_NOT_USED: case MMS_FILE_UPLOAD_STATE_NOT_USED:
@ -438,20 +459,12 @@ mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task)
case MMS_FILE_UPLOAD_STATE_SEND_FILE_READ: case MMS_FILE_UPLOAD_STATE_SEND_FILE_READ:
{ {
IsoConnection_lock(task->connection->isoConnection);
ByteBuffer* request = MmsServer_reserveTransmitBuffer(self);
task->lastRequestInvokeId = MmsServerConnection_getNextRequestInvokeId(task->connection); task->lastRequestInvokeId = MmsServerConnection_getNextRequestInvokeId(task->connection);
mmsClient_createFileReadRequest(task->lastRequestInvokeId, request, task->frmsId); mmsClient_createFileReadRequest(task->lastRequestInvokeId, message, task->frmsId);
task->state = MMS_FILE_UPLOAD_STATE_FILE_READ_SENT; task->state = MMS_FILE_UPLOAD_STATE_FILE_READ_SENT;
IsoConnection_sendMessage(task->connection->isoConnection, request); IsoConnection_sendMessage(task->connection->isoConnection, message);
MmsServer_releaseTransmitBuffer(self);
IsoConnection_unlock(task->connection->isoConnection);
task->nextTimeout = Hal_getTimeInMs() + 2000; /* timeout 2000 ms */ task->nextTimeout = Hal_getTimeInMs() + 2000; /* timeout 2000 ms */
} }
@ -478,21 +491,13 @@ mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task)
case MMS_FILE_UPLOAD_STATE_SEND_FILE_CLOSE: case MMS_FILE_UPLOAD_STATE_SEND_FILE_CLOSE:
{ {
IsoConnection_lock(task->connection->isoConnection);
ByteBuffer* request = MmsServer_reserveTransmitBuffer(self);
task->lastRequestInvokeId = MmsServerConnection_getNextRequestInvokeId(task->connection); task->lastRequestInvokeId = MmsServerConnection_getNextRequestInvokeId(task->connection);
mmsClient_createFileCloseRequest(task->lastRequestInvokeId, request, task->frmsId); mmsClient_createFileCloseRequest(task->lastRequestInvokeId, message, task->frmsId);
task->state = MMS_FILE_UPLOAD_STATE_FILE_CLOSE_SENT; task->state = MMS_FILE_UPLOAD_STATE_FILE_CLOSE_SENT;
IsoConnection_sendMessage(task->connection->isoConnection, request); IsoConnection_sendMessage(task->connection->isoConnection, message);
MmsServer_releaseTransmitBuffer(self);
IsoConnection_unlock(task->connection->isoConnection);
task->nextTimeout = Hal_getTimeInMs() + 2000; /* timeout 2000 ms */ task->nextTimeout = Hal_getTimeInMs() + 2000; /* timeout 2000 ms */
@ -519,17 +524,9 @@ mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task)
{ {
/* send ObtainFileError */ /* send ObtainFileError */
IsoConnection_lock(task->connection->isoConnection); createServiceErrorObtainFileError(task->obtainFileRequestInvokeId, message, MMS_ERROR_FILE_FILE_NON_EXISTENT, 0);
ByteBuffer* response = MmsServer_reserveTransmitBuffer(self);
createServiceErrorObtainFileError(task->obtainFileRequestInvokeId, response, MMS_ERROR_FILE_FILE_NON_EXISTENT, 0);
IsoConnection_sendMessage(task->connection->isoConnection, response); IsoConnection_sendMessage(task->connection->isoConnection, message);
MmsServer_releaseTransmitBuffer(self);
IsoConnection_unlock(task->connection->isoConnection);
if(task->fileHandle){ if(task->fileHandle){
FileSystem_closeFile(task->fileHandle); FileSystem_closeFile(task->fileHandle);
@ -548,17 +545,9 @@ mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task)
case MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_ERROR_DESTINATION: case MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_ERROR_DESTINATION:
{ {
/* send ObtainFileError */ /* send ObtainFileError */
IsoConnection_lock(task->connection->isoConnection); createServiceErrorObtainFileError(task->obtainFileRequestInvokeId, message, MMS_ERROR_FILE_OTHER, 1);
ByteBuffer* response = MmsServer_reserveTransmitBuffer(self);
createServiceErrorObtainFileError(task->obtainFileRequestInvokeId, response, MMS_ERROR_FILE_OTHER, 1); IsoConnection_sendMessage(task->connection->isoConnection, message);
IsoConnection_sendMessage(task->connection->isoConnection, response);
MmsServer_releaseTransmitBuffer(self);
IsoConnection_unlock(task->connection->isoConnection);
if (task->fileHandle) { if (task->fileHandle) {
FileSystem_closeFile(task->fileHandle); FileSystem_closeFile(task->fileHandle);
@ -578,24 +567,17 @@ mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task)
case MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_RESPONSE: case MMS_FILE_UPLOAD_STATE_SEND_OBTAIN_FILE_RESPONSE:
{ {
IsoConnection_lock(task->connection->isoConnection); createObtainFileResponse(task->obtainFileRequestInvokeId, message);
ByteBuffer* response = MmsServer_reserveTransmitBuffer(self);
createObtainFileResponse(task->obtainFileRequestInvokeId, response);
task->state = MMS_FILE_UPLOAD_STATE_NOT_USED; task->state = MMS_FILE_UPLOAD_STATE_NOT_USED;
IsoConnection_sendMessage(task->connection->isoConnection, response); IsoConnection_sendMessage(task->connection->isoConnection, message);
MmsServer_releaseTransmitBuffer(self);
IsoConnection_unlock(task->connection->isoConnection);
if (self->getFileCompleteHandler) if (self->getFileCompleteHandler)
self->getFileCompleteHandler(self->getFileCompleteHandlerParameter, task->connection, task->destinationFilename); self->getFileCompleteHandler(self->getFileCompleteHandlerParameter, task->connection, task->destinationFilename);
} }
break; break;
case MMS_FILE_UPLOAD_STATE_INTERRUPTED: case MMS_FILE_UPLOAD_STATE_INTERRUPTED:
{ {
if (DEBUG_MMS_SERVER) if (DEBUG_MMS_SERVER)
@ -612,6 +594,16 @@ mmsServer_fileUploadTask(MmsServer self, MmsObtainFileTask task)
} }
break; break;
} }
}
#if (CONFIG_MMS_THREADLESS_STACK != 1)
Semaphore_post(task->taskLock);
#endif
if (message) {
MmsServer_releaseTransmitBuffer(self);
IsoConnection_unlock(task->connection->isoConnection);
}
} }
#if (MMS_OBTAIN_FILE_SERVICE == 1) #if (MMS_OBTAIN_FILE_SERVICE == 1)
@ -770,6 +762,10 @@ mmsServer_handleObtainFileRequest(
task->state = MMS_FILE_UPLOAD_STATE_FILE_OPEN_SENT; task->state = MMS_FILE_UPLOAD_STATE_FILE_OPEN_SENT;
} }
#if (CONFIG_MMS_THREADLESS_STACK != 1)
Semaphore_post(task->taskLock);
#endif
} }
else else
goto exit_unavailable; goto exit_unavailable;

@ -316,10 +316,6 @@ MmsServer_getObtainFileTask(MmsServer self)
if (self->fileUploadTasks[i].state == 0) { if (self->fileUploadTasks[i].state == 0) {
self->fileUploadTasks[i].state = 1; self->fileUploadTasks[i].state = 1;
#if (CONFIG_MMS_THREADLESS_STACK != 1)
Semaphore_post(self->fileUploadTasks[i].taskLock);
#endif
return &(self->fileUploadTasks[i]); return &(self->fileUploadTasks[i]);
} }
@ -742,15 +738,15 @@ MmsServer_handleBackgroundTasks(MmsServer self)
Semaphore_wait(self->fileUploadTasks[i].taskLock); Semaphore_wait(self->fileUploadTasks[i].taskLock);
#endif #endif
if (self->fileUploadTasks[i].state != 0) { int taskState = self->fileUploadTasks[i].state;
if (self->fileUploadTasks[i].state != 0)
mmsServer_fileUploadTask(self, &(self->fileUploadTasks[i]));
}
#if (CONFIG_MMS_THREADLESS_STACK != 1) #if (CONFIG_MMS_THREADLESS_STACK != 1)
Semaphore_post(self->fileUploadTasks[i].taskLock); Semaphore_post(self->fileUploadTasks[i].taskLock);
#endif #endif
if (taskState != 0) {
mmsServer_fileUploadTask(self, &(self->fileUploadTasks[i]), taskState);
}
} }
#endif /* (MMS_OBTAIN_FILE_SERVICE == 1) */ #endif /* (MMS_OBTAIN_FILE_SERVICE == 1) */

Loading…
Cancel
Save