nuttx/configs/shenzhou/src/stm32_ili93xx.c

1989 lines
64 KiB
C

/************************************************************************************
* configs/shenzhou/src/stm32_ili93xx.c
*
* Copyright (C) 2012 Gregory Nutt. All rights reserved.
* Authors: Gregory Nutt <gnutt@nuttx.org>
* Diego Sanchez <dsanchez@nx-engineering.com>
*
* 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.
*
************************************************************************************/
/* TFT LCD
*
* -- ---- -------------- -----------------------------------------------------------
* PN NAME SIGNAL NOTES
* -- ---- -------------- -----------------------------------------------------------
* 37 PB2 DATA_LE To TFT LCD (CN13, ping 28)
* 96 PB9 F_CS To both the TFT LCD (CN13, pin 30) and to the W25X16 SPI FLASH
* 34 PC5 TP_INT JP6. To TFT LCD (CN13) module (CN13, pin 26)
* 65 PC8 LCD_CS Active low: Pulled high (CN13, pin 19)
* 66 PC9 TP_CS Active low: Pulled high (CN13, pin 31)
* 78 PC10 SPI3_SCK To TFT LCD (CN13, pin 29)
* 79 PC11 SPI3_MISO To TFT LCD (CN13, pin 25)
* 80 PC12 SPI3_MOSI To TFT LCD (CN13, pin 27)
* 58 PD11 SD_CS Active low: Pulled high (See also TFT LCD CN13, pin 32)
* 60 PD13 LCD_RS To TFT LCD (CN13, pin 20)
* 61 PD14 LCD_WR To TFT LCD (CN13, pin 21). Schematic is wrong LCD_WR is PB14.
* 62 PD15 LCD_RD To TFT LCD (CN13, pin 22)
* 97 PE0 DB00 To TFT LCD (CN13, pin 3)
* 98 PE1 DB01 To TFT LCD (CN13, pin 4)
* 1 PE2 DB02 To TFT LCD (CN13, pin 5)
* 2 PE3 DB03 To TFT LCD (CN13, pin 6)
* 3 PE4 DB04 To TFT LCD (CN13, pin 7)
* 4 PE5 DB05 To TFT LCD (CN13, pin 8)
* 5 PE6 DB06 To TFT LCD (CN13, pin 9)
* 38 PE7 DB07 To TFT LCD (CN13, pin 10)
* 39 PE8 DB08 To TFT LCD (CN13, pin 11)
* 40 PE9 DB09 To TFT LCD (CN13, pin 12)
* 41 PE10 DB10 To TFT LCD (CN13, pin 13)
* 42 PE11 DB11 To TFT LCD (CN13, pin 16)
* 43 PE12 DB12 To TFT LCD (CN13, pin 15)
* 44 PE13 DB13 To TFT LCD (CN13, pin 16)
* 45 PE14 DB14 To TFT LCD (CN13, pin 17)
* 46 PE15 DB15 To TFT LCD (CN13, pin 18)
*
* NOTE: The backlight signal NC_BL (CN13, pin 24) is pulled high and not under
* software control
*
* On LCD module:
* -- -------------- -------------------------------------------------------------------
* PN SIGNAL NOTES
* -- -------------- -------------------------------------------------------------------
* 3 DB01 To LCD DB1
* 4 DB00 To LCD DB0
* 5 DB03 To LCD DB3
* 6 DB02 To LCD DB2
* 7 DB05 To LCD DB5
* 8 DB04 To LCD DB4
* 9 DB07 To LCD DB7
* 10 DB06 To LCD DB6
* 11 DB09 To LCD DB9
* 12 DB08 To LCD DB8
* 13 DB11 To LCD DB11
* 14 DB10 To LCD DB10
* 15 DB13 To LCD DB13
* 16 DB12 To LCD DB12
* 17 DB15 To LCD DB15
* 18 DB14 To LCD DB14
* 19 RS To LCD RS
* 20 /LCD_CS To LCD CS
* 21 /RD To LCD RD
* 22 /WR To LCD WR
* 23 BL_EN (Not referenced)
* 24 /RESET
* 25 /INT To Touch IC /INT
* 26 MISO To Touch IC DOUT; To AT45DB161B SO; To SD card DAT0
* 27 LE To 74HC573 that controls LCD 8-bit/16-bit mode
* 28 MOSI To Touch IC DIN; To AT45DB161B SI; To SD card CMD
* 29 /DF_CS To AT45DB161B Data Flash /CS
* 30 SCLK To Touch IC DCLK; To AT45DB161B SCK; To SD card CLK
* 31 /SD_CS To SD card /CS
* 31 /TP_CS To Touch IC CS
*/
/************************************************************************************
* Included Files
************************************************************************************/
#include <nuttx/config.h>
#include <sys/types.h>
#include <stdint.h>
#include <stdbool.h>
#include <string.h>
#include <errno.h>
#include <debug.h>
#include <nuttx/arch.h>
#include <nuttx/board.h>
#include <nuttx/spi/spi.h>
#include <nuttx/lcd/lcd.h>
#include <arch/board/board.h>
#include "up_arch.h"
#include "stm32.h"
#include "shenzhou-internal.h"
/************************************************************************************
* Pre-processor Definitions
************************************************************************************/
/* Configuration **********************************************************************/
/* CONFIG_STM32_ILI1505_DISABLE may be defined to disable the LCD_ILI1505
* CONFIG_STM32_ILI9300_DISABLE may be defined to disable the LCD_ILI9300
* CONFIG_STM32_ILI9320_DISABLE may be defined to disable the LCD_ILI9320
* CONFIG_STM32_ILI9321_DISABLE may be defined to disable the LCD_ILI9321
* CONFIG_STM32_ILI9325_DISABLE may be defined to disabled the LCD_ILI9325
* CONFIG_STM32_ILI9328_DISABLE may be defined to disabled the LCD_ILI9328
* CONFIG_STM32_ILI9331_DISABLE may be defined to disabled the LCD_ILI9331
* CONFIG_STM32_ILI9919_DISABLE may be defined to disabled the LCD_ILI9919
*/
#undef HAVE_LCD
#if !defined(CONFIG_STM32_ILI1505_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9300_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9320_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9321_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9325_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9328_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9331_DISABLE)
# define HAVE_LCD 1
#elif !defined(CONFIG_STM32_ILI9919_DISABLE)
# define HAVE_LCD 1
#endif
#ifdef HAVE_LCD
/* Check contrast selection */
#if !defined(CONFIG_LCD_MAXCONTRAST)
# define CONFIG_LCD_MAXCONTRAST 1
#endif
/* Check power setting */
#if !defined(CONFIG_LCD_MAXPOWER) || CONFIG_LCD_MAXPOWER < 1
# define CONFIG_LCD_MAXPOWER 1
#endif
#if CONFIG_LCD_MAXPOWER > 255
# error "CONFIG_LCD_MAXPOWER must be less than 256 to fit in uint8_t"
#endif
/* Check orientation */
#if defined(CONFIG_LCD_PORTRAIT)
# if defined(CONFIG_LCD_LANDSCAPE) || defined(CONFIG_LCD_RLANDSCAPE) || defined(CONFIG_LCD_RPORTRAIT)
# error "Cannot define both portrait and any other orientations"
# endif
#elif defined(CONFIG_LCD_RPORTRAIT)
# if defined(CONFIG_LCD_LANDSCAPE) || defined(CONFIG_LCD_RLANDSCAPE)
# error "Cannot define both rportrait and any other orientations"
# endif
#elif defined(CONFIG_LCD_LANDSCAPE)
# ifdef CONFIG_LCD_RLANDSCAPE
# error "Cannot define both landscape and any other orientations"
# endif
#elif !defined(CONFIG_LCD_RLANDSCAPE)
# define CONFIG_LCD_LANDSCAPE 1
#endif
#undef CONFIG_LCD_FASTCONFIG
#define CONFIG_LCD_FASTCONFIG 1
/* Define CONFIG_DEBUG_LCD to enable detailed LCD debug output. Verbose debug must
* also be enabled.
*/
#ifndef CONFIG_DEBUG
# undef CONFIG_DEBUG_VERBOSE
# undef CONFIG_DEBUG_GRAPHICS
# undef CONFIG_DEBUG_LCD
# undef CONFIG_LCD_REGDEBUG
#endif
#ifndef CONFIG_DEBUG_VERBOSE
# undef CONFIG_DEBUG_LCD
#endif
/* Display/Color Properties ***********************************************************/
/* Display Resolution */
#if defined(CONFIG_LCD_LANDSCAPE) || defined(CONFIG_LCD_RLANDSCAPE)
# define STM32_XRES 320
# define STM32_YRES 240
#else
# define STM32_XRES 240
# define STM32_YRES 320
#endif
/* Color depth and format */
#define STM32_BPP 16
#define STM32_COLORFMT FB_FMT_RGB16_565
/* Shenzhou LCD Hardware Definitions **************************************************/
/* LCD /CS is CE4, Bank 3 of NOR/SRAM Bank 1~4 */
#define STM32_LCDBASE ((uintptr_t)(0x60000000 | 0x08000000))
#define LCD ((struct lcd_regs_s *)STM32_LCDBASE)
#define LCD_REG_0 0x00
#define LCD_REG_1 0x01
#define LCD_REG_2 0x02
#define LCD_REG_3 0x03
#define LCD_REG_4 0x04
#define LCD_REG_5 0x05
#define LCD_REG_6 0x06
#define LCD_REG_7 0x07
#define LCD_REG_8 0x08
#define LCD_REG_9 0x09
#define LCD_REG_10 0x0a
#define LCD_REG_11 0x0b
#define LCD_REG_12 0x0c
#define LCD_REG_13 0x0d
#define LCD_REG_14 0x0e
#define LCD_REG_15 0x0f
#define LCD_REG_16 0x10
#define LCD_REG_17 0x11
#define LCD_REG_18 0x12
#define LCD_REG_19 0x13
#define LCD_REG_20 0x14
#define LCD_REG_21 0x15
#define LCD_REG_22 0x16
#define LCD_REG_23 0x17
#define LCD_REG_24 0x18
#define LCD_REG_25 0x19
#define LCD_REG_26 0x1a
#define LCD_REG_27 0x1b
#define LCD_REG_28 0x1c
#define LCD_REG_29 0x1d
#define LCD_REG_30 0x1e
#define LCD_REG_31 0x1f
#define LCD_REG_32 0x20
#define LCD_REG_33 0x21
#define LCD_REG_34 0x22
#define LCD_REG_36 0x24
#define LCD_REG_37 0x25
#define LCD_REG_38 0x26
#define LCD_REG_39 0x27
#define LCD_REG_40 0x28
#define LCD_REG_41 0x29
#define LCD_REG_42 0x2a
#define LCD_REG_43 0x2b
#define LCD_REG_45 0x2d
#define LCD_REG_48 0x30
#define LCD_REG_49 0x31
#define LCD_REG_50 0x32
#define LCD_REG_51 0x33
#define LCD_REG_52 0x34
#define LCD_REG_53 0x35
#define LCD_REG_54 0x36
#define LCD_REG_55 0x37
#define LCD_REG_56 0x38
#define LCD_REG_57 0x39
#define LCD_REG_58 0x3a
#define LCD_REG_59 0x3b
#define LCD_REG_60 0x3c
#define LCD_REG_61 0x3d
#define LCD_REG_62 0x3e
#define LCD_REG_63 0x3f
#define LCD_REG_64 0x40
#define LCD_REG_65 0x41
#define LCD_REG_66 0x42
#define LCD_REG_67 0x43
#define LCD_REG_68 0x44
#define LCD_REG_69 0x45
#define LCD_REG_70 0x46
#define LCD_REG_71 0x47
#define LCD_REG_72 0x48
#define LCD_REG_73 0x49
#define LCD_REG_74 0x4a
#define LCD_REG_75 0x4b
#define LCD_REG_76 0x4c
#define LCD_REG_77 0x4d
#define LCD_REG_78 0x4e
#define LCD_REG_79 0x4f
#define LCD_REG_80 0x50
#define LCD_REG_81 0x51
#define LCD_REG_82 0x52
#define LCD_REG_83 0x53
#define LCD_REG_96 0x60
#define LCD_REG_97 0x61
#define LCD_REG_106 0x6a
#define LCD_REG_118 0x76
#define LCD_REG_128 0x80
#define LCD_REG_129 0x81
#define LCD_REG_130 0x82
#define LCD_REG_131 0x83
#define LCD_REG_132 0x84
#define LCD_REG_133 0x85
#define LCD_REG_134 0x86
#define LCD_REG_135 0x87
#define LCD_REG_136 0x88
#define LCD_REG_137 0x89
#define LCD_REG_139 0x8b
#define LCD_REG_140 0x8c
#define LCD_REG_141 0x8d
#define LCD_REG_143 0x8f
#define LCD_REG_144 0x90
#define LCD_REG_145 0x91
#define LCD_REG_146 0x92
#define LCD_REG_147 0x93
#define LCD_REG_148 0x94
#define LCD_REG_149 0x95
#define LCD_REG_150 0x96
#define LCD_REG_151 0x97
#define LCD_REG_152 0x98
#define LCD_REG_153 0x99
#define LCD_REG_154 0x9a
#define LCD_REG_157 0x9d
#define LCD_REG_164 0xa4
#define LCD_REG_192 0xc0
#define LCD_REG_193 0xc1
#define LCD_REG_227 0xe3
#define LCD_REG_229 0xe5
#define LCD_REG_231 0xe7
#define LCD_REG_239 0xef
/* LCD IDs */
#define ILI1505_ID 0x1505
#define ILI9300_ID 0x9300
#define ILI9320_ID 0x9320
#define ILI9321_ID 0x9321
#define ILI9325_ID 0x9325
#define ILI9328_ID 0x9328
#define ILI9331_ID 0x9331
#define ILI9919_ID 0x9919
/* Debug ******************************************************************************/
#ifdef CONFIG_DEBUG_LCD
# define lcddbg dbg
# define lcdvdbg vdbg
#else
# define lcddbg(x...)
# define lcdvdbg(x...)
#endif
/************************************************************************************
* Private Type Definition
************************************************************************************/
/* LCD type */
enum lcd_type_e
{
LCD_TYPE_UNKNOWN = 0,
LCD_TYPE_ILI1505,
LCD_TYPE_ILI9300,
LCD_TYPE_ILI9320,
LCD_TYPE_ILI9321,
LCD_TYPE_ILI9325,
LCD_TYPE_ILI9328,
LCD_TYPE_ILI9331,
LCD_TYPE_ILI9919
};
/* This structure describes the LCD registers */
struct lcd_regs_s
{
volatile uint16_t address;
volatile uint16_t value;
};
/* This structure describes the state of this driver */
struct stm32_dev_s
{
/* Publically visible device structure */
struct lcd_dev_s dev;
/* Private LCD-specific information follows */
uint8_t type; /* LCD type. See enum lcd_type_e */
uint8_t power; /* Current power setting */
bool output; /* True: Configured for output */
};
/************************************************************************************
* Private Function Protototypes
************************************************************************************/
/* Low Level LCD access */
#ifdef CONFIG_LCD_REGDEBUG
static void stm32_lcdshow(FAR struct stm32_lower_s *priv, FAR const char *msg);
#else
# define stm32_lcdshow(p,m)
#endif
static void stm32_writereg(FAR struct stm32_dev_s *priv, uint8_t regaddr,
uint16_t regval);
static uint16_t stm32_readreg(FAR struct stm32_dev_s *priv, uint8_t regaddr);
static void stm32_gramselect(FAR struct stm32_dev_s *priv);
static void stm32_writegram(FAR struct stm32_dev_s *priv, uint16_t rgbval);
static inline uint16_t stm32_readgram(FAR struct stm32_dev_s *priv);
static void stm32_readnosetup(FAR struct stm32_dev_s *priv, FAR uint16_t *accum);
static uint16_t stm32_readnoshift(FAR struct stm32_dev_s *priv, FAR uint16_t *accum);
static void stm32_setcursor(FAR struct stm32_dev_s *priv, uint16_t col, uint16_t row);
/* LCD Data Transfer Methods */
static int stm32_putrun(fb_coord_t row, fb_coord_t col, FAR const uint8_t *buffer,
size_t npixels);
static int stm32_getrun(fb_coord_t row, fb_coord_t col, FAR uint8_t *buffer,
size_t npixels);
/* LCD Configuration */
static int stm32_getvideoinfo(FAR struct lcd_dev_s *dev,
FAR struct fb_videoinfo_s *vinfo);
static int stm32_getplaneinfo(FAR struct lcd_dev_s *dev, unsigned int planeno,
FAR struct lcd_planeinfo_s *pinfo);
/* LCD RGB Mapping */
#ifdef CONFIG_FB_CMAP
# error "RGB color mapping not supported by this driver"
#endif
/* Cursor Controls */
#ifdef CONFIG_FB_HWCURSOR
# error "Cursor control not supported by this driver"
#endif
/* LCD Specific Controls */
static int stm32_getpower(struct lcd_dev_s *dev);
static int stm32_setpower(struct lcd_dev_s *dev, int power);
static int stm32_getcontrast(struct lcd_dev_s *dev);
static int stm32_setcontrast(struct lcd_dev_s *dev, unsigned int contrast);
/* Initialization */
static void stm32_lcdinput(FAR struct stm32_dev_s *priv);
static void stm32_lcdoutput(FAR struct stm32_dev_s *priv);
#if !defined(CONFIG_STM32_ILI9300_DISABLE) || !defined(CONFIG_STM32_ILI9320_DISABLE) || !defined(CONFIG_STM32_ILI9321_DISABLE)
static void stm32_lcd9300init(FAR struct stm32_dev_s *priv, enum lcd_type_e lcdtype);
#endif
#if !defined(CONFIG_STM32_ILI9325_DISABLE) || !defined(CONFIG_STM32_ILI9328_DISABLE)
static void stm32_lcd9325init(FAR struct stm32_dev_s *priv, enum lcd_type_e lcdtype);
#endif
#ifndef CONFIG_STM32_ILI9919_DISABLE
static inline void stm32_lcd9919init(FAR struct stm32_dev_s *priv);
#endif
#ifndef CONFIG_STM32_ILI1505_DISABLE
static inline void stm32_lcd1505init(FAR struct stm32_dev_s *priv);
#endif
static inline int stm32_lcdinitialize(FAR struct stm32_dev_s *priv);
/************************************************************************************
* Private Data
************************************************************************************/
/* LCD GPIO configurations */
#ifndef CONFIG_LCD_FASTCONFIG
static const uint32_t g_lcdout[16] =
{
GPIO_LCD_D0OUT, GPIO_LCD_D1OUT, GPIO_LCD_D2OUT, GPIO_LCD_D3OUT,
GPIO_LCD_D4OUT, GPIO_LCD_D5OUT, GPIO_LCD_D6OUT, GPIO_LCD_D7OUT,
GPIO_LCD_D8OUT, GPIO_LCD_D9OUT, GPIO_LCD_D10OUT, GPIO_LCD_D11OUT,
GPIO_LCD_D12OUT, GPIO_LCD_D13OUT, GPIO_LCD_D14OUT, GPIO_LCD_D15OUT
};
static const uint32_t g_lcdin[16] =
{
GPIO_LCD_D0IN, GPIO_LCD_D1IN, GPIO_LCD_D2IN, GPIO_LCD_D3IN,
GPIO_LCD_D4IN, GPIO_LCD_D5IN, GPIO_LCD_D6IN, GPIO_LCD_D7IN,
GPIO_LCD_D8IN, GPIO_LCD_D9IN, GPIO_LCD_D10IN, GPIO_LCD_D11IN,
GPIO_LCD_D12IN, GPIO_LCD_D13IN, GPIO_LCD_D14IN, GPIO_LCD_D15IN
};
#endif
static const uint32_t g_lcdctrl[] =
{
GPIO_LCD_RS, GPIO_LCD_CS, GPIO_LCD_RD, GPIO_LCD_WR,
GPIO_LCD_LE,
};
#define NLCD_CONFIG (sizeof(g_lcdctrl)/sizeof(uint32_t))
/* This is working memory allocated by the LCD driver for each LCD device
* and for each color plane. This memory will hold one raster line of data.
* The size of the allocated run buffer must therefore be at least
* (bpp * xres / 8). Actual alignment of the buffer must conform to the
* bitwidth of the underlying pixel type.
*
* If there are multiple planes, they may share the same working buffer
* because different planes will not be operate on concurrently. However,
* if there are multiple LCD devices, they must each have unique run buffers.
*/
static uint16_t g_runbuffer[STM32_XRES];
/* This structure describes the overall LCD video controller */
static const struct fb_videoinfo_s g_videoinfo =
{
.fmt = STM32_COLORFMT, /* Color format: RGB16-565: RRRR RGGG GGGB BBBB */
.xres = STM32_XRES, /* Horizontal resolution in pixel columns */
.yres = STM32_YRES, /* Vertical resolution in pixel rows */
.nplanes = 1, /* Number of color planes supported */
};
/* This is the standard, NuttX Plane information object */
static const struct lcd_planeinfo_s g_planeinfo =
{
.putrun = stm32_putrun, /* Put a run into LCD memory */
.getrun = stm32_getrun, /* Get a run from LCD memory */
.buffer = (uint8_t*)g_runbuffer, /* Run scratch buffer */
.bpp = STM32_BPP, /* Bits-per-pixel */
};
/* This is the standard, NuttX LCD driver object */
static struct stm32_dev_s g_lcddev =
{
.dev =
{
/* LCD Configuration */
.getvideoinfo = stm32_getvideoinfo,
.getplaneinfo = stm32_getplaneinfo,
/* LCD RGB Mapping -- Not supported */
/* Cursor Controls -- Not supported */
/* LCD Specific Controls */
.getpower = stm32_getpower,
.setpower = stm32_setpower,
.getcontrast = stm32_getcontrast,
.setcontrast = stm32_setcontrast,
},
};
/************************************************************************************
* Private Functions
************************************************************************************/
/************************************************************************************
* Name: stm32_lcdshow
*
* Description:
* Show the state of the interface
*
************************************************************************************/
#ifdef CONFIG_LCD_REGDEBUG
static void stm32_lcdshow(FAR struct stm32_lower_s *priv, FAR const char *msg)
{
dbg("%s:\n", msg);
dbg(" CRTL RS: %d CS: %d RD: %d WR: %d LE: %d\n",
getreg32(LCD_RS_READ), getreg32(LCD_CS_READ), getreg32(LCD_RD_READ),
getreg32(LCD_WR_READ), getreg32(LCD_LE_READ));
dbg(" DATA CR: %08x %08x\n", getreg32(LCD_CRL), getreg32(LCD_CRH));
if (priv->output)
{
dbg(" OUTPUT: %08x\n", getreg32(LCD_ODR));
}
else
{
dbg(" INPUT: %08x\n", getreg32(LCD_IDR));
}
}
#endif
/************************************************************************************
* Name: stm32_writereg
*
* Description:
* Write to an LCD register
*
************************************************************************************/
static void stm32_writereg(FAR struct stm32_dev_s *priv, uint8_t regaddr, uint16_t regval)
{
/* Make sure that we are configured for output */
stm32_lcdoutput(priv);
/* Write the 8-bit register index */
putreg32(1, LCD_CS_CLEAR);
putreg32(1, LCD_RS_CLEAR);
putreg32(1, LCD_WR_CLEAR);
putreg32((uint32_t)regaddr, LCD_ODR);
putreg32(1, LCD_WR_SET);
/* Then write the 16-bit register value */
putreg32(1, LCD_RS_SET);
putreg32(1, LCD_WR_CLEAR);
putreg32((uint32_t)regval, LCD_ODR);
putreg32(1, LCD_WR_SET);
putreg32(1, LCD_CS_SET);
}
/************************************************************************************
* Name: stm32_readreg
*
* Description:
* Read from an LCD register
*
************************************************************************************/
static uint16_t stm32_readreg(FAR struct stm32_dev_s *priv, uint8_t regaddr)
{
uint16_t regval;
/* Make sure that we are configured for output */
stm32_lcdoutput(priv);
/* Write the 8-bit register index */
putreg32(1, LCD_CS_CLEAR);
putreg32(1, LCD_RS_CLEAR);
putreg32(1, LCD_WR_CLEAR);
putreg32((uint32_t)regaddr, LCD_ODR);
putreg32(1, LCD_WR_SET);
/* Make sure that we are configure for input */
stm32_lcdinput(priv);
/* Read the 16-bit register value */
putreg32(1, LCD_RS_SET);
putreg32(1, LCD_RD_CLEAR);
putreg32(1, LCD_RD_SET);
regval = (uint16_t)getreg32(LCD_IDR);
putreg32(1, LCD_CS_SET);
return regval;
}
/************************************************************************************
* Name: stm32_gramselect
*
* Description:
* Setup to read or write multiple pixels to the GRAM memory
*
************************************************************************************/
static void stm32_gramselect(FAR struct stm32_dev_s *priv)
{
/* Make sure that we are configured for output */
stm32_lcdoutput(priv);
/* Write the command */
putreg32(1, LCD_CS_CLEAR);
putreg32(1, LCD_RS_CLEAR);
putreg32(1, LCD_WR_CLEAR);
putreg32((uint32_t)LCD_REG_34, LCD_ODR);
putreg32(1, LCD_WR_SET);
putreg32(1, LCD_CS_SET);
}
/************************************************************************************
* Name: stm32_writegram
*
* Description:
* Write one pixel to the GRAM memory
*
************************************************************************************/
static inline void stm32_writegram(FAR struct stm32_dev_s *priv, uint16_t rgbval)
{
/* Make sure that we are configured for output */
stm32_lcdoutput(priv);
/* Write the value (GRAM register already selected) */
putreg32(1, LCD_CS_CLEAR);
putreg32(1, LCD_RS_SET);
putreg32(1, LCD_WR_CLEAR);
putreg32((uint32_t)rgbval, LCD_ODR);
putreg32(1, LCD_WR_SET);
putreg32(1, LCD_CS_SET);
}
/************************************************************************************
* Name: stm32_readgram
*
* Description:
* Read one 16-bit pixel to the GRAM memory
*
************************************************************************************/
static inline uint16_t stm32_readgram(FAR struct stm32_dev_s *priv)
{
uint16_t regval;
/* Make sure that we are configure for input */
stm32_lcdinput(priv);
/* Read the 16-bit value */
putreg32(1, LCD_CS_CLEAR);
putreg32(1, LCD_RS_SET);
putreg32(1, LCD_RD_CLEAR);
putreg32(1, LCD_RD_SET);
regval = (uint16_t)getreg32(LCD_IDR);
putreg32(1, LCD_CS_SET);
return regval;
}
/************************************************************************************
* Name: stm32_readnosetup
*
* Description:
* Prime the operation by reading one pixel from the GRAM memory if necessary for
* this LCD type. When reading 16-bit gram data, there may be some shifts in the
* returned data:
*
* - ILI932x: Discard first dummy read; no shift in the return data
*
************************************************************************************/
static void stm32_readnosetup(FAR struct stm32_dev_s *priv, FAR uint16_t *accum)
{
/* Read-ahead one pixel */
*accum = stm32_readgram(priv);
}
/************************************************************************************
* Name: stm32_readnoshift
*
* Description:
* Read one correctly aligned pixel from the GRAM memory. Possibly shifting the
* data and possibly swapping red and green components.
*
* - ILI932x: Unknown -- assuming colors are in the color order
*
************************************************************************************/
static uint16_t stm32_readnoshift(FAR struct stm32_dev_s *priv, FAR uint16_t *accum)
{
/* Read the value (GRAM register already selected) */
return stm32_readgram(priv);
}
/************************************************************************************
* Name: stm32_setcursor
*
* Description:
* Set the cursor position. In landscape mode, the "column" is actually the physical
* Y position and the "row" is the physical X position.
*
************************************************************************************/
static void stm32_setcursor(FAR struct stm32_dev_s *priv, uint16_t col, uint16_t row)
{
if (priv->type == LCD_TYPE_ILI9919)
{
stm32_writereg(priv, LCD_REG_78, col); /* GRAM horizontal address */
stm32_writereg(priv, LCD_REG_79, row); /* GRAM vertical address */
}
else
{
stm32_writereg(priv, LCD_REG_32, row); /* GRAM vertical address */
stm32_writereg(priv, LCD_REG_33, col); /* GRAM horizontal address */
}
}
/************************************************************************************
* Name: stm32_dumprun
*
* Description:
* Dump the contexts of the run buffer:
*
* run - The buffer in containing the run read to be dumped
* npixels - The number of pixels to dump
*
************************************************************************************/
#if 0 /* Sometimes useful */
static void stm32_dumprun(FAR const char *msg, FAR uint16_t *run, size_t npixels)
{
int i, j;
syslog(LOG_DEBUG, "\n%s:\n", msg);
for (i = 0; i < npixels; i += 16)
{
up_putc(' ');
syslog(LOG_DEBUG, " ");
for (j = 0; j < 16; j++)
{
syslog(LOG_DEBUG, " %04x", *run++);
}
up_putc('\n');
}
}
#endif
/************************************************************************************
* Name: stm32_putrun
*
* Description:
* This method can be used to write a partial raster line to the LCD:
*
* row - Starting row to write to (range: 0 <= row < yres)
* col - Starting column to write to (range: 0 <= col <= xres-npixels)
* buffer - The buffer containing the run to be written to the LCD
* npixels - The number of pixels to write to the LCD
* (range: 0 < npixels <= xres-col)
*
************************************************************************************/
static int stm32_putrun(fb_coord_t row, fb_coord_t col, FAR const uint8_t *buffer,
size_t npixels)
{
FAR struct stm32_dev_s *priv = &g_lcddev;
FAR const uint16_t *src = (FAR const uint16_t*)buffer;
int i;
/* Buffer must be provided and aligned to a 16-bit address boundary */
lcdvdbg("row: %d col: %d npixels: %d\n", row, col, npixels);
DEBUGASSERT(buffer && ((uintptr_t)buffer & 1) == 0);
/* Write the run to GRAM */
#ifdef CONFIG_LCD_LANDSCAPE
/* Convert coordinates */
/* Write the GRAM data, manually incrementing X */
for (i = 0; i < npixels; i++)
{
/* Write the next pixel to this position */
stm32_setcursor(priv, col, row);
stm32_gramselect(priv);
stm32_writegram(priv, *src++);
/* Increment to next column */
col++;
}
#elif defined(CONFIG_LCD_RLANDSCAPE)
/* Convert coordinates */
col = (STM32_XRES-1) - col;
row = (STM32_YRES-1) - row;
/* Set the cursor position */
stm32_setcursor(priv, col, row);
/* Then write the GRAM data, auto-decrementing X */
stm32_gramselect(priv);
for (i = 0; i < npixels; i++)
{
/* Write the next pixel to this position (auto-decrements to the next column) */
stm32_writegram(priv, *src++);
}
#elif defined(CONFIG_LCD_PORTRAIT)
/* Convert coordinates */
col = (STM32_XRES-1) - col;
/* Then write the GRAM data, manually incrementing Y (which is col) */
for (i = 0; i < npixels; i++)
{
/* Write the next pixel to this position */
stm32_setcursor(priv, row, col);
stm32_gramselect(priv);
stm32_writegram(priv, *src++);
/* Increment to next column */
col--;
}
#else /* CONFIG_LCD_RPORTRAIT */
/* Convert coordinates */
row = (STM32_YRES-1) - row;
/* Then write the GRAM data, manually incrementing Y (which is col) */
for (i = 0; i < npixels; i++)
{
/* Write the next pixel to this position */
stm32_setcursor(priv, row, col);
stm32_gramselect(priv);
stm32_writegram(priv, *src++);
/* Decrement to next column */
col++;
}
#endif
return OK;
}
/************************************************************************************
* Name: stm32_getrun
*
* Description:
* This method can be used to read a partial raster line from the LCD:
*
* row - Starting row to read from (range: 0 <= row < yres)
* col - Starting column to read read (range: 0 <= col <= xres-npixels)
* buffer - The buffer in which to return the run read from the LCD
* npixels - The number of pixels to read from the LCD
* (range: 0 < npixels <= xres-col)
*
************************************************************************************/
static int stm32_getrun(fb_coord_t row, fb_coord_t col, FAR uint8_t *buffer,
size_t npixels)
{
FAR struct stm32_dev_s *priv = &g_lcddev;
FAR uint16_t *dest = (FAR uint16_t*)buffer;
void (*readsetup)(FAR struct stm32_dev_s *priv, FAR uint16_t *accum);
uint16_t (*readgram)(FAR struct stm32_dev_s *priv, FAR uint16_t *accum);
uint16_t accum;
int i;
/* Buffer must be provided and aligned to a 16-bit address boundary */
lcdvdbg("row: %d col: %d npixels: %d\n", row, col, npixels);
DEBUGASSERT(buffer && ((uintptr_t)buffer & 1) == 0);
/* Configure according to the LCD type. Kind of silly with only one LCD type */
switch (priv->type)
{
case LCD_TYPE_ILI1505:
case LCD_TYPE_ILI9300:
case LCD_TYPE_ILI9320:
case LCD_TYPE_ILI9321:
case LCD_TYPE_ILI9325:
case LCD_TYPE_ILI9328:
case LCD_TYPE_ILI9331:
case LCD_TYPE_ILI9919:
readsetup = stm32_readnosetup;
readgram = stm32_readnoshift;
break;
case LCD_TYPE_UNKNOWN:
default: /* Shouldn't happen */
return -ENOSYS;
}
/* Read the run from GRAM */
#ifdef CONFIG_LCD_LANDSCAPE
/* Convert coordinates */
for (i = 0; i < npixels; i++)
{
/* Read the next pixel from this position */
stm32_setcursor(priv, row, col);
stm32_gramselect(priv);
stm32_lcdinput(priv);
readsetup(priv, &accum);
*dest++ = readgram(priv, &accum);
/* Increment to next column */
col++;
}
#elif defined(CONFIG_LCD_RLANDSCAPE)
/* Convert coordinates */
col = (STM32_XRES-1) - col;
row = (STM32_YRES-1) - row;
/* Set the cursor position */
stm32_setcursor(priv, col, row);
/* Then read the GRAM data, auto-decrementing Y */
stm32_gramselect(priv);
stm32_lcdinput(priv);
/* Prime the pump for unaligned read data */
readsetup(priv, &accum);
for (i = 0; i < npixels; i++)
{
/* Read the next pixel from this position (autoincrements to the next row) */
*dest++ = readgram(priv, &accum);
}
#elif defined(CONFIG_LCD_PORTRAIT)
/* Convert coordinates */
col = (STM32_XRES-1) - col;
/* Then read the GRAM data, manually incrementing Y (which is col) */
for (i = 0; i < npixels; i++)
{
/* Read the next pixel from this position */
stm32_setcursor(priv, row, col);
stm32_gramselect(priv);
stm32_lcdinput(priv);
readsetup(priv, &accum);
*dest++ = readgram(priv, &accum);
/* Increment to next column */
col--;
}
#else /* CONFIG_LCD_RPORTRAIT */
/* Convert coordinates */
row = (STM32_YRES-1) - row;
/* Then write the GRAM data, manually incrementing Y (which is col) */
for (i = 0; i < npixels; i++)
{
/* Write the next pixel to this position */
stm32_setcursor(priv, row, col);
stm32_gramselect(priv);
stm32_lcdinput(priv);
readsetup(priv, &accum);
*dest++ = readgram(priv, &accum);
/* Decrement to next column */
col++;
}
#endif
return OK;
}
/************************************************************************************
* Name: stm32_getvideoinfo
*
* Description:
* Get information about the LCD video controller configuration.
*
************************************************************************************/
static int stm32_getvideoinfo(FAR struct lcd_dev_s *dev,
FAR struct fb_videoinfo_s *vinfo)
{
DEBUGASSERT(dev && vinfo);
lcdvdbg("fmt: %d xres: %d yres: %d nplanes: %d\n",
g_videoinfo.fmt, g_videoinfo.xres, g_videoinfo.yres, g_videoinfo.nplanes);
memcpy(vinfo, &g_videoinfo, sizeof(struct fb_videoinfo_s));
return OK;
}
/************************************************************************************
* Name: stm32_getplaneinfo
*
* Description:
* Get information about the configuration of each LCD color plane.
*
************************************************************************************/
static int stm32_getplaneinfo(FAR struct lcd_dev_s *dev, unsigned int planeno,
FAR struct lcd_planeinfo_s *pinfo)
{
DEBUGASSERT(dev && pinfo && planeno == 0);
lcdvdbg("planeno: %d bpp: %d\n", planeno, g_planeinfo.bpp);
memcpy(pinfo, &g_planeinfo, sizeof(struct lcd_planeinfo_s));
return OK;
}
/************************************************************************************
* Name: stm32_getpower
*
* Description:
* Get the LCD panel power status (0: full off - CONFIG_LCD_MAXPOWER: full on). On
* backlit LCDs, this setting may correspond to the backlight setting.
*
************************************************************************************/
static int stm32_getpower(struct lcd_dev_s *dev)
{
FAR struct stm32_dev_s *priv = (FAR struct stm32_dev_s *)dev;
lcdvdbg("power: %d\n", 0);
return priv->power;
}
/************************************************************************************
* Name: stm32_poweroff
*
* Description:
* Enable/disable LCD panel power (0: full off - CONFIG_LCD_MAXPOWER: full on). On
* backlit LCDs, this setting may correspond to the backlight setting.
*
************************************************************************************/
static int stm32_poweroff(FAR struct stm32_dev_s *priv)
{
/* Turn the display off */
stm32_writereg(priv, LCD_REG_7, 0);
/* Remember the power off state */
priv->power = 0;
return OK;
}
/************************************************************************************
* Name: stm32_setpower
*
* Description:
* Enable/disable LCD panel power (0: full off - CONFIG_LCD_MAXPOWER: full on). On
* backlit LCDs, this setting may correspond to the backlight setting.
*
************************************************************************************/
static int stm32_setpower(struct lcd_dev_s *dev, int power)
{
FAR struct stm32_dev_s *priv = (FAR struct stm32_dev_s *)dev;
lcdvdbg("power: %d\n", power);
DEBUGASSERT((unsigned)power <= CONFIG_LCD_MAXPOWER);
/* Set new power level */
if (power > 0)
{
/* Then turn the display on */
#ifndef CONFIG_STM32_ILI9300_DISABLE
if (priv->type == LCD_TYPE_ILI9300)
{
stm32_writereg(priv, LCD_REG_7, 0x0173);
}
else
#endif
#ifndef CONFIG_STM32_ILI9320_DISABLE
if (priv->type == LCD_TYPE_ILI9320)
{
stm32_writereg(priv, LCD_REG_7, 0x0173);
}
else
#endif
#ifndef CONFIG_STM32_ILI9321_DISABLE
if (priv->type == LCD_TYPE_ILI9321)
{
stm32_writereg(priv, LCD_REG_7, 0x0173);
}
else
#endif
#ifndef CONFIG_STM32_ILI9325_DISABLE
if (priv->type == LCD_TYPE_ILI9325)
{
stm32_writereg(priv, LCD_REG_7, 0x0133);
}
else
#endif
#ifndef CONFIG_STM32_ILI9328_DISABLE
if (priv->type == LCD_TYPE_ILI9328)
{
stm32_writereg(priv, LCD_REG_7, 0x0133);
}
else
#endif
#ifndef CONFIG_STM32_ILI9331_DISABLE
if (priv->type == LCD_TYPE_ILI9331)
{
stm32_writereg(priv, LCD_REG_7, 0x0021);
up_mdelay(50);
stm32_writereg(priv, LCD_REG_7, 0x0061);
up_mdelay(50);
stm32_writereg(priv, LCD_REG_7, 0x0133); /* 262K color and display ON */
}
else
#endif
#ifndef CONFIG_STM32_ILI9919_DISABLE
if (priv->type == LCD_TYPE_ILI9919)
{
stm32_writereg(priv, LCD_REG_7, 0x0033);
}
else
#endif
#ifndef CONFIG_STM32_ILI1505_DISABLE
if (priv->type == LCD_TYPE_ILI1505)
{
stm32_writereg(priv, LCD_REG_7, 0x0021);
up_mdelay(20);
stm32_writereg(priv, LCD_REG_7, 0x0061);
up_mdelay(20);
stm32_writereg(priv, LCD_REG_7, 0x0173);
}
else
#endif
{
gdbg("Unsupported LCD: %d\n", priv->type);
}
up_mdelay(50);
priv->power = power;
}
else
{
/* Turn the display off */
stm32_poweroff(priv);
}
return OK;
}
/************************************************************************************
* Name: stm32_getcontrast
*
* Description:
* Get the current contrast setting (0-CONFIG_LCD_MAXCONTRAST).
*
************************************************************************************/
static int stm32_getcontrast(struct lcd_dev_s *dev)
{
lcdvdbg("Not implemented\n");
return -ENOSYS;
}
/************************************************************************************
* Name: stm32_setcontrast
*
* Description:
* Set LCD panel contrast (0-CONFIG_LCD_MAXCONTRAST).
*
************************************************************************************/
static int stm32_setcontrast(struct lcd_dev_s *dev, unsigned int contrast)
{
lcdvdbg("contrast: %d\n", contrast);
return -ENOSYS;
}
/************************************************************************************
* Name: stm32_lcdinput
*
* Description:
* Config data lines for input operations.
*
************************************************************************************/
static void stm32_lcdinput(FAR struct stm32_dev_s *priv)
{
#ifndef CONFIG_LCD_FASTCONFIG
int i;
#endif
/* Check if we are already configured for input */
if (priv->output)
{
/* Configure GPIO data lines as inputs */
#ifdef CONFIG_LCD_FASTCONFIG
putreg32(LCD_INPUT, LCD_CRL);
putreg32(LCD_INPUT, LCD_CRH);
#else
for (i = 0; i < 16; i++)
{
stm32_configgpio(g_lcdin[i]);
}
#endif
/* No longer configured for output */
priv->output = false;
}
}
/************************************************************************************
* Name: stm32_lcdoutput
*
* Description:
* Config data lines for input operations.
*
************************************************************************************/
static void stm32_lcdoutput(FAR struct stm32_dev_s *priv)
{
#ifndef CONFIG_LCD_FASTCONFIG
int i;
#endif
/* Check if we are already configured for output */
if (!priv->output)
{
/* Configure GPIO data lines as outputs */
#ifdef CONFIG_LCD_FASTCONFIG
putreg32(LCD_OUTPUT, LCD_CRL);
putreg32(LCD_OUTPUT, LCD_CRH);
#else
for (i = 0; i < 16; i++)
{
stm32_configgpio(g_lcdout[i]);
}
#endif
/* Now we are configured for output */
priv->output = true;
}
}
/************************************************************************************
* Name: stm32_lcd9300init
*
* Description:
* Initialize the ILI9300/9220/9321 LCD.
*
************************************************************************************/
#if !defined(CONFIG_STM32_ILI9300_DISABLE) || !defined(CONFIG_STM32_ILI9320_DISABLE) || !defined(CONFIG_STM32_ILI9321_DISABLE)
static void stm32_lcd9300init(FAR struct stm32_dev_s *priv, enum lcd_type_e lcdtype)
{
stm32_writereg(priv, LCD_REG_0, 0x0001); /* Start internal OSC */
stm32_writereg(priv, LCD_REG_1, 0x0100); /* Driver Output Control */
stm32_writereg(priv, LCD_REG_2, 0x0700); /* LCD Driver Waveform Control */
stm32_writereg(priv, LCD_REG_3, 0x1018); /* Set GRAM write direction and BGR=1 (0x1030)*/
stm32_writereg(priv, LCD_REG_4, 0x0000); /* Scalling Control */
stm32_writereg(priv, LCD_REG_8, 0x0202); /* Set the back porch and front porch (0x0207) */
stm32_writereg(priv, LCD_REG_9, 0x0000); /* Set non-display area refresh cycle ISC[3:0] */
stm32_writereg(priv, LCD_REG_10, 0x0000); /* Frame Cycle Control */
stm32_writereg(priv, LCD_REG_12, (1<<0)); /* RGB interface setting (0x0000) */
stm32_writereg(priv, LCD_REG_13, 0x0000); /* Frame Maker Position */
stm32_writereg(priv, LCD_REG_15, 0x0000); /* RGB interface polarity */
up_mdelay(50);
stm32_writereg(priv, LCD_REG_7, 0x0101); /* Display Control */
up_mdelay(50);
/* Power On sequence */
stm32_writereg(priv, LCD_REG_16, (1<<12)|(0<<8)|(1<<7)|(1<<6)|(0<<4)); /* Power Control 1 (0x16b0) */
stm32_writereg(priv, LCD_REG_17, 0x0007); /* Power Control 2 (0x0001) */
stm32_writereg(priv, LCD_REG_18, (1<<8)|(1<<4)|(0<<0)); /* Power Control 3 (0x0138) */
stm32_writereg(priv, LCD_REG_19, 0x0b00); /* VDV[4:0] for VCOM amplitude */
stm32_writereg(priv, LCD_REG_41, 0x0000); /* VCM[4:0] for VCOMH */
stm32_writereg(priv, LCD_REG_43, (1<<14)|(1<<4));
stm32_writereg(priv, LCD_REG_80, 0); /* Set X Start */
stm32_writereg(priv, LCD_REG_81, 239); /* Set X End */
stm32_writereg(priv, LCD_REG_82, 0); /* Set Y Start */
stm32_writereg(priv, LCD_REG_83, 319); /* Set Y End */
stm32_writereg(priv, LCD_REG_96, 0x2700); /* Driver Output Control */
stm32_writereg(priv, LCD_REG_97, 0x0001); /* Driver Output Control */
stm32_writereg(priv, LCD_REG_106, 0x0000); /* Vertical Srcoll Control */
stm32_writereg(priv, LCD_REG_128, 0x0000); /* Display Position? Partial Display 1 */
stm32_writereg(priv, LCD_REG_129, 0x0000); /* RAM Address Start? Partial Display 1 */
stm32_writereg(priv, LCD_REG_130, 0x0000); /* RAM Address End-Partial Display 1 */
stm32_writereg(priv, LCD_REG_131, 0x0000); /* Display Position? Partial Display 2 */
stm32_writereg(priv, LCD_REG_132, 0x0000); /* RAM Address Start? Partial Display 2 */
stm32_writereg(priv, LCD_REG_133, 0x0000); /* RAM Address End? Partial Display 2 */
stm32_writereg(priv, LCD_REG_144, (0<<7)|(16<<0)); /* Frame Cycle Control (0x0013) */
stm32_writereg(priv, LCD_REG_146, 0x0000); /* Panel Interface Control 2 */
stm32_writereg(priv, LCD_REG_147, 0x0001); /* Panel Interface Control 3 */
stm32_writereg(priv, LCD_REG_149, 0x0110); /* Frame Cycle Control */
stm32_writereg(priv, LCD_REG_151, (0<<8));
stm32_writereg(priv, LCD_REG_152, 0x0000); /* Frame Cycle Control */
up_mdelay(50);
stm32_writereg(priv, LCD_REG_7, 0x0000); /* Display off */
}
#endif
/************************************************************************************
* Name: stm32_lcd9331init
*
* Description:
* Initialize the ILI9331 LCD.
*
************************************************************************************/
#ifndef CONFIG_STM32_ILI9331_DISABLE
static void stm32_lcd9331init(FAR struct stm32_dev_s *priv)
{
stm32_writereg(priv, LCD_REG_231, 0x1014);
stm32_writereg(priv, LCD_REG_1, 0x0100); /* Set SS and SM bit */
stm32_writereg(priv, LCD_REG_2, 0x0200); /* Set 1 line inversion */
stm32_writereg(priv, LCD_REG_3, 0x1030); /* Set GRAM write direction and BGR=1 */
stm32_writereg(priv, LCD_REG_8, 0x0202); /* Set the back porch and front porch */
stm32_writereg(priv, LCD_REG_9, 0x0000); /* Set non-display area refresh cycle ISC[3:0] */
stm32_writereg(priv, LCD_REG_10, 0x0000); /* FMARK function */
stm32_writereg(priv, LCD_REG_12, 0x0000); /* RGB interface setting */
stm32_writereg(priv, LCD_REG_13, 0x0000); /* Frame marker Position */
stm32_writereg(priv, LCD_REG_15, 0x0000); /* RGB interface polarity */
/* Power On sequence */
stm32_writereg(priv, LCD_REG_16, 0x0000); /* SAP, BT[3:0], AP, DSTB, SLP, STB */
stm32_writereg(priv, LCD_REG_17, 0x0007); /* DC1[2:0], DC0[2:0], VC[2:0] */
stm32_writereg(priv, LCD_REG_18, 0x0000); /* VREG1OUT voltage */
stm32_writereg(priv, LCD_REG_19, 0x0000); /* VDV[4:0] for VCOM amplitude */
up_mdelay(200); /* Dis-charge capacitor power voltage */
stm32_writereg(priv, LCD_REG_16, 0x1690); /* SAP, BT[3:0], AP, DSTB, SLP, STB */
stm32_writereg(priv, LCD_REG_17, 0x0227); /* DC1[2:0], DC0[2:0], VC[2:0] */
up_mdelay(50);
stm32_writereg(priv, LCD_REG_18, 0x000c); /* Internal reference voltage= Vci; */
up_mdelay(50);
stm32_writereg(priv, LCD_REG_19, 0x0800); /* Set VDV[4:0] for VCOM amplitude */
stm32_writereg(priv, LCD_REG_41, 0x0011); /* Set VCM[5:0] for VCOMH */
stm32_writereg(priv, LCD_REG_43, 0x000b); /* Set Frame Rate */
up_mdelay(50);
stm32_writereg(priv, LCD_REG_32, 0x0000); /* GRAM horizontal Address */
stm32_writereg(priv, LCD_REG_33, 0x0000); /* GRAM Vertical Address */
/* Adjust the Gamma Curve */
stm32_writereg(priv, LCD_REG_48, 0x0000);
stm32_writereg(priv, LCD_REG_49, 0x0106);
stm32_writereg(priv, LCD_REG_50, 0x0000);
stm32_writereg(priv, LCD_REG_53, 0x0204);
stm32_writereg(priv, LCD_REG_54, 0x160a);
stm32_writereg(priv, LCD_REG_55, 0x0707);
stm32_writereg(priv, LCD_REG_56, 0x0106);
stm32_writereg(priv, LCD_REG_57, 0x0707);
stm32_writereg(priv, LCD_REG_60, 0x0402);
stm32_writereg(priv, LCD_REG_61, 0x0c0f);
/* Set GRAM area */
stm32_writereg(priv, LCD_REG_80, 0x0000); /* Horizontal GRAM Start Address */
stm32_writereg(priv, LCD_REG_81, 0x00ef); /* Horizontal GRAM End Address */
stm32_writereg(priv, LCD_REG_82, 0x0000); /* Vertical GRAM Start Address */
stm32_writereg(priv, LCD_REG_83, 0x013f); /* Vertical GRAM Start Address */
stm32_writereg(priv, LCD_REG_96, 0x2700); /* Gate Scan Line */
stm32_writereg(priv, LCD_REG_97, 0x0001); /* NDL,VLE, REV */
stm32_writereg(priv, LCD_REG_106, 0x0000); /* set scrolling line */
/* Partial Display Control */
stm32_writereg(priv, LCD_REG_128, 0x0000);
stm32_writereg(priv, LCD_REG_129, 0x0000);
stm32_writereg(priv, LCD_REG_130, 0x0000);
stm32_writereg(priv, LCD_REG_131, 0x0000);
stm32_writereg(priv, LCD_REG_132, 0x0000);
stm32_writereg(priv, LCD_REG_133, 0x0000);
/* Panel Control */
stm32_writereg(priv, LCD_REG_144, 0x0010);
stm32_writereg(priv, LCD_REG_146, 0x0600);
stm32_writereg(priv, LCD_REG_7, 0x0000); /* Display off */
}
#endif
/************************************************************************************
* Name: stm32_lcd9325init
*
* Description:
* Initialize the ILI9325/9228 LCD.
*
************************************************************************************/
#if !defined(CONFIG_STM32_ILI9325_DISABLE) || !defined(CONFIG_STM32_ILI9328_DISABLE)
static void stm32_lcd9325init(FAR struct stm32_dev_s *priv, enum lcd_type_e lcdtype)
{
stm32_writereg(priv, LCD_REG_227, 0x3008);
stm32_writereg(priv, LCD_REG_231, 0x0012);
stm32_writereg(priv, LCD_REG_239, 0x1231); /* Set the internal vcore voltage */
/*stm32_writereg(priv, LCD_REG_231, 0x0010); */
stm32_writereg(priv, LCD_REG_0, 0x0001); /* Start internal osc */
stm32_writereg(priv, LCD_REG_1, 0x0100); /* Set SS and SM bit */
stm32_writereg(priv, LCD_REG_2, 0x0700); /* Power on sequence */
stm32_writereg(priv, LCD_REG_3, (1<<12)|(1<<5)|(1<<4) ); /* 65K */
stm32_writereg(priv, LCD_REG_4, 0x0000); /* Resize register */
stm32_writereg(priv, LCD_REG_8, 0x0207); /* Set the back porch and front porch */
stm32_writereg(priv, LCD_REG_9, 0x0000); /* Set non-display area refresh cycle ISC[3:0] */
stm32_writereg(priv, LCD_REG_10, 0x0000); /* FMARK function */
stm32_writereg(priv, LCD_REG_12, 0x0001); /* RGB interface setting */
stm32_writereg(priv, LCD_REG_13, 0x0000); /* Frame marker Position (0x0f3c) */
stm32_writereg(priv, LCD_REG_15, 0x0000); /* RGB interface polarity */
/* Power On sequence */
stm32_writereg(priv, LCD_REG_16, 0x0000); /* SAP, BT[3:0], AP, DSTB, SLP, STB */
stm32_writereg(priv, LCD_REG_17, 0x0007); /* DC1[2:0], DC0[2:0], VC[2:0] */
stm32_writereg(priv, LCD_REG_18, 0x0000); /* VREG1OUT voltage */
stm32_writereg(priv, LCD_REG_19, 0x0000); /* VDV[4:0] for VCOM amplitude */
up_mdelay(100);
stm32_writereg(priv, LCD_REG_16, 0x1590); /* SAP, BT[3:0], AP, DSTB, SLP, STB */
stm32_writereg(priv, LCD_REG_17, 0x0227); /* DC1[2:0], DC0[2:0], VC[2:0] */
up_mdelay(100);
stm32_writereg(priv, LCD_REG_18, 0x009c); /* VREG1OUT voltage */
up_mdelay(100);
stm32_writereg(priv, LCD_REG_19, 0x1900); /* VDV[4:0] for VCOM amplitude */
stm32_writereg(priv, LCD_REG_41, 0x0023); /* VCM[4:0] for VCOMH */
stm32_writereg(priv, LCD_REG_43, 0x000e);
up_mdelay(100);
stm32_writereg(priv, LCD_REG_32, 0x0000); /* GRAM horizontal Address */
stm32_writereg(priv, LCD_REG_33, 0x0000); /* GRAM Vertical Address */
up_mdelay(100);
/* Adjust the Gamma Curve */
stm32_writereg(priv, LCD_REG_48, 0x0007);
stm32_writereg(priv, LCD_REG_49, 0x0707);
stm32_writereg(priv, LCD_REG_50, 0x0006);
stm32_writereg(priv, LCD_REG_53, 0x0704);
stm32_writereg(priv, LCD_REG_54, 0x1f04);
stm32_writereg(priv, LCD_REG_55, 0x0004);
stm32_writereg(priv, LCD_REG_56, 0x0000);
stm32_writereg(priv, LCD_REG_57, 0x0706);
stm32_writereg(priv, LCD_REG_60, 0x0701);
stm32_writereg(priv, LCD_REG_61, 0x000f);
up_mdelay(100);
/* Set GRAM area */
stm32_writereg(priv, LCD_REG_80, 0x0000); /* Horizontal GRAM Start Address */
stm32_writereg(priv, LCD_REG_81, 0x00ef); /* Horizontal GRAM End Address */
stm32_writereg(priv, LCD_REG_82, 0x0000); /* Vertical GRAM Start Address */
stm32_writereg(priv, LCD_REG_83, 0x013f); /* Vertical GRAM End Address */
stm32_writereg(priv, LCD_REG_96, 0xa700); /* Gate Scan Line */
stm32_writereg(priv, LCD_REG_97, 0x0001); /* NDL, VLE, REV */
stm32_writereg(priv, LCD_REG_106, 0x0000); /* Set scrolling line */
/* Partial Display Control */
stm32_writereg(priv, LCD_REG_128, 0x0000);
stm32_writereg(priv, LCD_REG_129, 0x0000);
stm32_writereg(priv, LCD_REG_130, 0x0000);
stm32_writereg(priv, LCD_REG_131, 0x0000);
stm32_writereg(priv, LCD_REG_132, 0x0000);
stm32_writereg(priv, LCD_REG_133, 0x0000);
/* Panel Control */
stm32_writereg(priv, LCD_REG_144, 0x0010);
stm32_writereg(priv, LCD_REG_146, 0x0600);
if (lcdtype == LCD_TYPE_ILI9328)
{
stm32_writereg(priv, LCD_REG_147, 0x0003);
stm32_writereg(priv, LCD_REG_149, 0x0110);
stm32_writereg(priv, LCD_REG_151, 0x0000);
stm32_writereg(priv, LCD_REG_152, 0x0000);
}
stm32_writereg(priv, LCD_REG_7, 0x0000); /* Display off */
stm32_writereg(priv, LCD_REG_32, 0x0000); /* GRAM horizontal Address */
stm32_writereg(priv, LCD_REG_33, 0x0000); /* GRAM Vertical Address */
}
#endif
/************************************************************************************
* Name: stm32_lcd9919init
*
* Description:
* Initialize the ILI9919 LCD.
*
************************************************************************************/
#ifndef CONFIG_STM32_ILI9919_DISABLE
static inline void stm32_lcd9919init(FAR struct stm32_dev_s *priv)
{
/* Power on reset, display off */
stm32_writereg(priv, LCD_REG_40, 0x0006);
stm32_writereg(priv, LCD_REG_0, 0x0001); /* Start internal OSC */
stm32_writereg(priv, LCD_REG_16, 0x0000); /* SAP, BT[3:0], AP, DSTB, SLP, STB */
stm32_writereg(priv, LCD_REG_1, 0x72ef);
stm32_writereg(priv, LCD_REG_2, 0x0600); /* Set 1 line inversion */
stm32_writereg(priv, LCD_REG_3, 0x6a38);
stm32_writereg(priv, LCD_REG_17, 0x6874); /* DC1[2:0], DC0[2:0], VC[2:0] (0x0070) */
/* RAM write data mask */
stm32_writereg(priv, LCD_REG_15, 0x0000); /* RGB interface polarity */
stm32_writereg(priv, LCD_REG_11, 0x5308);
stm32_writereg(priv, LCD_REG_12, 0x0003); /* RGB interface setting */
stm32_writereg(priv, LCD_REG_13, 0x000a); /* Frame marker Position */
stm32_writereg(priv, LCD_REG_14, 0x2e00); /* 0x0030 */
stm32_writereg(priv, LCD_REG_30, 0x00be);
stm32_writereg(priv, LCD_REG_37, 0x8000);
stm32_writereg(priv, LCD_REG_38, 0x7800);
stm32_writereg(priv, LCD_REG_39, 0x0078);
stm32_writereg(priv, LCD_REG_78, 0x0000);
stm32_writereg(priv, LCD_REG_79, 0x0000);
stm32_writereg(priv, LCD_REG_18, 0x08d9); /* VREG1OUT voltage */
/* Adjust the Gamma Curve */
stm32_writereg(priv, LCD_REG_48, 0x0000); /* 0x0007 */
stm32_writereg(priv, LCD_REG_49, 0x0104); /* 0x0203 */
stm32_writereg(priv, LCD_REG_50, 0x0100); /* 0x0001 */
stm32_writereg(priv, LCD_REG_51, 0x0305); /* 0x0007 */
stm32_writereg(priv, LCD_REG_52, 0x0505); /* 0x0007 */
stm32_writereg(priv, LCD_REG_53, 0x0305); /* 0x0407 */
stm32_writereg(priv, LCD_REG_54, 0x0707); /* 0x0407 */
stm32_writereg(priv, LCD_REG_55, 0x0300); /* 0x0607 */
stm32_writereg(priv, LCD_REG_58, 0x1200); /* 0x0106 */
stm32_writereg(priv, LCD_REG_59, 0x0800);
stm32_writereg(priv, LCD_REG_7, 0x0000); /* Display off */
}
#endif
/************************************************************************************
* Name: stm32_lcd1505init
*
* Description:
* Initialize the ILI1505 LCD.
*
************************************************************************************/
#ifndef CONFIG_STM32_ILI1505_DISABLE
static inline void stm32_lcd1505init(FAR struct stm32_dev_s *priv)
{
stm32_writereg(priv, LCD_REG_7, 0x0000);
up_mdelay(5);
stm32_writereg(priv, LCD_REG_18, 0x011c);
stm32_writereg(priv, LCD_REG_164, 0x0001); /* NVM */
stm32_writereg(priv, LCD_REG_8, 0x000f);
stm32_writereg(priv, LCD_REG_10, 0x0008);
stm32_writereg(priv, LCD_REG_13, 0x0008);
/* Adjust the Gamma Curve */
stm32_writereg(priv, LCD_REG_48, 0x0707);
stm32_writereg(priv, LCD_REG_49, 0x0007); /* 0x0707 */
stm32_writereg(priv, LCD_REG_50, 0x0603);
stm32_writereg(priv, LCD_REG_51, 0x0700);
stm32_writereg(priv, LCD_REG_52, 0x0202);
stm32_writereg(priv, LCD_REG_53, 0x0002); /* 0x0606 */
stm32_writereg(priv, LCD_REG_54, 0x1f0f);
stm32_writereg(priv, LCD_REG_55, 0x0707); /* 0x0f0f, 0x0105 */
stm32_writereg(priv, LCD_REG_56, 0x0000);
stm32_writereg(priv, LCD_REG_57, 0x0000);
stm32_writereg(priv, LCD_REG_58, 0x0707);
stm32_writereg(priv, LCD_REG_59, 0x0000); /* 0x0303 */
stm32_writereg(priv, LCD_REG_60, 0x0007); /* 0x0707 */
stm32_writereg(priv, LCD_REG_61, 0x0000); /* 0x1313, 0x1f08 */
up_mdelay(5);
stm32_writereg(priv, LCD_REG_7, 0x0001);
stm32_writereg(priv, LCD_REG_23, 0x0001); /* Power supply startup enable */
up_mdelay(5);
/* Power Control */
stm32_writereg(priv, LCD_REG_16, 0x17a0); /* SAP, BT[3:0], AP, DSTB, SLP, STB */
stm32_writereg(priv, LCD_REG_17, 0x0217); /* Reference voltage VC[2:0] Vciout = 1.00*Vcivl */
stm32_writereg(priv, LCD_REG_18, 0x011e); /* Vreg1out = Vcilvl*1.80 (0x011c) */
stm32_writereg(priv, LCD_REG_19, 0x0f00); /* VDV[4:0]-->VCOM Amplitude VcomL = VcomH - Vcom Ampl */
stm32_writereg(priv, LCD_REG_42, 0x0000);
stm32_writereg(priv, LCD_REG_41, 0x000a); /* Vcomh = VCM1[4:0]*Vreg1out gate source voltage (0x001f) */
stm32_writereg(priv, LCD_REG_18, 0x013e); /* Power supply on (0x013c) */
/* Coordinates Control */
stm32_writereg(priv, LCD_REG_80, 0x0000); /* Horizontal GRAM Start Address */
stm32_writereg(priv, LCD_REG_81, 0x00ef); /* Horizontal GRAM End Address */
stm32_writereg(priv, LCD_REG_82, 0x0000); /* Vertical GRAM Start Address */
stm32_writereg(priv, LCD_REG_83, 0x013f); /* Vertical GRAM End Address */
/* Panel Image Control */
stm32_writereg(priv, LCD_REG_96, 0x2700); /* Gate Scan Line */
stm32_writereg(priv, LCD_REG_97, 0x0001); /* NDL, VLE, REV */
stm32_writereg(priv, LCD_REG_106, 0x0000); /* Set scrolling line */
/* Partial Image Control */
stm32_writereg(priv, LCD_REG_128, 0x0000);
stm32_writereg(priv, LCD_REG_129, 0x0000);
stm32_writereg(priv, LCD_REG_130, 0x0000);
stm32_writereg(priv, LCD_REG_131, 0x0000);
stm32_writereg(priv, LCD_REG_132, 0x0000);
stm32_writereg(priv, LCD_REG_133, 0x0000);
/* Panel Interface Control */
stm32_writereg(priv, LCD_REG_144, 0x0013);
stm32_writereg(priv, LCD_REG_146, 0x0300);
stm32_writereg(priv, LCD_REG_147, 0x0005);
stm32_writereg(priv, LCD_REG_149, 0x0000);
stm32_writereg(priv, LCD_REG_151, 0x0000);
stm32_writereg(priv, LCD_REG_152, 0x0000);
stm32_writereg(priv, LCD_REG_1, 0x0100); /* Set SS and SM bit */
stm32_writereg(priv, LCD_REG_2, 0x0700); /* Set 1 line inversion */
stm32_writereg(priv, LCD_REG_3, 0x1030); /* Set GRAM write direction and BGR=1 */
stm32_writereg(priv, LCD_REG_4, 0x0000); /* Resize register */
stm32_writereg(priv, LCD_REG_12, 0x0000); /* RGB interface setting */
stm32_writereg(priv, LCD_REG_15, 0x0000); /* RGB interface polarity */
stm32_writereg(priv, LCD_REG_32, 0x0000); /* GRAM horizontal Address */
stm32_writereg(priv, LCD_REG_33, 0x0000); /* GRAM Vertical Address */
stm32_writereg(priv, LCD_REG_7, 0x0000); /* Display off */
}
#endif
/************************************************************************************
* Name: stm32_lcdinitialize
*
* Description:
* Set LCD panel contrast (0-CONFIG_LCD_MAXCONTRAST).
*
************************************************************************************/
static inline int stm32_lcdinitialize(FAR struct stm32_dev_s *priv)
{
uint16_t id;
int ret = OK;
/* Check LCD ID */
stm32_writereg(priv, LCD_REG_0, 0x0001); /* Start internal oscillator */
up_mdelay(50);
id = stm32_readreg(priv, LCD_REG_0); /* Read the ID register */
lcddbg("LCD ID: %04x\n", id);
stm32_lcdoutput(priv);
up_mdelay(10);
/* Initialize the LCD hardware */
#ifndef CONFIG_STM32_ILI9300_DISABLE
if (id == ILI9300_ID)
{
priv->type = LCD_TYPE_ILI9300;
stm32_lcd9300init(priv, LCD_TYPE_ILI9325);
}
else
#endif
#ifndef CONFIG_STM32_ILI9320_DISABLE
if (id == ILI9320_ID)
{
priv->type = LCD_TYPE_ILI9320;
stm32_lcd9300init(priv, LCD_TYPE_ILI9320);
}
else
#endif
#ifndef CONFIG_STM32_ILI9321_DISABLE
if (id == ILI9321_ID)
{
priv->type = LCD_TYPE_ILI9321;
stm32_lcd9300init(priv, LCD_TYPE_ILI9321);
}
else
#endif
#ifndef CONFIG_STM32_ILI9331_DISABLE
if (id == ILI9331_ID)
{
priv->type = LCD_TYPE_ILI9331;
stm32_lcd9331init(priv);
}
else
#endif
#ifndef CONFIG_STM32_ILI9325_DISABLE
if (id == ILI9325_ID)
{
priv->type = LCD_TYPE_ILI9325;
stm32_lcd9325init(priv, LCD_TYPE_ILI9325);
}
else
#endif
#ifndef CONFIG_STM32_ILI9328_DISABLE
if (id == ILI9328_ID)
{
priv->type = LCD_TYPE_ILI9328;
stm32_lcd9325init(priv, LCD_TYPE_ILI9328);
}
else
#endif
#ifndef CONFIG_STM32_ILI1505_DISABLE
if (id == 0x1505)
{
priv->type = LCD_TYPE_ILI1505;
stm32_lcd1505init(priv);
}
else
#endif
#ifndef CONFIG_STM32_ILI9919_DISABLE
if (id == 0x9919)
{
priv->type = LCD_TYPE_ILI9919;
stm32_lcd9919init(priv);
}
else
#endif
{
lcddbg("Unsupported LCD type\n");
ret = -ENODEV;
}
lcddbg("LCD type: %d\n", priv->type);
return ret;
}
/************************************************************************************
* Public Functions
************************************************************************************/
/************************************************************************************
* Name: board_lcd_initialize
*
* Description:
* Initialize the LCD video hardware. The initial state of the LCD is fully
* initialized, display memory cleared, and the LCD ready to use, but with the power
* setting at 0 (full off).
*
************************************************************************************/
int board_lcd_initialize(void)
{
FAR struct stm32_dev_s *priv = &g_lcddev;
int ret;
int i;
lcdvdbg("Initializing\n");
/* Configure GPIO pins. The initial state of priv->output is false, so
* we need to configure pins for output initially.
*/
stm32_lcdoutput(priv);
/* Configure control pins */
for (i = 0; i < NLCD_CONFIG; i++)
{
stm32_configgpio(g_lcdctrl[i]);
}
/* Configure and enable LCD */
up_mdelay(50);
ret = stm32_lcdinitialize(priv);
if (ret == OK)
{
/* Clear the display (setting it to the color 0=black) */
stm32_lcdclear(0);
/* Turn the display off */
stm32_poweroff(priv);
}
return ret;
}
/************************************************************************************
* Name: board_lcd_getdev
*
* Description:
* Return a a reference to the LCD object for the specified LCD. This allows support
* for multiple LCD devices.
*
************************************************************************************/
FAR struct lcd_dev_s *board_lcd_getdev(int lcddev)
{
DEBUGASSERT(lcddev == 0);
return &g_lcddev.dev;
}
/************************************************************************************
* Name: board_lcd_uninitialize
*
* Description:
* Unitialize the LCD support
*
************************************************************************************/
void board_lcd_uninitialize(void)
{
FAR struct stm32_dev_s *priv = &g_lcddev;
/* Put the LCD in the lowest possible power state */
stm32_poweroff(priv);
/* Make sure that the LCD is not selected */
putreg32(1, LCD_CS_SET);
}
/************************************************************************************
* Name: stm32_lcdclear
*
* Description:
* This is a non-standard LCD interface just for the Shenzhou board. Because
* of the various rotations, clearing the display in the normal way by writing a
* sequences of runs that covers the entire display can be very slow. Here the
* display is cleared by simply setting all GRAM memory to the specified color.
*
************************************************************************************/
void stm32_lcdclear(uint16_t color)
{
FAR struct stm32_dev_s *priv = &g_lcddev;
uint32_t i = 0;
stm32_setcursor(priv, 0, 0);
stm32_gramselect(priv);
/* Make sure that we are configured for output */
stm32_lcdoutput(priv);
/* Write the selected color into the entire GRAM memory */
putreg32(1, LCD_CS_CLEAR);
putreg32(1, LCD_RS_SET);
for (i = 0; i < STM32_XRES * STM32_YRES; i++)
{
putreg32(1, LCD_WR_CLEAR);
putreg32((uint32_t)color, LCD_ODR);
putreg32(1, LCD_WR_SET);
}
putreg32(1, LCD_CS_SET);
}
#endif /* !HAVE_LCD */