drivers/cdcacm/serial: add release interface to release uart_dev resource

Signed-off-by: dongjiuzhu1 <dongjiuzhu1@xiaomi.com>
This commit is contained in:
dongjiuzhu1 2023-12-11 21:11:41 +08:00 committed by Xiang Xiao
parent 5af805b2ef
commit 9c55f21a6f
5 changed files with 106 additions and 10 deletions

View File

@ -112,6 +112,9 @@ static int uart_ioctl(FAR struct file *filep,
int cmd, unsigned long arg); int cmd, unsigned long arg);
static int uart_poll(FAR struct file *filep, static int uart_poll(FAR struct file *filep,
FAR struct pollfd *fds, bool setup); FAR struct pollfd *fds, bool setup);
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
static int uart_unlink(FAR struct inode *inode);
#endif
/**************************************************************************** /****************************************************************************
* Public Function Prototypes * Public Function Prototypes
@ -138,6 +141,9 @@ static const struct file_operations g_serialops =
NULL, /* mmap */ NULL, /* mmap */
NULL, /* truncate */ NULL, /* truncate */
uart_poll /* poll */ uart_poll /* poll */
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
, uart_unlink /* unlink */
#endif
}; };
#ifdef CONFIG_TTY_LAUNCH #ifdef CONFIG_TTY_LAUNCH
@ -725,6 +731,20 @@ static int uart_close(FAR struct file *filep)
*/ */
uart_reset_sem(dev); uart_reset_sem(dev);
if (dev->unlinked)
{
nxmutex_unlock(&dev->closelock);
nxmutex_destroy(&dev->xmit.lock);
nxmutex_destroy(&dev->recv.lock);
nxmutex_destroy(&dev->closelock);
nxmutex_destroy(&dev->polllock);
nxsem_destroy(&dev->xmitsem);
nxsem_destroy(&dev->recvsem);
uart_release(dev);
return OK;
}
nxmutex_unlock(&dev->closelock); nxmutex_unlock(&dev->closelock);
return OK; return OK;
} }
@ -1703,6 +1723,46 @@ errout:
return ret; return ret;
} }
/****************************************************************************
* Name: uart_unlink
****************************************************************************/
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
static int uart_unlink(FAR struct inode *inode)
{
FAR uart_dev_t *dev;
int ret;
DEBUGASSERT(inode->i_private != NULL);
dev = inode->i_private;
ret = nxmutex_lock(&dev->closelock);
if (ret < 0)
{
/* A signal received while waiting for the last close operation. */
return ret;
}
if (dev->open_count <= 0)
{
nxmutex_unlock(&dev->closelock);
nxmutex_destroy(&dev->xmit.lock);
nxmutex_destroy(&dev->recv.lock);
nxmutex_destroy(&dev->closelock);
nxmutex_destroy(&dev->polllock);
nxsem_destroy(&dev->xmitsem);
nxsem_destroy(&dev->recvsem);
uart_release(dev);
return OK;
}
dev->unlinked = true;
nxmutex_unlock(&dev->closelock);
return OK;
}
#endif
/**************************************************************************** /****************************************************************************
* Name: uart_nxsched_foreach_cb * Name: uart_nxsched_foreach_cb
****************************************************************************/ ****************************************************************************/

View File

@ -222,6 +222,7 @@ static bool cdcuart_rxflowcontrol(FAR struct uart_dev_s *dev,
#endif #endif
static void cdcuart_txint(FAR struct uart_dev_s *dev, bool enable); static void cdcuart_txint(FAR struct uart_dev_s *dev, bool enable);
static bool cdcuart_txempty(FAR struct uart_dev_s *dev); static bool cdcuart_txempty(FAR struct uart_dev_s *dev);
static int cdcuart_release(FAR struct uart_dev_s *dev);
/**************************************************************************** /****************************************************************************
* Private Data * Private Data
@ -272,7 +273,8 @@ static const struct uart_ops_s g_uartops =
NULL, /* send */ NULL, /* send */
cdcuart_txint, /* txinit */ cdcuart_txint, /* txinit */
NULL, /* txready */ NULL, /* txready */
cdcuart_txempty /* txempty */ cdcuart_txempty, /* txempty */
cdcuart_release /* release */
}; };
/**************************************************************************** /****************************************************************************
@ -2800,6 +2802,28 @@ static bool cdcuart_txempty(FAR struct uart_dev_s *dev)
return priv->nwrq >= CONFIG_CDCACM_NWRREQS; return priv->nwrq >= CONFIG_CDCACM_NWRREQS;
} }
/****************************************************************************
* Name: cdcuart_release
*
* Description:
* This is called to release some resource about the device when device
* was close and unregistered.
*
****************************************************************************/
static int cdcuart_release(FAR struct uart_dev_s *dev)
{
FAR struct cdcacm_dev_s *priv = (FAR struct cdcacm_dev_s *)dev->priv;
usbtrace(CDCACM_CLASSAPI_RELEASE, 0);
/* And free the memory resources. */
wd_cancel(&priv->rxfailsafe);
kmm_free(priv);
return OK;
}
/**************************************************************************** /****************************************************************************
* Public Functions * Public Functions
****************************************************************************/ ****************************************************************************/

View File

@ -181,6 +181,7 @@
#define CDCACM_CLASSAPI_TXREADY TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_TXREADY) #define CDCACM_CLASSAPI_TXREADY TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_TXREADY)
#define CDCACM_CLASSAPI_TXEMPTY TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_TXEMPTY) #define CDCACM_CLASSAPI_TXEMPTY TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_TXEMPTY)
#define CDCACM_CLASSAPI_FLOWCONTROL TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_FLOWCONTROL) #define CDCACM_CLASSAPI_FLOWCONTROL TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_FLOWCONTROL)
#define CDCACM_CLASSAPI_RELEASE TRACE_EVENT(TRACE_CLASSAPI_ID, USBSER_TRACECLASSAPI_RELEASE)
/**************************************************************************** /****************************************************************************
* Public Types * Public Types

View File

@ -88,6 +88,9 @@
#define uart_send(dev,ch) dev->ops->send(dev,ch) #define uart_send(dev,ch) dev->ops->send(dev,ch)
#define uart_receive(dev,s) dev->ops->receive(dev,s) #define uart_receive(dev,s) dev->ops->receive(dev,s)
#define uart_release(dev) \
((dev)->ops->release ? (dev)->ops->release(dev) : -ENOSYS)
#ifdef CONFIG_SERIAL_TXDMA #ifdef CONFIG_SERIAL_TXDMA
#define uart_dmasend(dev) \ #define uart_dmasend(dev) \
((dev)->ops->dmasend ? (dev)->ops->dmasend(dev) : -ENOSYS) ((dev)->ops->dmasend ? (dev)->ops->dmasend(dev) : -ENOSYS)
@ -254,6 +257,12 @@ struct uart_ops_s
*/ */
CODE bool (*txempty)(FAR struct uart_dev_s *dev); CODE bool (*txempty)(FAR struct uart_dev_s *dev);
/* Call to release some resource about the device when device was close
* and unregistered.
*/
CODE int (*release)(FAR struct uart_dev_s *dev);
}; };
/* This is the device structure used by the driver. The caller of /* This is the device structure used by the driver. The caller of
@ -276,6 +285,7 @@ struct uart_dev_s
volatile bool disconnected; /* true: Removable device is not connected */ volatile bool disconnected; /* true: Removable device is not connected */
#endif #endif
bool isconsole; /* true: This is the serial console */ bool isconsole; /* true: This is the serial console */
bool unlinked; /* true: This device driver has been unlinked. */
#if defined(CONFIG_TTY_SIGINT) || defined(CONFIG_TTY_SIGTSTP) || \ #if defined(CONFIG_TTY_SIGINT) || defined(CONFIG_TTY_SIGTSTP) || \
defined(CONFIG_TTY_FORCE_PANIC) || defined(CONFIG_TTY_LAUNCH) defined(CONFIG_TTY_FORCE_PANIC) || defined(CONFIG_TTY_LAUNCH)

View File

@ -187,6 +187,7 @@
#define USBSER_TRACECLASSAPI_TXREADY 0x000b #define USBSER_TRACECLASSAPI_TXREADY 0x000b
#define USBSER_TRACECLASSAPI_TXEMPTY 0x000c #define USBSER_TRACECLASSAPI_TXEMPTY 0x000c
#define USBSER_TRACECLASSAPI_FLOWCONTROL 0x000d #define USBSER_TRACECLASSAPI_FLOWCONTROL 0x000d
#define USBSER_TRACECLASSAPI_RELEASE 0x000e
/* Values of the class error ID used by the USB serial driver */ /* Values of the class error ID used by the USB serial driver */