You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

189 lines
3.4 KiB
C

/**
* @file lcd.c
* A basic library for writing a 16x2 text LCD.
*
* Copyright (C) 2018 Clyne Sullivan
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include <lcd.h>
#include <clock.h>
#include <gpio.h>
#include <string.h>
//#define USE_DELAY
#define LCD_D0 GPIO_PORT(A, 0)
#define LCD_D1 GPIO_PORT(A, 1)
#define LCD_D2 GPIO_PORT(A, 4)
#define LCD_D3 GPIO_PORT(B, 0)
#define LCD_D4 GPIO_PORT(C, 1)
#define LCD_D5 GPIO_PORT(C, 0)
#define LCD_D6 GPIO_PORT(C, 2)
#define LCD_D7 GPIO_PORT(C, 3)
#define LCD_E GPIO_PORT(C, 12)
#define LCD_RS GPIO_PORT(C, 10)
#define lcd_data() gpio_dout(LCD_RS, 1)
void lcd_pulse(void)
{
gpio_dout(LCD_E, 1);
#ifdef USE_DELAY
delay(1);
#else
for (uint16_t i = 0; i < 10000; i++)
asm("");
#endif // USE_DELAY
gpio_dout(LCD_E, 0);
}
void lcd_byte(uint8_t byte)
{
gpio_dout(LCD_D0, byte & 0x01);
gpio_dout(LCD_D1, byte & 0x02);
gpio_dout(LCD_D2, byte & 0x04);
gpio_dout(LCD_D3, byte & 0x08);
gpio_dout(LCD_D4, byte & 0x10);
gpio_dout(LCD_D5, byte & 0x20);
gpio_dout(LCD_D6, byte & 0x40);
gpio_dout(LCD_D7, byte & 0x80);
}
void lcd_cmd(uint8_t cmd)
{
gpio_dout(LCD_RS, 0);
lcd_byte(cmd);
lcd_pulse();
}
void lcd_putchar(int c)
{
lcd_data();
lcd_byte((uint8_t)c);
lcd_pulse();
}
static int lcd_index = 0;
void lcd_puts(const char *s)
{
lcd_cmd(0x06);
while (*s) {
lcd_putchar(*s++);
if (++lcd_index == 0x10) {
lcd_cmd(0x80 | 0x40);
} else if (lcd_index == 0x20) {
lcd_cmd(0x80);
lcd_index = 0;
}
}
}
extern char *itoa(int n, int base);
void lcd_puti(int i)
{
lcd_puts(itoa(i, 10));
}
void lcd_puth(int h)
{
lcd_puts(itoa(h, 16));
}
void lcd_putb(uint8_t b)
{
lcd_puts(itoa(b, 2));
}
void lcd_clear(void)
{
lcd_cmd(0x01);
delay(2);
lcd_index = 0;
}
void lcd_init(void)
{
gpio_mode(LCD_D0, OUTPUT);
gpio_mode(LCD_D1, OUTPUT);
gpio_mode(LCD_D2, OUTPUT);
gpio_mode(LCD_D3, OUTPUT);
gpio_mode(LCD_D4, OUTPUT);
gpio_mode(LCD_D5, OUTPUT);
gpio_mode(LCD_D6, OUTPUT);
gpio_mode(LCD_D7, OUTPUT);
gpio_mode(LCD_E, OUTPUT);
gpio_mode(LCD_RS, OUTPUT);
gpio_dout(LCD_D0, 0);
gpio_dout(LCD_D1, 0);
gpio_dout(LCD_D2, 0);
gpio_dout(LCD_D3, 0);
gpio_dout(LCD_D4, 0);
gpio_dout(LCD_D5, 0);
gpio_dout(LCD_D6, 0);
gpio_dout(LCD_D7, 0);
gpio_dout(LCD_E, 0);
gpio_dout(LCD_RS, 0);
lcd_cmd(0x38);
lcd_cmd(0x10);
lcd_cmd(0x0D);
delay(5);
lcd_clear();
}
/**
* Task code
*/
volatile int bufpos = 0;
volatile char buf[32];
volatile uint8_t using = 0;
void lcd_clearbuf(void)
{
bufpos = 0;
for (int i = 0; i < 32; i++)
buf[i] = 0;
}
void lcd_put(const char *s)
{
int len = strlen(s);
int i;
using = 1;
for (i = 0; i < len; bufpos++, i++) {
if (bufpos > 31)
bufpos = 0;
buf[bufpos] = s[i];
}
using = 0;
}
void lcd_handler(void)
{
lcd_init();
lcd_clearbuf();
while (1) {
if (!using && buf[0] != '\0') {
lcd_puts(buf);
lcd_clearbuf();
}
delay(100);
}
}