From 9436a78a5fda012f3d80f6311c91664247822d62 Mon Sep 17 00:00:00 2001 From: Xiang Xiao Date: Sun, 27 Feb 2022 02:25:32 +0800 Subject: [PATCH] input/ajoystick: Fix the event lose between the invocation of poll Signed-off-by: Xiang Xiao --- drivers/input/ajoystick.c | 45 ++++++++++++++++++++++++--------------- 1 file changed, 28 insertions(+), 17 deletions(-) diff --git a/drivers/input/ajoystick.c b/drivers/input/ajoystick.c index 312f8ea2d3..be0f897ad2 100644 --- a/drivers/input/ajoystick.c +++ b/drivers/input/ajoystick.c @@ -99,6 +99,7 @@ struct ajoy_open_s * driver events. */ + bool ao_pollpending; FAR struct pollfd *ao_fds[CONFIG_INPUT_AJOYSTICK_NPOLLWAITERS]; }; @@ -174,7 +175,6 @@ static void ajoy_enable(FAR struct ajoy_upperhalf_s *priv) ajoy_buttonset_t press; ajoy_buttonset_t release; irqstate_t flags; - int i; DEBUGASSERT(priv); lower = priv->au_lower; @@ -193,19 +193,8 @@ static void ajoy_enable(FAR struct ajoy_upperhalf_s *priv) for (opriv = priv->au_open; opriv; opriv = opriv->ao_flink) { - /* Are there any poll waiters? */ - - for (i = 0; i < CONFIG_INPUT_AJOYSTICK_NPOLLWAITERS; i++) - { - if (opriv->ao_fds[i]) - { - /* Yes.. OR in the poll event buttons */ - - press |= opriv->ao_pollevents.ap_press; - release |= opriv->ao_pollevents.ap_release; - break; - } - } + press |= opriv->ao_pollevents.ap_press; + release |= opriv->ao_pollevents.ap_release; /* OR in the signal events */ @@ -285,8 +274,8 @@ static void ajoy_sample(FAR struct ajoy_upperhalf_s *priv) * newly released. */ - change = sample ^ priv->au_sample; - press = change & sample; + change = sample ^ priv->au_sample; + press = change & sample; DEBUGASSERT(lower->al_supported); release = change & (lower->al_supported(lower) & ~sample); @@ -300,6 +289,8 @@ static void ajoy_sample(FAR struct ajoy_upperhalf_s *priv) if ((press & opriv->ao_pollevents.ap_press) != 0 || (release & opriv->ao_pollevents.ap_release) != 0) { + opriv->ao_pollpending = true; + /* Yes.. Notify all waiters */ for (i = 0; i < CONFIG_INPUT_AJOYSTICK_NPOLLWAITERS; i++) @@ -385,6 +376,10 @@ static int ajoy_open(FAR struct file *filep) opriv->ao_flink = priv->au_open; priv->au_open = opriv; + /* Enable/disable interrupt handling */ + + ajoy_enable(priv); + /* Attach the open structure to the file structure */ filep->f_priv = (FAR void *)opriv; @@ -498,11 +493,13 @@ static ssize_t ajoy_read(FAR struct file *filep, FAR char *buffer, size_t len) { FAR struct inode *inode; + FAR struct ajoy_open_s *opriv; FAR struct ajoy_upperhalf_s *priv; FAR const struct ajoy_lowerhalf_s *lower; int ret; DEBUGASSERT(filep && filep->f_inode); + opriv = filep->f_priv; inode = filep->f_inode; DEBUGASSERT(inode->i_private); priv = (FAR struct ajoy_upperhalf_s *)inode->i_private; @@ -535,6 +532,7 @@ static ssize_t ajoy_read(FAR struct file *filep, FAR char *buffer, ret = lower->al_sample(lower, (FAR struct ajoy_sample_s *)buffer); if (ret >= 0) { + opriv->ao_pollpending = false; ret = sizeof(struct ajoy_sample_s); } @@ -717,6 +715,19 @@ static int ajoy_poll(FAR struct file *filep, FAR struct pollfd *fds, opriv->ao_fds[i] = fds; fds->priv = &opriv->ao_fds[i]; + + /* Report if the event is pending */ + + if (opriv->ao_pollpending) + { + fds->revents |= (fds->events & POLLIN); + if (fds->revents != 0) + { + iinfo("Report events: %02x\n", fds->revents); + nxsem_post(fds->sem); + } + } + break; } } @@ -769,7 +780,7 @@ errout_with_dusem: * * Input Parameters: * devname - The name of the analog joystick device to be registers. - * This should be a string of the form "/priv/ajoyN" where N is the + * This should be a string of the form "/dev/ajoyN" where N is the * minor device number. * lower - An instance of the platform-specific analog joystick lower * half driver.