1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/aip31068/aip31068.c
2020-09-29 14:42:15 +02:00

410 lines
11 KiB
C

/*
* Copyright (C) 2020 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup drivers_aip31068
* @brief Device driver for AIP31068
* @author Hendrik van Essen <hendrik.ve@fu-berlin.de>
* @file
* @{
*/
#include <stdio.h>
#include "xtimer.h"
#include "aip31068.h"
#include "aip31068_regs.h"
#include "aip31068_internal.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
/**
* @brief Write a data byte to the device.
*
* @param[in] dev Device descriptor of the AIP31068
* @param[in] value Data byte to write
*
* @return 0 on success
* @return -1 if acquiring of I2C bus fails
* @return -EIO When slave device doesn't ACK the byte
* @return -ENXIO When no devices respond on the address sent on the bus
* @return -ETIMEDOUT When timeout occurs before device's response
* @return -EINVAL When an invalid argument is given
* @return -EOPNOTSUPP When MCU driver doesn't support the flag operation
* @return -EAGAIN When a lost bus arbitration occurs
*/
static inline int _data(aip31068_t *dev, uint8_t value);
/**
* @brief Write a command byte with it's arguments (modified bits) to the device.
*
* @param[in] dev Device descriptor of the AIP31068
* @param[in] value Command byte to write
*
* @return 0 on success
* @return -1 if acquiring of I2C bus fails
* @return -EIO When slave device doesn't ACK the byte
* @return -ENXIO When no devices respond on the address sent on the bus
* @return -ETIMEDOUT When timeout occurs before device's response
* @return -EINVAL When an invalid argument is given
* @return -EOPNOTSUPP When MCU driver doesn't support the flag operation
* @return -EAGAIN When a lost bus arbitration occurs
*/
static inline int _command(aip31068_t *dev, uint8_t value);
/**
* @brief Write a command or data byte to the device.
*
* @param[in] dev Device descriptor of the AIP31068
* @param[in] data_byte Byte to write
* @param[in] is_cmd Whether byte should be interpreted as data or command
*
* @return 0 on success
* @return -1 if acquiring of I2C bus fails
* @return -EIO When slave device doesn't ACK the byte
* @return -ENXIO When no devices respond on the address sent on the bus
* @return -ETIMEDOUT When timeout occurs before device's response
* @return -EINVAL When an invalid argument is given
* @return -EOPNOTSUPP When MCU driver doesn't support the flag operation
* @return -EAGAIN When a lost bus arbitration occurs
*/
static inline int _write(aip31068_t *dev, uint8_t data_byte, bool is_cmd);
/**
* @brief Write data to the device.
*
* @param[in] dev Device descriptor of the AIP31068
* @param[in] data Data to write
* @param[in] len Length of the data
*
* @return 0 on success
* @return -1 if acquiring of I2C bus fails
* @return -EIO When slave device doesn't ACK the byte
* @return -ENXIO When no devices respond on the address sent on the bus
* @return -ETIMEDOUT When timeout occurs before device's response
* @return -EINVAL When an invalid argument is given
* @return -EOPNOTSUPP When MCU driver doesn't support the flag operation
* @return -EAGAIN When a lost bus arbitration occurs
*/
static int _device_write(aip31068_t *dev, uint8_t *data, uint8_t len);
int aip31068_init(aip31068_t *dev, const aip31068_params_t *params)
{
assert(dev);
assert(params);
/* displays with more than 4 lines are not supported
* (see aip31068_set_cursor_position) */
assert(params->row_count <= 4);
dev->params = *params;
dev->_curr_display_control = 0;
dev->_curr_entry_mode_set = 0;
i2c_init(dev->params.i2c_dev);
uint8_t _function_set = 0;
/* configure bit mode */
if (params->bit_mode == BITMODE_8_BIT) {
SETBIT(_function_set, AIP31068_BIT_FUNCTION_SET_BITMODE);
}
/* configure line count */
if (params->row_count >= 2) {
SETBIT(_function_set, AIP31068_BIT_FUNCTION_SET_LINECOUNT);
}
/* configure character size */
if (params->font_size == FONT_SIZE_5x10) {
SETBIT(_function_set, AIP31068_BIT_FUNCTION_SET_FONTSIZE);
}
/* Begin of the initialization sequence (page 20 in the datasheet).
* The initialization sequence here is writing AIP31068_CMD_FUNCTION_SET
* three times with different sleep periods in between according to the
* data sheet of the HD44780. The AIP31068 is forwarding commands to the
* HD44780 hidden behind it so to be sure that it works for all kinds of
* HD44780 we follow the initialization sequence of the HD44780, even though
* it might be unnecessary for others. */
xtimer_usleep(50 * US_PER_MS);
int rc = 0;
/* send function set command sequence */
rc = _command(dev, AIP31068_CMD_FUNCTION_SET | _function_set);
if (rc < 0) {
return rc;
}
/* wait after the first try */
xtimer_usleep(5 * US_PER_MS);
/* second try */
rc = _command(dev, AIP31068_CMD_FUNCTION_SET | _function_set);
if (rc < 0) {
return rc;
}
/* wait after the second try */
xtimer_usleep(500);
/* third go */
rc = _command(dev, AIP31068_CMD_FUNCTION_SET | _function_set);
if (rc < 0) {
return rc;
}
rc = aip31068_turn_off(dev);
if (rc < 0) {
return rc;
}
rc = aip31068_clear(dev);
if (rc < 0) {
return rc;
}
rc = aip31068_set_text_insertion_mode(dev, LEFT_TO_RIGHT);
if (rc < 0) {
return rc;
}
/* End of the initialization sequence. */
return 0;
}
int aip31068_turn_on(aip31068_t *dev)
{
SETBIT(dev->_curr_display_control, AIP31068_BIT_DISPLAY_CONTROL_DISPLAY);
return _command(dev, AIP31068_CMD_DISPLAY_CONTROL | dev->_curr_display_control);
}
int aip31068_turn_off(aip31068_t *dev)
{
CLRBIT(dev->_curr_display_control, AIP31068_BIT_DISPLAY_CONTROL_DISPLAY);
return _command(dev, AIP31068_CMD_DISPLAY_CONTROL | dev->_curr_display_control);
}
int aip31068_clear(aip31068_t *dev)
{
int rc = _command(dev, AIP31068_CMD_CLEAR_DISPLAY);
xtimer_usleep(AIP31068_EXECUTION_TIME_MAX);
return rc;
}
int aip31068_return_home(aip31068_t *dev)
{
int rc = _command(dev, AIP31068_CMD_RETURN_HOME);
xtimer_usleep(AIP31068_EXECUTION_TIME_MAX);
return rc;
}
int aip31068_set_auto_scroll_enabled(aip31068_t *dev, bool enabled)
{
if (enabled) {
SETBIT(dev->_curr_entry_mode_set, AIP31068_BIT_ENTRY_MODE_AUTOINCREMENT);
}
else {
CLRBIT(dev->_curr_entry_mode_set, AIP31068_BIT_ENTRY_MODE_AUTOINCREMENT);
}
return _command(dev, AIP31068_CMD_ENTRY_MODE_SET | dev->_curr_entry_mode_set);
}
int aip31068_set_cursor_blinking_enabled(aip31068_t *dev, bool enabled)
{
if (enabled) {
SETBIT(dev->_curr_display_control, AIP31068_BIT_DISPLAY_CONTROL_CURSOR_BLINKING);
}
else {
CLRBIT(dev->_curr_display_control, AIP31068_BIT_DISPLAY_CONTROL_CURSOR_BLINKING);
}
return _command(dev, AIP31068_CMD_DISPLAY_CONTROL | dev->_curr_display_control);
}
int aip31068_set_cursor_visible(aip31068_t *dev, bool visible)
{
if (visible) {
SETBIT(dev->_curr_display_control, AIP31068_BIT_DISPLAY_CONTROL_CURSOR);
}
else {
CLRBIT(dev->_curr_display_control, AIP31068_BIT_DISPLAY_CONTROL_CURSOR);
}
return _command(dev, AIP31068_CMD_DISPLAY_CONTROL | dev->_curr_display_control);
}
int aip31068_set_cursor_position(aip31068_t *dev, uint8_t row, uint8_t col)
{
uint8_t row_offsets[4];
row_offsets[0] = 0x00;
row_offsets[1] = 0x40;
row_offsets[2] = 0x00 + dev->params.col_count;
row_offsets[3] = 0x40 + dev->params.col_count;
if (row >= dev->params.row_count) {
row = dev->params.row_count - 1;
}
return _command(dev, AIP31068_CMD_SET_DDRAM_ADDR | (col | row_offsets[row]));
}
int aip31068_set_text_insertion_mode(aip31068_t *dev,
aip31068_text_insertion_mode_t mode)
{
if (mode == LEFT_TO_RIGHT) {
SETBIT(dev->_curr_entry_mode_set, AIP31068_BIT_ENTRY_MODE_INCREMENT);
}
else {
CLRBIT(dev->_curr_entry_mode_set, AIP31068_BIT_ENTRY_MODE_INCREMENT);
}
return _command(dev, AIP31068_CMD_ENTRY_MODE_SET | dev->_curr_entry_mode_set);
}
int aip31068_move_cursor_left(aip31068_t *dev)
{
uint8_t cmd = AIP31068_CMD_CURSOR_DISPLAY_SHIFT;
CLRBIT(cmd, AIP31068_BIT_CURSOR_DISPLAY_SHIFT_DIRECTION);
return _command(dev, cmd);
}
int aip31068_move_cursor_right(aip31068_t *dev)
{
uint8_t cmd = AIP31068_CMD_CURSOR_DISPLAY_SHIFT;
SETBIT(cmd, AIP31068_BIT_CURSOR_DISPLAY_SHIFT_DIRECTION);
return _command(dev, cmd);
}
int aip31068_scroll_display_left(aip31068_t *dev)
{
uint8_t cmd = AIP31068_CMD_CURSOR_DISPLAY_SHIFT;
SETBIT(cmd, AIP31068_BIT_CURSOR_DISPLAY_SHIFT_SELECTION);
CLRBIT(cmd, AIP31068_BIT_CURSOR_DISPLAY_SHIFT_DIRECTION);
return _command(dev, cmd);
}
int aip31068_scroll_display_right(aip31068_t *dev)
{
uint8_t cmd = AIP31068_CMD_CURSOR_DISPLAY_SHIFT;
SETBIT(cmd, AIP31068_BIT_CURSOR_DISPLAY_SHIFT_SELECTION);
SETBIT(cmd, AIP31068_BIT_CURSOR_DISPLAY_SHIFT_DIRECTION);
return _command(dev, cmd);
}
int aip31068_set_custom_symbol(aip31068_t *dev,
aip31068_custom_symbol_t custom_symbol,
const uint8_t charmap[])
{
/* Bits 0-2 define the row address of a custom character in CGRAM.
* Bits 3-5 define the base address of a custom character in CGRAM. */
uint8_t location = custom_symbol << 3;
int rc = _command(dev, AIP31068_CMD_SET_CGRAM_ADDR | location);
if (rc < 0) {
return rc;
}
/* How many rows are necessary for a complete character for given font? */
int row_count = dev->params.font_size == FONT_SIZE_5x8 ? 8 : 10;
for (int i = 0; i < row_count; i++) {
rc = _data(dev, charmap[i]);
if (rc < 0) {
return rc;
}
}
return _command(dev, AIP31068_CMD_SET_DDRAM_ADDR);
}
int aip31068_print_custom_symbol(aip31068_t *dev,
aip31068_custom_symbol_t custom_symbol)
{
return _data(dev, (uint8_t) custom_symbol);
}
int aip31068_print(aip31068_t *dev, const char *data)
{
while (*data != '\0') {
int rc;
if ((rc = _data(dev, *data)) != 0) {
return rc;
}
data++;
}
return 0;
}
int aip31068_print_char(aip31068_t *dev, char c)
{
return _data(dev, c);
}
static inline int _data(aip31068_t *dev, uint8_t value)
{
int rc = _write(dev, value, false);
xtimer_usleep(AIP31068_EXECUTION_TIME_DEFAULT);
return rc;
}
static inline int _command(aip31068_t *dev, uint8_t value)
{
int rc = _write(dev, value, true);
xtimer_usleep(AIP31068_EXECUTION_TIME_DEFAULT);
return rc;
}
static inline int _write(aip31068_t *dev, uint8_t data_byte, bool is_cmd)
{
uint8_t control_byte = 0;
if (!is_cmd) {
SETBIT(control_byte, AIP31068_BIT_CONTROL_BYTE_RS);
}
uint8_t data[] = { control_byte, data_byte };
return _device_write(dev, data, sizeof(data));
}
static int _device_write(aip31068_t* dev, uint8_t *data, uint8_t len)
{
i2c_t i2c_dev = dev->params.i2c_dev;
if (i2c_acquire(i2c_dev) != 0) {
return -1;
}
int rc = i2c_write_bytes(i2c_dev, dev->params.i2c_addr, data, len, 0);
i2c_release(i2c_dev);
return rc;
}