Fix some odd logic in CC3000 semaphore handling. Noted by Vladimir Komendantskiy
This commit is contained in:
parent
2c0ad2564f
commit
bf8120d51d
@ -211,27 +211,23 @@ uint8_t spi_readCommand[] = READ_COMMAND;
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static int cc3000_devtake(FAR struct cc3000_dev_s *priv)
|
static inline void cc3000_devtake(FAR struct cc3000_dev_s *priv)
|
||||||
{
|
{
|
||||||
int rv;
|
|
||||||
|
|
||||||
/* Take the semaphore (perhaps waiting) */
|
/* Take the semaphore (perhaps waiting) */
|
||||||
|
|
||||||
while ((rv = sem_wait(&priv->devsem)) != 0)
|
while (sem_wait(&priv->devsem) < 0)
|
||||||
{
|
{
|
||||||
/* The only case that an error should occur here is if the wait was awakened
|
/* The only case that an error should occur here is if the wait was
|
||||||
* by a signal.
|
* awakened by a signal.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
DEBUGASSERT(rv == OK || errno == EINTR);
|
DEBUGASSERT(errno == EINTR);
|
||||||
}
|
}
|
||||||
|
|
||||||
return rv;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int cc3000_devgive(FAR struct cc3000_dev_s *priv)
|
static inline void cc3000_devgive(FAR struct cc3000_dev_s *priv)
|
||||||
{
|
{
|
||||||
return sem_post(&priv->devsem);
|
(void)sem_post(&priv->devsem);
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -354,27 +350,16 @@ static int cc3000_wait(FAR struct cc3000_dev_s *priv, sem_t *psem)
|
|||||||
/* Wait on first psem to become signaled */
|
/* Wait on first psem to become signaled */
|
||||||
|
|
||||||
ret = sem_wait(psem);
|
ret = sem_wait(psem);
|
||||||
if (ret >= 0)
|
|
||||||
{
|
|
||||||
/* Yes... then retake the mutual exclusion semaphore */
|
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
|
||||||
}
|
|
||||||
|
|
||||||
sched_unlock();
|
|
||||||
|
|
||||||
/* Was the semaphore wait successful? Did we successful re-take the
|
|
||||||
* mutual exclusion semaphore?
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
/* No.. One of the two sem_wait's failed. */
|
return -errno;
|
||||||
|
|
||||||
ret = -errno;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
/* Then retake the mutual exclusion semaphore */
|
||||||
|
|
||||||
|
cc3000_devtake(priv);
|
||||||
|
sched_unlock();
|
||||||
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -820,11 +805,7 @@ static int cc3000_open(FAR struct file *filep)
|
|||||||
|
|
||||||
/* Get exclusive access to the driver data structure */
|
/* Get exclusive access to the driver data structure */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
return -ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Increment the reference count */
|
/* Increment the reference count */
|
||||||
|
|
||||||
@ -986,9 +967,8 @@ out_with_sem:
|
|||||||
|
|
||||||
static int cc3000_close(FAR struct file *filep)
|
static int cc3000_close(FAR struct file *filep)
|
||||||
{
|
{
|
||||||
FAR struct inode *inode;
|
FAR struct inode *inode;
|
||||||
FAR struct cc3000_dev_s *priv;
|
FAR struct cc3000_dev_s *priv;
|
||||||
int ret;
|
|
||||||
#ifdef CONFIG_CC3000_MT
|
#ifdef CONFIG_CC3000_MT
|
||||||
int s;
|
int s;
|
||||||
#endif
|
#endif
|
||||||
@ -1005,11 +985,7 @@ static int cc3000_close(FAR struct file *filep)
|
|||||||
|
|
||||||
/* Get exclusive access to the driver data structure */
|
/* Get exclusive access to the driver data structure */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
return -EINTR;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Decrement the reference count unless it would decrement a negative
|
/* Decrement the reference count unless it would decrement a negative
|
||||||
* value. When the count decrements to zero, there are no further
|
* value. When the count decrements to zero, there are no further
|
||||||
@ -1086,13 +1062,7 @@ static ssize_t cc3000_read(FAR struct file *filep, FAR char *buffer, size_t len)
|
|||||||
|
|
||||||
/* Get exclusive access to the driver data structure */
|
/* Get exclusive access to the driver data structure */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
|
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
nread = -errno;
|
|
||||||
goto errout_without_sem;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Verify that the caller has provided a buffer large enough to receive
|
/* Verify that the caller has provided a buffer large enough to receive
|
||||||
* the maximum data.
|
* the maximum data.
|
||||||
@ -1152,7 +1122,7 @@ static ssize_t cc3000_read(FAR struct file *filep, FAR char *buffer, size_t len)
|
|||||||
{
|
{
|
||||||
/* Yes... then retake the mutual exclusion semaphore */
|
/* Yes... then retake the mutual exclusion semaphore */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Was the semaphore wait successful? Did we successful re-take the
|
/* Was the semaphore wait successful? Did we successful re-take the
|
||||||
@ -1239,15 +1209,7 @@ static ssize_t cc3000_write(FAR struct file *filep, FAR const char *usrbuffer, s
|
|||||||
|
|
||||||
/* Get exclusive access to the driver data structure */
|
/* Get exclusive access to the driver data structure */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
/* This should only happen if the wait was canceled by an signal */
|
|
||||||
|
|
||||||
ndbg("sem_wait: %d\n", errno);
|
|
||||||
nwritten = -errno;
|
|
||||||
goto errout_without_sem;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Figure out the total length of the packet in order to figure out if there is padding or not */
|
/* Figure out the total length of the packet in order to figure out if there is padding or not */
|
||||||
|
|
||||||
@ -1352,13 +1314,7 @@ static int cc3000_ioctl(FAR struct file *filep, int cmd, unsigned long arg)
|
|||||||
|
|
||||||
/* Get exclusive access to the driver data structure */
|
/* Get exclusive access to the driver data structure */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
/* This should only happen if the wait was canceled by an signal */
|
|
||||||
|
|
||||||
return -errno;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Process the IOCTL by command */
|
/* Process the IOCTL by command */
|
||||||
|
|
||||||
@ -1463,13 +1419,7 @@ static int cc3000_poll(FAR struct file *filep, FAR struct pollfd *fds,
|
|||||||
|
|
||||||
/* Are we setting up the poll? Or tearing it down? */
|
/* Are we setting up the poll? Or tearing it down? */
|
||||||
|
|
||||||
ret = cc3000_devtake(priv);
|
cc3000_devtake(priv);
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
/* This should only happen if the wait was canceled by an signal */
|
|
||||||
|
|
||||||
return -errno;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (setup)
|
if (setup)
|
||||||
{
|
{
|
||||||
|
@ -139,7 +139,10 @@ void spinlock(FAR struct spinlock_s *lock)
|
|||||||
|
|
||||||
# warning Missing logic
|
# warning Missing logic
|
||||||
#endif
|
#endif
|
||||||
/* Take the lock */
|
/* Take the lock. REVISIT: We should set an indication in the TCB
|
||||||
|
* that the thread is spinning. This might be useful in determining
|
||||||
|
* some scheduling actions?
|
||||||
|
*/
|
||||||
|
|
||||||
while (up_testset(&lock->sp_lock) == SP_LOCKED)
|
while (up_testset(&lock->sp_lock) == SP_LOCKED)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user