2013-05-01 23:53:28 +00:00
|
|
|
/*
|
2013-06-15 11:37:22 +00:00
|
|
|
* This file is subject to the terms of the GFX License. If a copy of
|
2013-06-02 08:58:12 +00:00
|
|
|
* the license was not distributed with this file, you can obtain one at:
|
|
|
|
*
|
|
|
|
* http://chibios-gfx.com/license.html
|
|
|
|
*/
|
2013-02-18 07:31:49 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @file drivers/tdisp/HD44780/tdisp_lld.c
|
|
|
|
* @brief TDISP driver subsystem low level driver source for the HD44780 display
|
|
|
|
*
|
|
|
|
* @addtogroup TDISP
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "gfx.h"
|
|
|
|
|
2013-06-02 08:58:12 +00:00
|
|
|
#if GFX_USE_TDISP /*|| defined(__DOXYGEN__)*/
|
2013-02-18 07:31:49 +00:00
|
|
|
|
2013-04-12 07:12:00 +00:00
|
|
|
/* check first if the user has defined his/her own lowlevel-board file */
|
|
|
|
#if defined(TDISP_USE_CUSTOM_BOARD) && TDISP_USE_CUSTOM_BOARD
|
|
|
|
/* Include the user supplied board definitions */
|
|
|
|
#include "tdisp_lld_board.h"
|
|
|
|
#elif defined(BOARD_OLIMEX_STM32_E407)
|
2013-03-02 21:13:44 +00:00
|
|
|
#include "tdisp_lld_board_olimex_e407.h"
|
|
|
|
#elif defined(BOARD_ST_STM32F4_DISCOVERY)
|
|
|
|
#include "tdisp_lld_board_example.h"
|
2013-02-18 07:31:49 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Controller Specific Properties */
|
|
|
|
#define CUSTOM_CHAR_COUNT 8
|
|
|
|
#define CUSTOM_CHAR_XBITS 5
|
|
|
|
#define CUSTOM_CHAR_YBITS 8
|
|
|
|
|
|
|
|
/* Define the properties of our controller */
|
|
|
|
tdispStruct TDISP = {
|
2013-04-12 07:12:00 +00:00
|
|
|
TDISP_COLUMNS, TDISP_ROWS, /* cols, rows */
|
2013-03-02 21:13:44 +00:00
|
|
|
CUSTOM_CHAR_XBITS, CUSTOM_CHAR_YBITS, /* charBitsX, charBitsY */
|
2013-04-12 07:12:00 +00:00
|
|
|
CUSTOM_CHAR_COUNT /* maxCustomChars */
|
2013-03-02 21:13:44 +00:00
|
|
|
};
|
2013-02-18 07:31:49 +00:00
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* display control settings */
|
2013-04-12 07:12:00 +00:00
|
|
|
#define TDISP_DISPLAY_ON 0x04
|
|
|
|
#define TDISP_CURSOR_ON 0x02
|
|
|
|
#define TDISP_CURSOR_BLINK 0x01
|
2013-02-18 07:31:49 +00:00
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* cursor movement settings */
|
|
|
|
#define TDISP_SHIFT_ON 0x01
|
|
|
|
#define TDISP_CURSOR_INC 0x02
|
2013-02-18 07:31:49 +00:00
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* display scroll settings */
|
|
|
|
#define TDISP_SCROLL_DISPLAY 0x08
|
|
|
|
#define TDISP_SCROLL_RIGHT 0x04
|
|
|
|
|
|
|
|
static uint8_t displaycontrol;
|
|
|
|
static uint8_t cursorcontrol;
|
2013-02-18 07:31:49 +00:00
|
|
|
|
|
|
|
bool_t tdisp_lld_init(void) {
|
|
|
|
/* initialise hardware */
|
|
|
|
init_board();
|
2013-04-12 07:12:00 +00:00
|
|
|
|
|
|
|
/* The first part is the initialing code.
|
|
|
|
* In this part only the lower nibble of the
|
|
|
|
* byte is written directly to the display, thus
|
|
|
|
* without write_cmd, which sends both high and
|
|
|
|
* low nibble.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Give the LCD a little time to wake up */
|
2013-06-02 08:58:12 +00:00
|
|
|
gfxSleepMilliseconds(15);
|
2013-04-12 07:12:00 +00:00
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* clear the RS-line to the display */
|
2013-04-12 07:12:00 +00:00
|
|
|
palClearPad(PORT_CTRL, PIN_RS);
|
2013-05-31 23:37:53 +00:00
|
|
|
// #if BUS_4BITS
|
|
|
|
/* write three times 0x03 to display (4-bit mode only)
|
|
|
|
* with RS = low.
|
|
|
|
*/
|
2013-04-12 07:12:00 +00:00
|
|
|
writeToLCD(0x03); // 1x
|
2013-06-02 08:58:12 +00:00
|
|
|
gfxSleepMilliseconds(20);
|
2013-04-12 07:12:00 +00:00
|
|
|
writeToLCD(0x03); // 2x
|
2013-06-02 08:58:12 +00:00
|
|
|
// gfxSleepMilliseconds(20);
|
2013-04-12 07:12:00 +00:00
|
|
|
writeToLCD(0x03); // 3x
|
2013-06-02 08:58:12 +00:00
|
|
|
// gfxSleepMilliseconds(20);
|
2013-04-12 07:12:00 +00:00
|
|
|
/* Put display in 4-bit mode by
|
|
|
|
* write 0x02 to display.
|
|
|
|
*/
|
|
|
|
writeToLCD(0x02); // 4bit-modus
|
2013-06-02 08:58:12 +00:00
|
|
|
// gfxSleepMilliseconds(20);
|
2013-05-31 23:37:53 +00:00
|
|
|
// #else
|
|
|
|
/* write three times 0x30 to display (8-bit mode only)
|
|
|
|
* with RS = low.
|
|
|
|
*/
|
|
|
|
// writeToLCD(0x30); // 1x
|
|
|
|
// writeToLCD(0x30); // 2x
|
|
|
|
// writeToLCD(0x30); // 3x
|
|
|
|
// #endif
|
2013-04-12 07:12:00 +00:00
|
|
|
|
|
|
|
/* From this point on, the LCD accepts
|
|
|
|
* bytes sent with highnibbel first and than
|
2013-05-31 23:37:53 +00:00
|
|
|
*the lownibble if working in 4-bit mode.
|
|
|
|
* In 8-bit mode bytes are written in 1 action.
|
2013-04-12 07:12:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* 4-bit modus, 2 lines en 5x7 characters */
|
|
|
|
write_cmd(0x28);
|
2013-06-02 08:58:12 +00:00
|
|
|
// gfxSleepMilliseconds(20);
|
2013-04-12 07:12:00 +00:00
|
|
|
|
|
|
|
/* set display on, cursor off and no blinking */
|
|
|
|
write_cmd(0x0C);
|
2013-06-02 08:58:12 +00:00
|
|
|
// gfxSleepMilliseconds(20);
|
2013-05-31 23:37:53 +00:00
|
|
|
/* set cursor increase and direction */
|
2013-02-18 07:31:49 +00:00
|
|
|
write_cmd(0x06);
|
2013-06-02 08:58:12 +00:00
|
|
|
// gfxSleepMilliseconds(20);
|
2013-05-31 23:37:53 +00:00
|
|
|
|
|
|
|
displaycontrol = TDISP_DISPLAY_ON;
|
|
|
|
cursorcontrol = TDISP_CURSOR_INC;
|
2013-04-12 07:12:00 +00:00
|
|
|
/* END OF INITIALISATION */
|
|
|
|
|
2013-02-18 07:31:49 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-04-12 07:12:00 +00:00
|
|
|
/* Clears the display. The display needs
|
|
|
|
* a long time to process this command. So
|
|
|
|
* an extra delay is insterted.
|
|
|
|
*/
|
2013-02-18 07:31:49 +00:00
|
|
|
void tdisp_lld_clear(void) {
|
|
|
|
write_cmd(0x01);
|
|
|
|
}
|
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* Writes a character to the display */
|
2013-02-18 07:31:49 +00:00
|
|
|
void tdisp_lld_draw_char(char c) {
|
|
|
|
write_data(c);
|
|
|
|
}
|
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* Set cursor position */
|
2013-02-18 07:31:49 +00:00
|
|
|
void tdisp_lld_set_cursor(coord_t col, coord_t row) {
|
|
|
|
static const uint8_t row_offsets[] = { 0x00, 0x40, 0x14, 0x54 };
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Short-cut:
|
|
|
|
*
|
|
|
|
* If x and y = 0 then use the home command.
|
|
|
|
*
|
|
|
|
* Note: There is probably no advantage as both commands are a single byte
|
|
|
|
*/
|
|
|
|
// if (col == 0 && row == 0) {
|
|
|
|
// write_cmd(0x02);
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
|
|
|
|
write_cmd(0x80 | (col + row_offsets[row]));
|
2013-05-31 23:37:53 +00:00
|
|
|
// write_cmd(0x80 | (0x40 + col));
|
2013-02-18 07:31:49 +00:00
|
|
|
}
|
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* Create a custom character to the display */
|
2013-02-18 07:31:49 +00:00
|
|
|
void tdisp_lld_create_char(uint8_t address, uint8_t *charmap) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
write_cmd(0x40 | (address << 3));
|
2013-04-12 07:12:00 +00:00
|
|
|
for(i = 0; i < CUSTOM_CHAR_YBITS; i++) {
|
2013-02-18 07:31:49 +00:00
|
|
|
write_data(charmap[i]);
|
2013-04-12 07:12:00 +00:00
|
|
|
}
|
|
|
|
|
2013-02-18 07:31:49 +00:00
|
|
|
}
|
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* Write display control commands to the display */
|
2013-04-12 07:12:00 +00:00
|
|
|
void tdisp_lld_control(uint16_t what, uint16_t value) {
|
2013-02-18 07:31:49 +00:00
|
|
|
switch(what) {
|
2013-05-31 23:37:53 +00:00
|
|
|
case TDISP_CTRL_DISPLAY:
|
|
|
|
switch (value) {
|
|
|
|
case displayOff:
|
|
|
|
displaycontrol &= ~TDISP_DISPLAY_ON;
|
|
|
|
break;
|
|
|
|
case displayOn:
|
|
|
|
displaycontrol |= TDISP_DISPLAY_ON;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
write_cmd(0x08 | displaycontrol);
|
2013-02-18 07:31:49 +00:00
|
|
|
break;
|
2013-05-31 23:37:53 +00:00
|
|
|
case TDISP_CTRL_CURSOR:
|
|
|
|
switch (value) {
|
|
|
|
case cursorBlinkingBlock:
|
|
|
|
case cursorBlinkingUnderline:
|
|
|
|
case cursorBlinkingBar:
|
|
|
|
displaycontrol |= TDISP_CURSOR_ON + TDISP_CURSOR_BLINK;
|
|
|
|
break;
|
|
|
|
case cursorBlock:
|
|
|
|
case cursorUnderline:
|
|
|
|
case cursorBar:
|
|
|
|
displaycontrol = (displaycontrol | TDISP_CURSOR_ON) & ~TDISP_CURSOR_BLINK;
|
|
|
|
break;
|
|
|
|
case cursorOff:
|
|
|
|
default:
|
|
|
|
displaycontrol &= ~(TDISP_CURSOR_ON | TDISP_CURSOR_BLINK); // zet alleen de cursor uit. Bewaar de overige instellingen
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
write_cmd(0x08 | displaycontrol);
|
2013-02-18 07:31:49 +00:00
|
|
|
break;
|
2013-05-31 23:37:53 +00:00
|
|
|
case TDISP_CTRL_MOVE:
|
|
|
|
switch (value) {
|
|
|
|
case cursorIncrease:
|
|
|
|
cursorcontrol |= TDISP_CURSOR_INC; // increase cursor position
|
|
|
|
break;
|
|
|
|
case cursorDecrease:
|
|
|
|
cursorcontrol &= ~TDISP_CURSOR_INC; // decrease cursor position
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
write_cmd(0x04 | cursorcontrol);
|
|
|
|
break;
|
|
|
|
case TDISP_CTRL_SHIFT:
|
|
|
|
switch (value) {
|
|
|
|
case shiftOn:
|
|
|
|
cursorcontrol |= TDISP_SHIFT_ON;
|
|
|
|
break;
|
|
|
|
case shiftOff:
|
|
|
|
cursorcontrol &= ~TDISP_SHIFT_ON;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
write_cmd(0x04 | cursorcontrol);
|
2013-02-18 07:31:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-31 23:37:53 +00:00
|
|
|
/* Scrolls the display in a particular direction with an amount of characters and delays the scroll between each scroll step */
|
|
|
|
void tdisp_lld_scroll(uint16_t direction, uint16_t amount, uint16_t delay) {
|
|
|
|
uint16_t scrollcontrol = 0;
|
|
|
|
uint16_t i;
|
|
|
|
|
|
|
|
switch (direction) {
|
|
|
|
case displayScrollLeft:
|
|
|
|
scrollcontrol = TDISP_SCROLL_DISPLAY;
|
|
|
|
break;
|
|
|
|
case displayScrollRight:
|
|
|
|
scrollcontrol = TDISP_SCROLL_DISPLAY | TDISP_SCROLL_RIGHT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(i = 0; i < amount; i++) {
|
|
|
|
write_cmd(0x10 | scrollcontrol);
|
2013-06-02 08:58:12 +00:00
|
|
|
gfxSleepMilliseconds(delay);
|
2013-05-31 23:37:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if TDISP_USE_BACKLIGHT
|
|
|
|
/* sets the brightness of the display backlight */
|
|
|
|
void tdisp_lld_set_backlight(uint16_t percentage) {
|
|
|
|
set_backlight(percentage);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-02-18 07:31:49 +00:00
|
|
|
#endif /* GFX_USE_TDISP */
|
|
|
|
/** @} */
|
|
|
|
|