sched/mqueue: remove sched_lock to improve performance
remove the sched_lock/unlock to improve the performance by 18% mq_send Flow Cycle Count mq_send Origin Optimized | ->nxmq_send 24 24 | ->file_mq_send 209 209 | |->sched_lock 243 N/A <- |->nxmq_do_send 391 348 | | | |->sched_lock 434 N/A <- | |->up_unblock_task 545 459 | ->sched_unlock 675 N/A <- | ->sched_unlock 684 N/A <- | ->up_release_pending 701 N/A | ->arm_switchcontext 856 610 mq_receive | ->arm_fullcontextrestore 1375 1133 | ->up_block_task 1375 1133 | ->nxmq_wait_receive 1530 1288 | ->file_mq_receive 1606 1310 | ->nxmq_receive 1616 1320 | ->mq_receive 1628 1332 - 18% Signed-off-by: chao.an <anchao@xiaomi.com>
This commit is contained in:
parent
644283c8ff
commit
f5d4571abc
@ -29,6 +29,7 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
|
|
||||||
|
#include <nuttx/irq.h>
|
||||||
#include <nuttx/sched.h>
|
#include <nuttx/sched.h>
|
||||||
|
|
||||||
#include "sched/sched.h"
|
#include "sched/sched.h"
|
||||||
@ -97,11 +98,13 @@ int mq_notify(mqd_t mqdes, FAR const struct sigevent *notification)
|
|||||||
FAR struct inode *inode;
|
FAR struct inode *inode;
|
||||||
FAR struct file *filep;
|
FAR struct file *filep;
|
||||||
FAR struct tcb_s *rtcb;
|
FAR struct tcb_s *rtcb;
|
||||||
|
irqstate_t flags;
|
||||||
int errval;
|
int errval;
|
||||||
|
|
||||||
errval = fs_getfilep(mqdes, &filep);
|
errval = fs_getfilep(mqdes, &filep);
|
||||||
if (errval < 0)
|
if (errval < 0)
|
||||||
{
|
{
|
||||||
|
errval = -errval;
|
||||||
goto errout_without_lock;
|
goto errout_without_lock;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -119,7 +122,7 @@ int mq_notify(mqd_t mqdes, FAR const struct sigevent *notification)
|
|||||||
|
|
||||||
/* Get a pointer to the message queue */
|
/* Get a pointer to the message queue */
|
||||||
|
|
||||||
sched_lock();
|
flags = enter_critical_section();
|
||||||
|
|
||||||
/* Get the current process ID */
|
/* Get the current process ID */
|
||||||
|
|
||||||
@ -177,11 +180,11 @@ int mq_notify(mqd_t mqdes, FAR const struct sigevent *notification)
|
|||||||
nxsig_cancel_notification(&msgq->ntwork);
|
nxsig_cancel_notification(&msgq->ntwork);
|
||||||
}
|
}
|
||||||
|
|
||||||
sched_unlock();
|
leave_critical_section(flags);
|
||||||
return OK;
|
return OK;
|
||||||
|
|
||||||
errout:
|
errout:
|
||||||
sched_unlock();
|
leave_critical_section(flags);
|
||||||
|
|
||||||
errout_without_lock:
|
errout_without_lock:
|
||||||
set_errno(errval);
|
set_errno(errval);
|
||||||
|
@ -99,15 +99,6 @@ ssize_t file_mq_receive(FAR struct file *mq, FAR char *msg, size_t msglen,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the next message from the message queue. We will disable
|
|
||||||
* pre-emption until we have completed the message received. This
|
|
||||||
* is not too bad because if the receipt takes a long time, it will
|
|
||||||
* be because we are blocked waiting for a message and pre-emption
|
|
||||||
* will be re-enabled while we are blocked
|
|
||||||
*/
|
|
||||||
|
|
||||||
sched_lock();
|
|
||||||
|
|
||||||
/* Furthermore, nxmq_wait_receive() expects to have interrupts disabled
|
/* Furthermore, nxmq_wait_receive() expects to have interrupts disabled
|
||||||
* because messages can be sent from interrupt level.
|
* because messages can be sent from interrupt level.
|
||||||
*/
|
*/
|
||||||
@ -132,7 +123,6 @@ ssize_t file_mq_receive(FAR struct file *mq, FAR char *msg, size_t msglen,
|
|||||||
ret = nxmq_do_receive(msgq, mqmsg, msg, prio);
|
ret = nxmq_do_receive(msgq, mqmsg, msg, prio);
|
||||||
}
|
}
|
||||||
|
|
||||||
sched_unlock();
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -94,10 +94,6 @@ int file_mq_send(FAR struct file *mq, FAR const char *msg, size_t msglen,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get a pointer to the message queue */
|
|
||||||
|
|
||||||
sched_lock();
|
|
||||||
|
|
||||||
/* Allocate a message structure:
|
/* Allocate a message structure:
|
||||||
* - Immediately if we are called from an interrupt handler.
|
* - Immediately if we are called from an interrupt handler.
|
||||||
* - Immediately if the message queue is not full, or
|
* - Immediately if the message queue is not full, or
|
||||||
@ -156,7 +152,6 @@ int file_mq_send(FAR struct file *mq, FAR const char *msg, size_t msglen,
|
|||||||
ret = nxmq_do_send(msgq, mqmsg, msg, msglen, prio);
|
ret = nxmq_do_send(msgq, mqmsg, msg, msglen, prio);
|
||||||
}
|
}
|
||||||
|
|
||||||
sched_unlock();
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -327,10 +327,6 @@ int nxmq_do_send(FAR struct mqueue_inode_s *msgq,
|
|||||||
FAR struct mqueue_msg_s *prev;
|
FAR struct mqueue_msg_s *prev;
|
||||||
irqstate_t flags;
|
irqstate_t flags;
|
||||||
|
|
||||||
/* Get a pointer to the message queue */
|
|
||||||
|
|
||||||
sched_lock();
|
|
||||||
|
|
||||||
/* Construct the message header info */
|
/* Construct the message header info */
|
||||||
|
|
||||||
mqmsg->priority = prio;
|
mqmsg->priority = prio;
|
||||||
@ -405,8 +401,9 @@ int nxmq_do_send(FAR struct mqueue_inode_s *msgq,
|
|||||||
{
|
{
|
||||||
/* Find the highest priority task that is waiting for
|
/* Find the highest priority task that is waiting for
|
||||||
* this queue to be non-empty in g_waitingformqnotempty
|
* this queue to be non-empty in g_waitingformqnotempty
|
||||||
* list. sched_lock() should give us sufficient protection since
|
* list. leave_critical_section() should give us sufficient
|
||||||
* interrupts should never cause a change in this list
|
* protection since interrupts should never cause a change
|
||||||
|
* in this list
|
||||||
*/
|
*/
|
||||||
|
|
||||||
for (btcb = (FAR struct tcb_s *)g_waitingformqnotempty.head;
|
for (btcb = (FAR struct tcb_s *)g_waitingformqnotempty.head;
|
||||||
@ -425,6 +422,5 @@ int nxmq_do_send(FAR struct mqueue_inode_s *msgq,
|
|||||||
}
|
}
|
||||||
|
|
||||||
leave_critical_section(flags);
|
leave_critical_section(flags);
|
||||||
sched_unlock();
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
@ -170,15 +170,6 @@ ssize_t file_mq_timedreceive(FAR struct file *mq, FAR char *msg,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the next message from the message queue. We will disable
|
|
||||||
* pre-emption until we have completed the message received. This
|
|
||||||
* is not too bad because if the receipt takes a long time, it will
|
|
||||||
* be because we are blocked waiting for a message and pre-emption
|
|
||||||
* will be re-enabled while we are blocked
|
|
||||||
*/
|
|
||||||
|
|
||||||
sched_lock();
|
|
||||||
|
|
||||||
/* Furthermore, nxmq_wait_receive() expects to have interrupts disabled
|
/* Furthermore, nxmq_wait_receive() expects to have interrupts disabled
|
||||||
* because messages can be sent from interrupt level.
|
* because messages can be sent from interrupt level.
|
||||||
*/
|
*/
|
||||||
@ -213,7 +204,6 @@ ssize_t file_mq_timedreceive(FAR struct file *mq, FAR char *msg,
|
|||||||
if (result != OK)
|
if (result != OK)
|
||||||
{
|
{
|
||||||
leave_critical_section(flags);
|
leave_critical_section(flags);
|
||||||
sched_unlock();
|
|
||||||
return -result;
|
return -result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -250,7 +240,6 @@ ssize_t file_mq_timedreceive(FAR struct file *mq, FAR char *msg,
|
|||||||
ret = nxmq_do_receive(msgq, mqmsg, msg, prio);
|
ret = nxmq_do_receive(msgq, mqmsg, msg, prio);
|
||||||
}
|
}
|
||||||
|
|
||||||
sched_unlock();
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -185,10 +185,6 @@ int file_mq_timedsend(FAR struct file *mq, FAR const char *msg,
|
|||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get a pointer to the message queue */
|
|
||||||
|
|
||||||
sched_lock();
|
|
||||||
|
|
||||||
/* OpenGroup.org: "Under no circumstance shall the operation fail with a
|
/* OpenGroup.org: "Under no circumstance shall the operation fail with a
|
||||||
* timeout if there is sufficient room in the queue to add the message
|
* timeout if there is sufficient room in the queue to add the message
|
||||||
* immediately. The validity of the abstime parameter need not be checked
|
* immediately. The validity of the abstime parameter need not be checked
|
||||||
@ -209,9 +205,7 @@ int file_mq_timedsend(FAR struct file *mq, FAR const char *msg,
|
|||||||
* Currently nxmq_do_send() always returns OK.
|
* Currently nxmq_do_send() always returns OK.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
ret = nxmq_do_send(msgq, mqmsg, msg, msglen, prio);
|
return nxmq_do_send(msgq, mqmsg, msg, msglen, prio);
|
||||||
sched_unlock();
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* The message queue is full... We are going to wait. Now we must have a
|
/* The message queue is full... We are going to wait. Now we must have a
|
||||||
@ -285,10 +279,7 @@ int file_mq_timedsend(FAR struct file *mq, FAR const char *msg,
|
|||||||
* Currently nxmq_do_send() always returns OK.
|
* Currently nxmq_do_send() always returns OK.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
ret = nxmq_do_send(msgq, mqmsg, msg, msglen, prio);
|
return nxmq_do_send(msgq, mqmsg, msg, msglen, prio);
|
||||||
|
|
||||||
sched_unlock();
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
/* Exit here with (1) the scheduler locked, (2) a message allocated, (3) a
|
/* Exit here with (1) the scheduler locked, (2) a message allocated, (3) a
|
||||||
* wdog allocated, and (4) interrupts disabled.
|
* wdog allocated, and (4) interrupts disabled.
|
||||||
@ -303,7 +294,6 @@ errout_in_critical_section:
|
|||||||
|
|
||||||
errout_with_mqmsg:
|
errout_with_mqmsg:
|
||||||
nxmq_free_msg(mqmsg);
|
nxmq_free_msg(mqmsg);
|
||||||
sched_unlock();
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user