summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--apps/examples/slcd/slcd_main.c1
-rw-r--r--nuttx/ChangeLog6
-rw-r--r--nuttx/configs/pcblogic-pic32mx/src/pic32mx_lcd1602.c577
-rw-r--r--nuttx/configs/stm32ldiscovery/README.txt2
-rw-r--r--nuttx/configs/stm32ldiscovery/src/stm32_lcd.c37
-rw-r--r--nuttx/include/nuttx/lcd/hd4478ou.h13
-rw-r--r--nuttx/include/nuttx/lcd/slcd_codec.h49
-rw-r--r--nuttx/include/nuttx/lcd/slcd_ioctl.h133
8 files changed, 750 insertions, 68 deletions
diff --git a/apps/examples/slcd/slcd_main.c b/apps/examples/slcd/slcd_main.c
index 1259d0642..0a0a5e0f2 100644
--- a/apps/examples/slcd/slcd_main.c
+++ b/apps/examples/slcd/slcd_main.c
@@ -47,6 +47,7 @@
#include <fcntl.h>
#include <errno.h>
+#include <nuttx/lcd/slcd_ioctl.h>
#include <nuttx/lcd/slcd_codec.h>
/****************************************************************************
diff --git a/nuttx/ChangeLog b/nuttx/ChangeLog
index e7b251d02..6d7f6ae50 100644
--- a/nuttx/ChangeLog
+++ b/nuttx/ChangeLog
@@ -4797,3 +4797,9 @@
kconfig-frontends tool (2013-5-25).
* configs/pcblogic-pic32mx/src: Renamed files using pic32mx_ vs up_
prefix. Enable building of LCD1602 LCD (2013-5-25).
+ * configs/pcblogic-pic32mx/src/pic32mx_lcd1602.c: Now uses SLCD CODEC
+ (2013-5-25)
+ * configs/stm32ldiscovery/src/stm32_lcd.c: Now supports ioctl to get
+ cursor position (2013-5-25).
+ * include/nuttx/lcd/slcd_ioctl.h: Moved ioctls commands and structures
+ from slcd_codec.h (2013-5-25)
diff --git a/nuttx/configs/pcblogic-pic32mx/src/pic32mx_lcd1602.c b/nuttx/configs/pcblogic-pic32mx/src/pic32mx_lcd1602.c
index 411ced32e..7740048a0 100644
--- a/nuttx/configs/pcblogic-pic32mx/src/pic32mx_lcd1602.c
+++ b/nuttx/configs/pcblogic-pic32mx/src/pic32mx_lcd1602.c
@@ -71,14 +71,18 @@
#include <sys/types.h>
#include <stdint.h>
#include <stdbool.h>
+#include <string.h>
#include <semaphore.h>
#include <poll.h>
#include <errno.h>
#include <debug.h>
#include <nuttx/arch.h>
+#include <nuttx/ascii.h>
#include <nuttx/fs/fs.h>
#include <nuttx/lcd/hd4478ou.h>
+#include <nuttx/lcd/slcd_ioctl.h>
+#include <nuttx/lcd/slcd_codec.h>
#include "up_arch.h"
#include "pic32mx-pmp.h"
@@ -116,6 +120,13 @@
#define GPIO_LCD_RS (GPIO_OUTPUT|GPIO_VALUE_ZERO|GPIO_PORTB|GPIO_PIN15)
+/* LCD **********************************************************************/
+
+
+#define LCD_NROWS 2
+#define LCD_NCOLUMNS 16
+#define LCD_NCHARS (LCD_NROWS * LCD_NCOLUMNS)
+
/* Debug ********************************************************************/
#ifdef CONFIG_DEBUG_LCD
@@ -130,20 +141,53 @@
* Private Type Definition
****************************************************************************/
+/* SLCD incoming stream structure */
+
+struct lcd_instream_s
+{
+ struct lib_instream_s stream;
+ FAR const char *buffer;
+ ssize_t nbytes;
+};
+
+/* Global LCD state */
+
struct lcd1602_2
{
bool initialized; /* True: Completed initialization sequence */
+ uint8_t currow; /* Current row */
+ uint8_t curcol; /* Current column */
};
/****************************************************************************
* Private Function Protototypes
****************************************************************************/
+/* Debug */
+
+ #if defined(CONFIG_DEBUG_LCD) && defined(CONFIG_DEBUG_VERBOSE)
+static void lcd_dumpstate(FAR const char *msg);
+#else
+# define lcd_dumpstate(msg)
+#endif
+
+/* Internal functions */
+
+static int lcd_getstream(FAR struct lib_instream_s *instream);
+static void lcd_wrcommand(uint8_t cmd);
+static void lcd_wrdata(uint8_t data);
+static uint8_t lcd_rddata(void);
+static uint8_t lcd_readch(uint8_t row, uint8_t column);
+static void lcd_writech(uint8_t ch, uint8_t row, uint8_t column);
+static void lcd_appendch(uint8_t ch);
+static void lcd_action(enum slcdcode_e code, uint8_t count);
+
+/* Character driver operations */
static ssize_t lcd_read(FAR struct file *, FAR char *, size_t);
static ssize_t lcd_write(FAR struct file *, FAR const char *, size_t);
+static int lcd_ioctl(FAR struct file *filp, int cmd, unsigned long arg);
#ifndef CONFIG_DISABLE_POLL
-static int lcd_poll(FAR struct file *filp, FAR struct pollfd *fds,
- bool setup);
+static int lcd_poll(FAR struct file *filp, FAR struct pollfd *fds, bool setup);
#endif
/****************************************************************************
@@ -159,7 +203,7 @@ static const struct file_operations g_lcdops =
lcd_read, /* read */
lcd_write, /* write */
0, /* seek */
- 0 /* ioctl */
+ lcd_ioctl /* ioctl */
#ifndef CONFIG_DISABLE_POLL
, lcd_poll /* poll */
#endif
@@ -174,6 +218,62 @@ static struct lcd1602_2 g_lcd1602;
****************************************************************************/
/****************************************************************************
+ * Name: lcd_dumpstate
+ ****************************************************************************/
+
+#if defined(CONFIG_DEBUG_LCD) && defined(CONFIG_DEBUG_VERBOSE)
+static void lcd_dumpstate(FAR const char *msg)
+{
+ uint8_t buffer[LCD_NCOLUMNS];
+ uint8_t row;
+ uint8_t column;
+
+ lcdvdbg("%s:\n", msg);
+ lcdvdbg(" currow: %d curcol: %d\n",
+ g_lcd1602.currow, g_lcd1602.curcol);
+
+ for (row = 0, column = 0; row < LCD_NROWS; )
+ {
+ buffer[column] = lcd_readch(row, column);
+ if (++column >= LCD_NCOLUMNS)
+ {
+ lcdvdbg(" %c%c%c%c%c%c%c%c%c%c%c%c%c%c%c%c\n",
+ buffer[0], buffer[1], buffer[2], buffer[3],
+ buffer[4], buffer[5], buffer[6], buffer[7],
+ buffer[8], buffer[9], buffer[10], buffer[11],
+ buffer[12], buffer[13], buffer[14], buffer[15]);
+
+ column = 0;
+ row++;
+ }
+ }
+}
+#endif
+
+/****************************************************************************
+ * Name: lcd_getstream
+ *
+ * Description:
+ * Get one character from the keyboard.
+ *
+ ****************************************************************************/
+
+static int lcd_getstream(FAR struct lib_instream_s *instream)
+{
+ FAR struct lcd_instream_s *lcdstream = (FAR struct lcd_instream_s *)instream;
+
+ DEBUGASSERT(lcdstream && lcdstream->buffer);
+ if (lcdstream->nbytes > 0)
+ {
+ lcdstream->nbytes--;
+ lcdstream->stream.nget++;
+ return (int)*lcdstream->buffer++;
+ }
+
+ return EOF;
+}
+
+/****************************************************************************
* Name: lcd_wrcommand
*
* Description:
@@ -231,36 +331,485 @@ static uint8_t lcd_rddata(void)
}
/****************************************************************************
+ * Name: lcd_readch
+ ****************************************************************************/
+
+static uint8_t lcd_readch(uint8_t row, uint8_t column)
+{
+ uint8_t addr;
+
+ /* Set the cursor position. Internally, the HD44780U supports a display
+ * size of up to 2x40 addressed as follows:
+ *
+ * Column 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 ... 39
+ * Row 0 00 01 02 03 04 05 06 07 08 09 0a 0b 0c 0d 0e 0f ... 27
+ * Ro1 1 40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f ... 67
+ */
+
+ addr = column;
+ if (row > 0)
+ {
+ addr |= HD4478OU_DDRAM_ROW1;
+ }
+
+ lcd_wrcommand(HD4478OU_DDRAM_AD(addr));
+
+ /* And write the character here */
+
+ return lcd_rddata();
+}
+
+/****************************************************************************
+ * Name: lcd_writech
+ ****************************************************************************/
+
+static void lcd_writech(uint8_t ch, uint8_t row, uint8_t column)
+{
+ uint8_t addr;
+
+ /* Set the cursor position. Internally, the HD44780U supports a display
+ * size of up to 2x40 addressed as follows:
+ *
+ * Column 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 ... 39
+ * Row 0 00 01 02 03 04 05 06 07 08 09 0a 0b 0c 0d 0e 0f ... 27
+ * Ro1 1 40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f ... 67
+ */
+
+ addr = column;
+ if (row > 0)
+ {
+ addr |= HD4478OU_DDRAM_ROW1;
+ }
+
+ lcd_wrcommand(HD4478OU_DDRAM_AD(addr));
+
+ /* And write the character here */
+
+ lcd_wrdata(ch);
+}
+
+/****************************************************************************
+ * Name: lcd_appendch
+ ****************************************************************************/
+
+static void lcd_appendch(uint8_t ch)
+{
+ if (g_lcd1602.curcol < LCD_NCOLUMNS)
+ {
+ lcd_writech(ch, g_lcd1602.currow, g_lcd1602.curcol);
+ g_lcd1602.curcol++;
+ }
+}
+
+/****************************************************************************
+ * Name: lcd_action
+ ****************************************************************************/
+
+static void lcd_action(enum slcdcode_e code, uint8_t count)
+{
+ lcdvdbg("Action: %d count: %d\n", code, count);
+ lcd_dumpstate("BEFORE ACTION");
+
+ switch (code)
+ {
+ /* Erasure */
+
+ case SLCDCODE_BACKDEL: /* Backspace (backward delete) N characters */
+ {
+ /* If we are at the home position, then ignore the action */
+
+ if (g_lcd1602.curcol < 1)
+ {
+ break;
+ }
+
+ /* Otherwise, BACKDEL is like moving the cursor back one then doing a
+ * forward deletion. Decrement the cursor position and fall through.
+ */
+
+ g_lcd1602.curcol--;
+ }
+
+ case SLCDCODE_FWDDEL: /* DELete (forward delete) N characters moving text */
+ {
+ uint8_t ch;
+ int i;
+
+ /* Move all characters after the current cursor position left by one */
+
+ for (i = g_lcd1602.curcol + 1; i < LCD_NCOLUMNS - 1; i++)
+ {
+ ch = lcd_readch(g_lcd1602.currow, i);
+ lcd_writech(ch, g_lcd1602.currow, i - 1);
+ }
+
+ /* Erase the last character on the display */
+
+ lcd_writech(' ', g_lcd1602.currow, LCD_NCOLUMNS - 1);
+ }
+ break;
+
+ case SLCDCODE_ERASE: /* Erase N characters from the cursor position */
+ {
+ int last;
+ int i;
+
+ /* Get the last position to clear and make sure that the last
+ * position is on the SLCD.
+ */
+
+ last = g_lcd1602.curcol + count - 1;
+ if (last >= LCD_NCOLUMNS)
+ {
+ last = LCD_NCOLUMNS - 1;
+ }
+
+ /* Erase N characters after the current cursor position left by one */
+
+ for (i = g_lcd1602.curcol; i < last; i++)
+ {
+ lcd_writech(' ', g_lcd1602.currow, i);
+ }
+ }
+ break;
+
+ case SLCDCODE_CLEAR: /* Home the cursor and erase the entire display */
+ {
+ /* Clear the display */
+
+ lcd_wrcommand(HD4478OU_CLEAR);
+
+ /* And home the cursor */
+
+ g_lcd1602.currow = 0;
+ g_lcd1602.curcol = 0;
+ }
+ break;
+
+ case SLCDCODE_ERASEEOL: /* Erase from the cursor position to the end of line */
+ {
+ int i;
+
+ /* Erase characters after the current cursor position to the end of the line */
+
+ for (i = g_lcd1602.curcol; i < LCD_NCOLUMNS; i++)
+ {
+ lcd_writech(' ', g_lcd1602.currow, i);
+ }
+ }
+ break;
+
+ /* Cursor movement */
+
+ case SLCDCODE_HOME: /* Cursor home */
+ {
+ g_lcd1602.currow = 0;
+ g_lcd1602.curcol = 0;
+ }
+ break;
+
+ case SLCDCODE_END: /* Cursor end */
+ {
+ g_lcd1602.curcol = LCD_NCOLUMNS - 1;
+ }
+ break;
+
+ case SLCDCODE_LEFT: /* Cursor left by N characters */
+ {
+ /* Don't permit movement past the beginning of the SLCD */
+
+ if (g_lcd1602.curcol > 0)
+ {
+ g_lcd1602.curcol--;
+ }
+ }
+ break;
+
+ case SLCDCODE_RIGHT: /* Cursor right by N characters */
+ {
+ /* Don't permit movement past the end of the SLCD */
+
+ if (g_lcd1602.curcol < (LCD_NCOLUMNS - 1))
+ {
+ g_lcd1602.curcol++;
+ }
+ }
+ break;
+
+ case SLCDCODE_UP: /* Cursor up by N lines */
+ {
+ /* Don't permit movement past the top of the SLCD */
+
+ if (g_lcd1602.currow > 0)
+ {
+ g_lcd1602.currow--;
+ }
+ }
+ break;
+
+ case SLCDCODE_DOWN: /* Cursor down by N lines */
+ {
+ /* Don't permit movement past the bottom of the SLCD */
+
+ if (g_lcd1602.curcol < (LCD_NCOLUMNS - 1))
+ {
+ g_lcd1602.curcol++;
+ }
+ }
+ break;
+
+ case SLCDCODE_PAGEUP: /* Cursor up by N pages */
+ case SLCDCODE_PAGEDOWN: /* Cursor down by N pages */
+ break; /* Not supportable on this SLCD */
+
+ /* Blinking */
+
+ case SLCDCODE_BLINKSTART: /* Start blinking with current cursor position */
+ case SLCDCODE_BLINKEND: /* End blinking after the current cursor position */
+ case SLCDCODE_BLINKOFF: /* Turn blinking off */
+ break; /* Not implemented */
+
+ /* These are actually unreportable errors */
+
+ default:
+ case SLCDCODE_NORMAL: /* Not a special keycode */
+ break;
+ }
+
+ lcd_dumpstate("AFTER ACTION");
+}
+
+/****************************************************************************
* Name: lcd_read
****************************************************************************/
static ssize_t lcd_read(FAR struct file *filp, FAR char *buffer, size_t len)
{
- int i;
+ uint8_t row;
+ uint8_t column;
+ int nread;
+
+ /* Try to read the entire display. Notice that the seek offset
+ * (filp->f_pos) is ignored. It probably should be taken into account
+ * and also updated after each read and write.
+ */
- for (i = 0; i < len; i++)
+ row = 0;
+ column = 0;
+
+ for (nread = 0; nread < len; nread++)
{
- *buffer++ = lcd_rddata();
+ *buffer++ = lcd_readch(row, column);
+ if (++column >= LCD_NCOLUMNS)
+ {
+ column = 0;
+ if (++row >= LCD_NROWS)
+ {
+ break;
+ }
+ }
}
- return len;
+ return nread;
}
/****************************************************************************
* Name: lcd_write
****************************************************************************/
-static ssize_t lcd_write(FAR struct file *filp, FAR const char *buffer, size_t len)
+static ssize_t lcd_write(FAR struct file *filp, FAR const char *buffer,
+ size_t len)
{
- int i;
+ struct lcd_instream_s instream;
+ struct slcdstate_s state;
+ enum slcdret_e result;
+ uint8_t ch;
+ uint8_t count;
+ uint8_t prev = ' ';
+ bool valid = false;
+
+ /* Initialize the stream for use with the SLCD CODEC */
+
+ instream.stream.get = lcd_getstream;
+ instream.stream.nget = 0;
+ instream.buffer = buffer;
+ instream.nbytes = len;
+
+ /* Prime the pump */
- for (i = 0; i < len; i++)
+ memset(&state, 0, sizeof(struct slcdstate_s));
+ result = slcd_decode(&instream.stream, &state, &prev, &count);
+
+ lcdvdbg("slcd_decode returned result=%d char=%d count=%d\n",
+ result, prev, count);
+
+ switch (result)
+ {
+ case SLCDRET_CHAR:
+ valid = true;
+ break;
+
+ case SLCDRET_SPEC:
+ {
+ lcd_action((enum slcdcode_e)prev, count);
+ prev = ' ';
+ }
+ break;
+
+ case SLCDRET_EOF:
+ return 0;
+ }
+
+ /* Now decode and process every byte in the input buffer */
+
+ while ((result = slcd_decode(&instream.stream, &state, &ch, &count)) != SLCDRET_EOF)
+ {
+ lcdvdbg("slcd_decode returned result=%d char=%d count=%d\n",
+ result, ch, count);
+
+ if (result == SLCDRET_CHAR) /* A normal character was returned */
+ {
+ /* Check for ASCII control characters */
+
+ if (ch < ASCII_SPACE)
+ {
+ /* All are ignored except for backspace and carriage return */
+
+ if (ch == ASCII_BS)
+ {
+ lcd_action(SLCDCODE_BACKDEL, 1);
+ }
+ else if (ch == ASCII_CR)
+ {
+ lcd_action(SLCDCODE_HOME, 0);
+ }
+ }
+
+ /* Handle characters decoreated with a period or a colon */
+
+ else if (ch == '.')
+ {
+ /* Write the previous character with the decimal point appended */
+
+ lcd_appendch(prev);
+ prev = ' ';
+ valid = false;
+ }
+ else if (ch == ':')
+ {
+ /* Write the previous character with the colon appended */
+
+ lcd_appendch(prev);
+ prev = ' ';
+ valid = false;
+ }
+
+ /* Handle ASCII_DEL */
+
+ else if (ch == ASCII_DEL)
+ {
+ lcd_action(SLCDCODE_FWDDEL, 1);
+ }
+
+ /* The rest of the 7-bit ASCII characters are fair game */
+
+ else if (ch < 128)
+ {
+ /* Write the previous character if it valid */
+
+ if (valid)
+ {
+ lcd_appendch(prev);
+ }
+
+ /* There is now a valid output character */
+
+ prev = ch;
+ valid = true;
+ }
+ }
+ else /* (result == SLCDRET_SPEC) */ /* A special SLCD action was returned */
+ {
+ lcd_action((enum slcdcode_e)ch, count);
+ }
+ }
+
+ /* Handle any unfinished output */
+
+ if (valid)
+ {
+ lcd_appendch(prev);
+ }
+
+ /* Assume that the entire input buffer was processed */
+
+ return (ssize_t)len;
+}
+
+/****************************************************************************
+ * Name: lcd_ioctl
+ ****************************************************************************/
+
+static int lcd_ioctl(FAR struct file *filp, int cmd, unsigned long arg)
+{
+ switch (cmd)
{
- uint8_t data = *buffer++;
- lcd_wrdata(data);
+
+ /* SLCDIOC_GEOMETRY: Get the SLCD geometry (rows x characters)
+ *
+ * argument: Pointer to struct slcd_geometry_s in which values will be
+ * returned
+ */
+
+ case SLCDIOC_GEOMETRY:
+ {
+ FAR struct slcd_geometry_s *geo = (FAR struct slcd_geometry_s *)((uintptr_t)arg);
+
+ lcdvdbg("SLCDIOC_GEOMETRY: nrows=%d ncolumns=%d\n", LCD_NROWS, LCD_NCOLUMNS);
+
+ if (!geo)
+ {
+ return -EINVAL;
+ }
+
+ geo->nrows = LCD_NROWS;
+ geo->ncolumns = LCD_NCOLUMNS;
+ geo->nbars = 0;
+ }
+ break;
+
+ /* SLCDIOC_CURPOS: Get the SLCD cursor positioni (rows x characters)
+ *
+ * argument: Pointer to struct slcd_curpos_s in which values will be
+ * returned
+ */
+
+
+ case SLCDIOC_CURPOS:
+ {
+ FAR struct slcd_curpos_s *curpos = (FAR struct slcd_curpos_s *)((uintptr_t)arg);
+
+ lcdvdbg("SLCDIOC_CURPOS: row=%d column=%d\n", g_lcd1602.currow, g_lcd1602.curcol);
+
+ if (!curpos)
+ {
+ return -EINVAL;
+ }
+
+ curpos->row = g_lcd1602.currow;
+ curpos->column = g_lcd1602.curcol;
+ }
+ break;
+
+ case SLCDIOC_SETBAR: /* SLCDIOC_SETBAR: Set bars on a bar display */
+ case SLCDIOC_GETCONTRAST: /* SLCDIOC_GETCONTRAST: Get the current contrast setting */
+ case SLCDIOC_MAXCONTRAST: /* SLCDIOC_MAXCONTRAST: Get the maximum contrast setting */
+ case SLCDIOC_SETCONTRAST: /* SLCDIOC_SETCONTRAST: Set the contrast to a new value */
+ default:
+ return -ENOTTY;
}
- return len;
+ return OK;
}
/****************************************************************************
@@ -338,7 +887,7 @@ int up_lcd1602_initialize(void)
* Enable PMRD/PMWR, PMENB, and the PMP.
*/
-
+
regval = (PMP_CON_RDSP | PMP_CON_WRSP | PMP_CON_ALP |
PMP_CON_CSF_ADDR1415 | PMP_CON_PTRDEN | PMP_CON_PTWREN |
PMP_CON_ADRMUX_NONE | PMP_CON_ON);
diff --git a/nuttx/configs/stm32ldiscovery/README.txt b/nuttx/configs/stm32ldiscovery/README.txt
index e1f7b35fe..88234b761 100644
--- a/nuttx/configs/stm32ldiscovery/README.txt
+++ b/nuttx/configs/stm32ldiscovery/README.txt
@@ -788,7 +788,7 @@ Configuration sub-directories
4. To enable SLCD support:
Board Selection:
- CONFIG_ARCH_LEDS=y : Disable LED support
+ CONFIG_ARCH_LEDS=n : Disable LED support
Library Routines:
CONFIG_LIB_SLCDCODEC=y : Enable the SLCD CODEC
diff --git a/nuttx/configs/stm32ldiscovery/src/stm32_lcd.c b/nuttx/configs/stm32ldiscovery/src/stm32_lcd.c
index ace9dbc9c..abc9a0644 100644
--- a/nuttx/configs/stm32ldiscovery/src/stm32_lcd.c
+++ b/nuttx/configs/stm32ldiscovery/src/stm32_lcd.c
@@ -59,6 +59,7 @@
#include <nuttx/ascii.h>
#include <nuttx/streams.h>
#include <nuttx/fs/fs.h>
+#include <nuttx/lcd/slcd_ioctl.h>
#include <nuttx/lcd/slcd_codec.h>
#include "up_arch.h"
@@ -313,8 +314,8 @@ static void slcd_writebar(void);
static inline uint16_t slcd_mapch(uint8_t ch);
static inline void slcd_writemem(uint16_t segset, int curpos);
static void slcd_writech(uint8_t ch, uint8_t curpos, uint8_t options);
-static inline void slcd_appendch(uint8_t ch, uint8_t options);
-static inline void slcd_action(enum slcdcode_e code, uint8_t count);
+static void slcd_appendch(uint8_t ch, uint8_t options);
+static void slcd_action(enum slcdcode_e code, uint8_t count);
/* Character driver methods */
@@ -999,7 +1000,7 @@ static void slcd_action(enum slcdcode_e code, uint8_t count)
case SLCDCODE_RIGHT: /* Cursor right by N characters */
{
- /* Don't permit movement past the lcd of the SLCD */
+ /* Don't permit movement past the end of the SLCD */
if (g_slcdstate.curpos < (SLCD_NCHARS - 1))
{
@@ -1040,6 +1041,11 @@ static ssize_t slcd_read(FAR struct file *filp, FAR char *buffer, size_t len)
int ret = 0;
int i;
+ /* Try to read the entire display. Notice that the seek offset
+ * (filp->f_pos) is ignored. It probably should be taken into account
+ * and also updated after each read and write.
+ */
+
for (i = 0; i < SLCD_NCHARS && ret < len; i++)
{
/* Return the character */
@@ -1233,6 +1239,29 @@ static int slcd_ioctl(FAR struct file *filp, int cmd, unsigned long arg)
}
break;
+ /* SLCDIOC_CURPOS: Get the SLCD cursor positioni (rows x characters)
+ *
+ * argument: Pointer to struct slcd_curpos_s in which values will be
+ * returned
+ */
+
+
+ case SLCDIOC_CURPOS:
+ {
+ FAR struct slcd_curpos_s *curpos = (FAR struct slcd_curpos_s *)((uintptr_t)arg);
+
+ lcdvdbg("SLCDIOC_CURPOS: row=0 column=%d\n", g_slcdstate.curpos);
+
+ if (!curpos)
+ {
+ return -EINVAL;
+ }
+
+ curpos->row = 0;
+ curpos->column = g_slcdstate.curpos;
+ }
+ break;
+
/* SLCDIOC_SETBAR: Set bars on a bar display
*
* argument: 32-bit bitset, with each bit corresponding to one bar.
@@ -1477,7 +1506,7 @@ int stm32_slcd_initialize(void)
/* Register the LCD device driver */
- ret = register_driver("/dev/slcd", &g_slcdops, 0644, (FAR struct file_operations *)&g_slcdops);
+ ret = register_driver("/dev/slcd", &g_slcdops, 0644, &g_slcdstate);
g_slcdstate.initialized = true;
/* Then clear the display */
diff --git a/nuttx/include/nuttx/lcd/hd4478ou.h b/nuttx/include/nuttx/lcd/hd4478ou.h
index dc92e9fb9..7f969379d 100644
--- a/nuttx/include/nuttx/lcd/hd4478ou.h
+++ b/nuttx/include/nuttx/lcd/hd4478ou.h
@@ -79,6 +79,19 @@
#define HD4478OU_BUSY(bf,ac) ((bf) << 7 | (ac))
+/* DDRAM Addressing.
+ *
+ * Internally, the HD44780U supports a display size of up to 2x40 addressed
+ * as follows:
+ *
+ * Column 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 ... 39
+ * Row 0 00 01 02 03 04 05 06 07 08 09 0a 0b 0c 0d 0e 0f ... 27
+ * Ro1 1 40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f ... 67
+ */
+
+#define HD4478OU_DDRAM_ROW0 0x00
+#define HD4478OU_DDRAM_ROW1 0x40
+
/********************************************************************************************
* Pre-processor Definitions
********************************************************************************************/
diff --git a/nuttx/include/nuttx/lcd/slcd_codec.h b/nuttx/include/nuttx/lcd/slcd_codec.h
index a1037eb1f..3c7d44b67 100644
--- a/nuttx/include/nuttx/lcd/slcd_codec.h
+++ b/nuttx/include/nuttx/lcd/slcd_codec.h
@@ -43,52 +43,12 @@
#include <nuttx/config.h>
#include <nuttx/streams.h>
-#include <nuttx/fs/ioctl.h>
#ifdef CONFIG_LIB_SLCDCODEC
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
-/* IOCTL commands that may be supported by some SLCD drivers */
-
-/* SLCDIOC_GEOMETRY: Get the SLCD geometry (rows x characters)
- *
- * argument: Pointer to struct slcd_geometry_s in which values will be
- * returned
- */
-
-#define SLCDIOC_GEOMETRY _SLCDIOC(0x0001)
-
-/* SLCDIOC_SETBAR: Set bars on a bar display
- *
- * argument: 32-bit bitset, with each bit corresponding to one bar.
- */
-
-#define SLCDIOC_SETBAR _SLCDIOC(0x0002)
-
-/* SLCDIOC_GETCONTRAST: Get the current contrast setting
- *
- * argument: Pointer type int that will receive the current contrast
- * setting
- */
-
-#define SLCDIOC_GETCONTRAST _SLCDIOC(0x0003)
-
-/* SLCDIOC_MAXCONTRAST: Get the maximum contrast setting
- *
- * argument: Pointer type int that will receive the maximum contrast
- * setting
- */
-
-#define SLCDIOC_MAXCONTRAST _SLCDIOC(0x0004)
-
-/* SLCDIOC_SETCONTRAST: Set the contrast to a new value
- *
- * argument: The new contrast value
- */
-
-#define SLCDIOC_SETCONTRAST _SLCDIOC(0x0005)
/****************************************************************************
* Public Types
@@ -142,15 +102,6 @@ enum slcdret_e
SLCDRET_EOF /* An EOF (or possibly an error) occurred */
};
-/* Used with the SLCDIOC_GEOMETRY ioctl call */
-
-struct slcd_geometry_s
-{
- uint16_t nrows; /* Number of the rows on the SLCD */
- uint16_t ncolumns; /* Number of characters in one row on the SLCD */
- uint8_t nbars; /* Number of bars supported by the SLCD */
-};
-
/* Working data needed by slcd_encode that must be provided and initialized
* by the caller.
*/
diff --git a/nuttx/include/nuttx/lcd/slcd_ioctl.h b/nuttx/include/nuttx/lcd/slcd_ioctl.h
new file mode 100644
index 000000000..1e668702e
--- /dev/null
+++ b/nuttx/include/nuttx/lcd/slcd_ioctl.h
@@ -0,0 +1,133 @@
+/************************************************************************************
+ * include/nuttx/input/slcd_ioctl.h
+ * IOCTL commands for segment LCDs
+ *
+ * Copyright (C) 2013 Gregory Nutt. All rights reserved.
+ * Author: Gregory Nutt <gnutt@nuttx.org>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in
+ * the documentation and/or other materials provided with the
+ * distribution.
+ * 3. Neither the name NuttX nor the names of its contributors may be
+ * used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+ * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+ * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
+ * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
+ * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
+ * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
+ * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ *
+ ************************************************************************************/
+
+#ifndef __INCLUDE_NUTTX_INPUT_SLCD_IOCTL_H
+#define __INCLUDE_NUTTX_INPUT_SLCD_IOCTL_H
+
+/****************************************************************************
+ * Included Files
+ ****************************************************************************/
+
+#include <nuttx/config.h>
+#include <nuttx/fs/ioctl.h>
+
+/****************************************************************************
+ * Pre-processor Definitions
+ ****************************************************************************/
+/* IOCTL commands that may be supported by some SLCD drivers */
+
+/* SLCDIOC_GEOMETRY: Get the SLCD geometry (rows x characters)
+ *
+ * argument: Pointer to struct slcd_geometry_s in which values will be
+ * returned
+ */
+
+#define SLCDIOC_GEOMETRY _SLCDIOC(0x0001)
+
+/* SLCDIOC_CURPOS: Get the SLCD cursor positioni (rows x characters)
+ *
+ * argument: Pointer to struct slcd_curpos_s in which values will be
+ * returned
+ */
+
+#define SLCDIOC_CURPOS _SLCDIOC(0x0002)
+
+/* SLCDIOC_SETBAR: Set bars on a bar display
+ *
+ * argument: 32-bit bitset, with each bit corresponding to one bar.
+ */
+
+#define SLCDIOC_SETBAR _SLCDIOC(0x0003)
+
+/* SLCDIOC_GETCONTRAST: Get the current contrast setting
+ *
+ * argument: Pointer type int that will receive the current contrast
+ * setting
+ */
+
+#define SLCDIOC_GETCONTRAST _SLCDIOC(0x0004)
+
+/* SLCDIOC_MAXCONTRAST: Get the maximum contrast setting
+ *
+ * argument: Pointer type int that will receive the maximum contrast
+ * setting
+ */
+
+#define SLCDIOC_MAXCONTRAST _SLCDIOC(0x0005)
+
+/* SLCDIOC_SETCONTRAST: Set the contrast to a new value
+ *
+ * argument: The new contrast value
+ */
+
+#define SLCDIOC_SETCONTRAST _SLCDIOC(0x0006)
+
+/****************************************************************************
+ * Public Types
+ ****************************************************************************/
+
+/* Used with the SLCDIOC_GEOMETRY ioctl call */
+
+struct slcd_geometry_s
+{
+ uint16_t nrows; /* Number of the rows on the SLCD */
+ uint16_t ncolumns; /* Number of characters in one row on the SLCD */
+ uint8_t nbars; /* Number of bars supported by the SLCD */
+};
+
+/* Used with the SLCDIOC_CURPOS ioctl call */
+
+struct slcd_curpos_s
+{
+ uint16_t row; /* Current row (zero-based) */
+ uint16_t column; /* Current column (zero-based) */
+};
+
+/****************************************************************************
+ * Public Function Prototypes
+ ****************************************************************************/
+
+#ifdef __cplusplus
+extern "C"
+{
+#endif
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __INCLUDE_NUTTX_INPUT_SLCD_IOCTL_H */
+