Basic HID keyboard funcionality. More testing is needed

git-svn-id: svn://svn.code.sf.net/p/nuttx/code/trunk@3260 42af7a65-404d-4744-a932-0658087f49c3
This commit is contained in:
patacongo 2011-01-19 04:17:45 +00:00
parent 09107aa9ff
commit f389facd71
4 changed files with 286 additions and 99 deletions

View File

@ -1423,13 +1423,14 @@
* include/nuttx/usb -- rename usb_storage.h to storage.h. * include/nuttx/usb -- rename usb_storage.h to storage.h.
* arch/arm/src/lpc17xx/lpc17_usbhost.c -- Add support for low-speed devices. * arch/arm/src/lpc17xx/lpc17_usbhost.c -- Add support for low-speed devices.
* drivers/usbhost/usbhost_skeleton.c -- Template for new class drivers * drivers/usbhost/usbhost_skeleton.c -- Template for new class drivers
* include/nuttx/usb/hid.h and drivers/usbhost/usbhost_hidkbd.c -- Initial * include/nuttx/usb/hid.h and drivers/usbhost/usbhost_hidkbd.c -- New
files for HID keyboard support. These are mostly empty files on the files for HID keyboard support.
initial checkin.
* arch/arm/src/lpc17xx/lpc17_usbhost.c -- Will now handle multiple * arch/arm/src/lpc17xx/lpc17_usbhost.c -- Will now handle multiple
concurrent transfers on different endpoints (still only one TD per concurrent transfers on different endpoints (still only one TD per
endpoint). All methods are protected from re-entrancy; lots of re- endpoint). All methods are protected from re-entrancy; lots of re-
structuring in preparation for interrupt endpoint support. structuring in preparation for interrupt endpoint support.
* arch/arm/src/lpc17xx/lpc17_usbhost.c -- Add support for periodic
interrupt transfers.
* examples/hidkbd - Added a simple test for the USB host HID keyboard * examples/hidkbd - Added a simple test for the USB host HID keyboard
class driver. class driver.
* configs/olimex-lpc1766stk/hidkbd - Added a configuration to build the * configs/olimex-lpc1766stk/hidkbd - Added a configuration to build the

View File

@ -8,7 +8,7 @@
<tr align="center" bgcolor="#e4e4e4"> <tr align="center" bgcolor="#e4e4e4">
<td> <td>
<h1><big><font color="#3c34ec"><i>NuttX RTOS</i></font></big></h1> <h1><big><font color="#3c34ec"><i>NuttX RTOS</i></font></big></h1>
<p>Last Updated: January 15, 2011</p> <p>Last Updated: January 18, 2011</p>
</td> </td>
</tr> </tr>
</table> </table>
@ -547,7 +547,7 @@
<td><br></td> <td><br></td>
<td> <td>
<p> <p>
<li>Device-dependent USB class drivers available for USB mass storage.</li> <li>Device-dependent USB class drivers available for USB mass storage and HID keyboard.</li>
</p> </p>
</tr> </tr>
@ -2104,13 +2104,14 @@ nuttx-5.17 2011-xx-xx Gregory Nutt &lt;spudmonkey@racsa.co.cr&gt;
* include/nuttx/usb -- rename usb_storage.h to storage.h. * include/nuttx/usb -- rename usb_storage.h to storage.h.
* arch/arm/src/lpc17xx/lpc17_usbhost.c -- Add support for low-speed devices. * arch/arm/src/lpc17xx/lpc17_usbhost.c -- Add support for low-speed devices.
* drivers/usbhost/usbhost_skeleton.c -- Template for new class drivers * drivers/usbhost/usbhost_skeleton.c -- Template for new class drivers
* include/nuttx/usb/hid.h and drivers/usbhost/usbhost_hidkbd.c -- Initial * include/nuttx/usb/hid.h and drivers/usbhost/usbhost_hidkbd.c -- New
files for HID keyboard support. These are mostly empty files on the files for HID keyboard support.
initial checkin.
* arch/arm/src/lpc17xx/lpc17_usbhost.c -- Will now handle multiple * arch/arm/src/lpc17xx/lpc17_usbhost.c -- Will now handle multiple
concurrent transfers on different endpoints (still only one TD per concurrent transfers on different endpoints (still only one TD per
endpoint). All methods are protected from re-entrancy; lots of re- endpoint). All methods are protected from re-entrancy; lots of re-
structuring in preparation for interrupt endpoint support. structuring in preparation for interrupt endpoint support.
* arch/arm/src/lpc17xx/lpc17_usbhost.c -- Add support for periodic
interrupt transfers.
* examples/hidkbd - Added a simple test for the USB host HID keyboard * examples/hidkbd - Added a simple test for the USB host HID keyboard
class driver. class driver.
* configs/olimex-lpc1766stk/hidkbd - Added a configuration to build the * configs/olimex-lpc1766stk/hidkbd - Added a configuration to build the

View File

@ -48,6 +48,7 @@
#include <poll.h> #include <poll.h>
#include <semaphore.h> #include <semaphore.h>
#include <time.h> #include <time.h>
#include <fcntl.h>
#include <assert.h> #include <assert.h>
#include <errno.h> #include <errno.h>
#include <debug.h> #include <debug.h>
@ -106,6 +107,10 @@
# define CONFIG_USBHID_BUFSIZE 64 # define CONFIG_USBHID_BUFSIZE 64
#endif #endif
#ifndef CONFIG_USBHID_NPOLLWAITERS
# define CONFIG_USBHID_NPOLLWAITERS 2
#endif
/* The default is to support scancode mapping for the standard 104 key /* The default is to support scancode mapping for the standard 104 key
* keyboard. Setting CONFIG_USBHID_RAWSCANCODES will disable all scancode * keyboard. Setting CONFIG_USBHID_RAWSCANCODES will disable all scancode
* mapping; Setting CONFIG_USBHID_ALLSCANCODES will enable mapping of all * mapping; Setting CONFIG_USBHID_ALLSCANCODES will enable mapping of all
@ -161,10 +166,12 @@ struct usbhost_state_s
char devchar; /* Character identifying the /dev/kbd[n] device */ char devchar; /* Character identifying the /dev/kbd[n] device */
volatile bool disconnected; /* TRUE: Device has been disconnected */ volatile bool disconnected; /* TRUE: Device has been disconnected */
volatile bool polling; /* TRUE: Poll thread is running */ volatile bool polling; /* TRUE: Poll thread is running */
bool open; /* TRUE: The keyboard device is open */ volatile bool open; /* TRUE: The keyboard device is open */
volatile bool waiting; /* TRUE: waiting for keyboard data */
uint8_t ifno; /* Interface number */ uint8_t ifno; /* Interface number */
int16_t crefs; /* Reference count on the driver instance */ int16_t crefs; /* Reference count on the driver instance */
sem_t exclsem; /* Used to maintain mutual exclusive access */ sem_t exclsem; /* Used to maintain mutual exclusive access */
sem_t waitsem; /* Used to wait for keyboard data */
FAR uint8_t *tbuffer; /* The allocated transfer buffer */ FAR uint8_t *tbuffer; /* The allocated transfer buffer */
size_t tbuflen; /* Size of the allocated transfer buffer */ size_t tbuflen; /* Size of the allocated transfer buffer */
pid_t pollpid; /* PID of the poll task */ pid_t pollpid; /* PID of the poll task */
@ -185,11 +192,20 @@ struct usbhost_state_s
usbhost_ep_t epin; /* Interrupt IN endpoint */ usbhost_ep_t epin; /* Interrupt IN endpoint */
usbhost_ep_t epout; /* Optional interrupt OUT endpoint */ usbhost_ep_t epout; /* Optional interrupt OUT endpoint */
/* The following is a list if poll structures of threads waiting for
* driver events. The 'struct pollfd' reference for each open is also
* retained in the f_priv field of the 'struct file'.
*/
#ifndef CONFIG_DISABLE_POLL
struct pollfd *fds[CONFIG_USBHID_NPOLLWAITERS];
#endif
/* Buffer used to collect and buffer incoming keyboard characters */ /* Buffer used to collect and buffer incoming keyboard characters */
uint16_t headndx; /* Buffer head index */ volatile uint16_t headndx; /* Buffer head index */
uint16_t tailndx; /* Buffer tail index */ volatile uint16_t tailndx; /* Buffer tail index */
uint8_t buffer[CONFIG_USBHID_BUFSIZE]; uint8_t kbdbuffer[CONFIG_USBHID_BUFSIZE];
}; };
/**************************************************************************** /****************************************************************************
@ -201,6 +217,14 @@ struct usbhost_state_s
static void usbhost_takesem(sem_t *sem); static void usbhost_takesem(sem_t *sem);
#define usbhost_givesem(s) sem_post(s); #define usbhost_givesem(s) sem_post(s);
/* Polling support */
#ifndef CONFIG_DISABLE_POLL
static void usbhost_pollnotify(FAR struct usbhost_state_s *dev);
#else
# define usbhost_pollnotify(dev)
#endif
/* Memory allocation services */ /* Memory allocation services */
static inline FAR struct usbhost_state_s *usbhost_allocclass(void); static inline FAR struct usbhost_state_s *usbhost_allocclass(void);
@ -417,6 +441,31 @@ static void usbhost_takesem(sem_t *sem)
} }
} }
/****************************************************************************
* Name: usbhost_pollnotify
****************************************************************************/
#ifndef CONFIG_DISABLE_POLL
static void usbhost_pollnotify(FAR struct usbhost_state_s *priv)
{
int i;
for (i = 0; i < CONFIG_USBHID_NPOLLWAITERS; i++)
{
struct pollfd *fds = priv->fds[i];
if (fds)
{
fds->revents |= (fds->events & POLLIN);
if (fds->revents != 0)
{
uvdbg("Report events: %02x\n", fds->revents);
sem_post(fds->sem);
}
}
}
}
#endif
/**************************************************************************** /****************************************************************************
* Name: usbhost_allocclass * Name: usbhost_allocclass
* *
@ -572,6 +621,7 @@ static void usbhost_destroy(FAR void *arg)
/* Destroy the semaphores */ /* Destroy the semaphores */
sem_destroy(&priv->exclsem); sem_destroy(&priv->exclsem);
sem_destroy(&priv->waitsem);
/* Disconnect the USB host device */ /* Disconnect the USB host device */
@ -646,7 +696,7 @@ static int usbhost_kbdpoll(int argc, char *argv[])
{ {
FAR struct usbhost_state_s *priv; FAR struct usbhost_state_s *priv;
FAR struct usb_ctrlreq_s *ctrlreq; FAR struct usb_ctrlreq_s *ctrlreq;
#ifdef CONFIG_DEBUG_USB #if defined(CONFIG_DEBUG_USB) && defined(CONFIG_DEBUG_VERBOSE)
unsigned int npolls = 0; unsigned int npolls = 0;
#endif #endif
unsigned int nerrors; unsigned int nerrors;
@ -685,9 +735,9 @@ static int usbhost_kbdpoll(int argc, char *argv[])
/* Format the HID report request: /* Format the HID report request:
* *
* bmRequestType 10000001 * bmRequestType 10100001
* bRequest GET_DESCRIPTOR (0x06) * bRequest GET_REPORT (0x01)
* wValue Descriptor Type and Descriptor Index * wValue Report Type and Report Index
* wIndex Interface Number * wIndex Interface Number
* wLength Descriptor Length * wLength Descriptor Length
* Data Descriptor Data * Data Descriptor Data
@ -729,7 +779,7 @@ static int usbhost_kbdpoll(int argc, char *argv[])
else if (priv->open) else if (priv->open)
{ {
struct usbhid_kbdreport_s *rpt = (struct usbhid_kbdreport_s *)priv->buffer; struct usbhid_kbdreport_s *rpt = (struct usbhid_kbdreport_s *)priv->tbuffer;
unsigned int head; unsigned int head;
unsigned int tail; unsigned int tail;
uint8_t ascii; uint8_t ascii;
@ -745,7 +795,7 @@ static int usbhost_kbdpoll(int argc, char *argv[])
{ {
/* Is this key pressed? */ /* Is this key pressed? */
if (rpt->key[i] == USBHID_KBDUSE_NONE) if (rpt->key[i] != USBHID_KBDUSE_NONE)
{ {
/* Yes.. Add it to the buffer. */ /* Yes.. Add it to the buffer. */
@ -768,7 +818,7 @@ static int usbhost_kbdpoll(int argc, char *argv[])
* a valid, NUL character. * a valid, NUL character.
*/ */
if ((rpt->modifier & (USBHID_MODIFER_LSHIFT|USBHID_MODIFER_RSHIFT)) != 0) if ((rpt->modifier & (USBHID_MODIFER_LCTRL|USBHID_MODIFER_RCTRL)) != 0)
{ {
ascii &= 0x1f; ascii &= 0x1f;
} }
@ -777,7 +827,7 @@ static int usbhost_kbdpoll(int argc, char *argv[])
* buffer. * buffer.
*/ */
priv->buffer[head] = ascii; priv->kbdbuffer[head] = ascii;
/* Increment the head index */ /* Increment the head index */
@ -802,6 +852,25 @@ static int usbhost_kbdpoll(int argc, char *argv[])
} }
} }
/* Did we just transition from no data available to data available? */
if (head != tail && priv->headndx == priv->tailndx)
{
/* Yes.. Is there a thread waiting for keyboard data now? */
if (priv->waiting)
{
/* Yes.. wake it up */
usbhost_givesem(&priv->waitsem);
priv->waiting = false;
}
/* And wake up any threads waiting for the POLLIN event */
usbhost_pollnotify(priv);
}
/* Update the head/tail indices */ /* Update the head/tail indices */
priv->headndx = head; priv->headndx = head;
@ -813,9 +882,9 @@ static int usbhost_kbdpoll(int argc, char *argv[])
* polling is still happening. * polling is still happening.
*/ */
#ifdef CONFIG_DEBUG_USB #if defined(CONFIG_DEBUG_USB) && defined(CONFIG_DEBUG_VERBOSE)
npolls++; npolls++;
if (!(npolls & ~31) == 0) if ((npolls & 31) == 0)
{ {
udbg("Still polling: %d\n", npolls); udbg("Still polling: %d\n", npolls);
} }
@ -1193,7 +1262,7 @@ static inline int usbhost_devinit(FAR struct usbhost_state_s *priv)
uvdbg("Register driver\n"); uvdbg("Register driver\n");
usbhost_mkdevname(priv, devname); usbhost_mkdevname(priv, devname);
ret = register_driver(devname, &usbhost_fops, 0666, NULL); ret = register_driver(devname, &usbhost_fops, 0666, priv);
/* We now have to be concerned about asynchronous modification of crefs /* We now have to be concerned about asynchronous modification of crefs
* because the driver has been registerd. * because the driver has been registerd.
@ -1400,9 +1469,10 @@ static FAR struct usbhost_class_s *usbhost_create(FAR struct usbhost_driver_s *d
priv->crefs = 1; priv->crefs = 1;
/* Initialize semphores (this works okay in the interrupt context) */ /* Initialize semaphores */
sem_init(&priv->exclsem, 0, 1); sem_init(&priv->exclsem, 0, 1);
sem_init(&priv->waitsem, 0, 0);
/* Bind the driver to the storage class instance */ /* Bind the driver to the storage class instance */
@ -1530,6 +1600,16 @@ static int usbhost_disconnected(struct usbhost_class_s *class)
priv->disconnected = true; priv->disconnected = true;
ullvdbg("Disconnected\n"); ullvdbg("Disconnected\n");
/* Is there a thread waiting for keyboard data that will never come? */
if (priv->waiting)
{
/* Yes.. wake it up */
usbhost_givesem(&priv->waitsem);
priv->waiting = false;
}
/* Possibilities: /* Possibilities:
* *
* - Failure occurred before the kbdpoll task was started successfully. * - Failure occurred before the kbdpoll task was started successfully.
@ -1588,7 +1668,7 @@ static int usbhost_open(FAR struct file *filep)
/* Make sure that we have exclusive access to the private data structure */ /* Make sure that we have exclusive access to the private data structure */
DEBUGASSERT(priv->crefs > 0 && priv->crefs < USBHOST_MAX_CREFS); DEBUGASSERT(priv && priv->crefs > 0 && priv->crefs < USBHOST_MAX_CREFS);
usbhost_takesem(&priv->exclsem); usbhost_takesem(&priv->exclsem);
/* Check if the keyboard device is still connected. We need to disable /* Check if the keyboard device is still connected. We need to disable
@ -1727,6 +1807,37 @@ static ssize_t usbhost_read(FAR struct file *filep, FAR char *buffer, size_t len
} }
else else
{ {
/* Is there keyboard data now? */
while (priv->tailndx == priv->headndx)
{
/* No.. were we open non-blocking? */
if (filep->f_oflags & O_NONBLOCK)
{
/* Yes.. then return a failure */
ret = -EAGAIN;
goto errout;
}
/* Wait for data to be available */
uvdbg("Waiting...\n");
priv->waiting = true;
usbhost_givesem(&priv->exclsem);
usbhost_takesem(&priv->waitsem);
usbhost_takesem(&priv->exclsem);
/* Did the keyboard become disconnected while we were waiting */
if (priv->disconnected)
{
ret = -ENODEV;
goto errout;
}
}
/* Read data from our internal buffer of received characters */ /* Read data from our internal buffer of received characters */
for (tail = priv->tailndx, nbytes = 0; for (tail = priv->tailndx, nbytes = 0;
@ -1735,7 +1846,7 @@ static ssize_t usbhost_read(FAR struct file *filep, FAR char *buffer, size_t len
{ {
/* Copy the next keyboard character into the user buffer */ /* Copy the next keyboard character into the user buffer */
*buffer += priv->buffer[tail]; *buffer++ = priv->kbdbuffer[tail];
/* Handle wrap-around of the tail index */ /* Handle wrap-around of the tail index */
@ -1751,6 +1862,7 @@ static ssize_t usbhost_read(FAR struct file *filep, FAR char *buffer, size_t len
priv->tailndx = tail; priv->tailndx = tail;
} }
errout:
usbhost_givesem(&priv->exclsem); usbhost_givesem(&priv->exclsem);
return (ssize_t)ret; return (ssize_t)ret;
} }
@ -1782,15 +1894,86 @@ static ssize_t usbhost_write(FAR struct file *filep, FAR const char *buffer, siz
static int usbhost_poll(FAR struct file *filep, FAR struct pollfd *fds, static int usbhost_poll(FAR struct file *filep, FAR struct pollfd *fds,
bool setup) bool setup)
{ {
if (setup) FAR struct inode *inode;
FAR struct usbhost_state_s *priv;
int ret = OK;
int i;
uvdbg("Entry\n");
DEBUGASSERT(filep && filep->f_inode && fds);
inode = filep->f_inode;
priv = inode->i_private;
/* Make sure that we have exclusive access to the private data structure */
DEBUGASSERT(priv);
usbhost_takesem(&priv->exclsem);
/* Check if the keyboard is still connected. We need to disable interrupts
* momentarily to assure that there are no asynchronous disconnect events.
*/
if (priv->disconnected)
{ {
fds->revents |= (fds->events & (POLLIN|POLLOUT)); /* No... the driver is no longer bound to the class. That means that
if (fds->revents != 0) * the USB keybaord is no longer connected. Refuse any further attempts
* to access the driver.
*/
ret = -ENODEV;
}
else if (setup)
{ {
sem_post(fds->sem); /* This is a request to set up the poll. Find an availableslot for
* the poll structure reference
*/
for (i = 0; i < CONFIG_USBHID_NPOLLWAITERS; i++)
{
/* Find an available slot */
if (!priv->fds[i])
{
/* Bind the poll structure and this slot */
priv->fds[i] = fds;
fds->priv = &priv->fds[i];
break;
} }
} }
return OK;
if (i >= CONFIG_USBHID_NPOLLWAITERS)
{
fds->priv = NULL;
ret = -EBUSY;
goto errout;
}
/* Should we immediately notify on any of the requested events? Notify
* the POLLIN event if there is buffered keyboard data.
*/
if (priv->headndx != priv->tailndx)
{
usbhost_pollnotify(priv);
}
}
else
{
/* This is a request to tear down the poll. */
struct pollfd **slot = (struct pollfd **)fds->priv;
DEBUGASSERT(slot);
/* Remove all memory of the poll setup */
*slot = NULL;
fds->priv = NULL;
}
errout:
sem_post(&priv->exclsem);
return ret;
} }
#endif #endif

View File

@ -46,6 +46,7 @@
#include <unistd.h> #include <unistd.h>
#include <fcntl.h> #include <fcntl.h>
#include <sched.h> #include <sched.h>
#include <errno.h>
#include <nuttx/usb/usbhost.h> #include <nuttx/usb/usbhost.h>
@ -200,10 +201,11 @@ int user_start(int argc, char *argv[])
do do
{ {
printf("Opening device %s\n", CONFIG_EXAMPLES_HIDKBD_DEVNAME); printf("Opening device %s\n", CONFIG_EXAMPLES_HIDKBD_DEVNAME);
fflush(stdout);
fd = open(CONFIG_EXAMPLES_HIDKBD_DEVNAME, O_RDONLY); fd = open(CONFIG_EXAMPLES_HIDKBD_DEVNAME, O_RDONLY);
if (fd < 0) if (fd < 0)
{ {
printf("Failed: %d\n", errno);
fflush(stdout);
sleep(3); sleep(3);
} }
} }
@ -228,7 +230,7 @@ int user_start(int argc, char *argv[])
} }
while (nbytes >= 0); while (nbytes >= 0);
printf("Closing device %s\n", CONFIG_EXAMPLES_HIDKBD_DEVNAME); printf("Closing device %s: %d\n", CONFIG_EXAMPLES_HIDKBD_DEVNAME, (int)nbytes);
fflush(stdout); fflush(stdout);
close(fd); close(fd);
} }