2018-02-22 20:19:18 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016-2018 Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
*
|
|
|
|
* 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 pkg_u8g2
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2019-01-03 00:39:56 +01:00
|
|
|
* @brief U8g2 driver for interacting with RIOT-OS peripherals
|
2018-02-22 20:19:18 +01:00
|
|
|
*
|
|
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
#include <assert.h>
|
2018-02-22 20:19:18 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
#include "u8x8_riotos.h"
|
2018-02-22 20:19:18 +01:00
|
|
|
|
|
|
|
#include "xtimer.h"
|
2019-01-03 00:39:56 +01:00
|
|
|
|
|
|
|
#ifdef MODULE_PERIPH_SPI
|
2018-02-22 20:19:18 +01:00
|
|
|
#include "periph/spi.h"
|
2019-01-03 00:39:56 +01:00
|
|
|
#endif
|
|
|
|
#ifdef MODULE_PERIPH_I2C
|
2018-02-22 20:19:18 +01:00
|
|
|
#include "periph/i2c.h"
|
2019-01-03 00:39:56 +01:00
|
|
|
#endif
|
2018-02-22 20:19:18 +01:00
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
#ifdef MODULE_PERIPH_SPI
|
2018-02-22 20:19:18 +01:00
|
|
|
static spi_clk_t u8x8_pulse_width_to_spi_speed(uint32_t pulse_width)
|
|
|
|
{
|
2019-01-03 00:39:56 +01:00
|
|
|
const uint32_t cycle_time = 2 * pulse_width;
|
2018-02-22 20:19:18 +01:00
|
|
|
|
|
|
|
if (cycle_time < 100) {
|
|
|
|
return SPI_CLK_10MHZ;
|
2019-01-03 00:39:56 +01:00
|
|
|
}
|
|
|
|
else if (cycle_time < 200) {
|
2018-02-22 20:19:18 +01:00
|
|
|
return SPI_CLK_5MHZ;
|
2019-01-03 00:39:56 +01:00
|
|
|
}
|
|
|
|
else if (cycle_time < 1000) {
|
2018-02-22 20:19:18 +01:00
|
|
|
return SPI_CLK_1MHZ;
|
2019-01-03 00:39:56 +01:00
|
|
|
}
|
|
|
|
else if (cycle_time < 2500) {
|
2018-02-22 20:19:18 +01:00
|
|
|
return SPI_CLK_400KHZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SPI_CLK_100KHZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
static spi_mode_t u8x8_spi_mode_to_spi_conf(uint32_t spi_mode)
|
|
|
|
{
|
|
|
|
return (spi_mode_t) spi_mode;
|
|
|
|
}
|
2019-01-03 00:39:56 +01:00
|
|
|
#endif /* MODULE_PERIPH_SPI */
|
2018-02-22 20:19:18 +01:00
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
/**
|
|
|
|
* @brief Enable the selected pins in RIOT-OS.
|
|
|
|
*/
|
|
|
|
static void _enable_pins(const u8x8_riotos_t *u8x8_riot_ptr)
|
2018-02-22 20:19:18 +01:00
|
|
|
{
|
2019-01-03 00:39:56 +01:00
|
|
|
/* no hardware peripheral is being used, nothing to be done */
|
|
|
|
if (u8x8_riot_ptr == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-08-31 13:02:32 +02:00
|
|
|
if (gpio_is_valid(u8x8_riot_ptr->pin_cs)) {
|
2019-01-03 00:39:56 +01:00
|
|
|
gpio_init(u8x8_riot_ptr->pin_cs, GPIO_OUT);
|
|
|
|
}
|
|
|
|
|
2020-08-31 13:02:32 +02:00
|
|
|
if (gpio_is_valid(u8x8_riot_ptr->pin_dc)) {
|
2019-01-03 00:39:56 +01:00
|
|
|
gpio_init(u8x8_riot_ptr->pin_dc, GPIO_OUT);
|
|
|
|
}
|
|
|
|
|
2020-08-31 13:02:32 +02:00
|
|
|
if (gpio_is_valid(u8x8_riot_ptr->pin_reset)) {
|
2019-01-03 00:39:56 +01:00
|
|
|
gpio_init(u8x8_riot_ptr->pin_reset, GPIO_OUT);
|
2018-02-22 20:19:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t u8x8_gpio_and_delay_riotos(u8x8_t *u8g2, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
(void) arg_ptr;
|
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
const u8x8_riotos_t *u8x8_riot_ptr = u8x8_GetUserPtr(u8g2);
|
|
|
|
|
|
|
|
/* assert that user_ptr is correctly set */
|
|
|
|
assert(u8x8_riot_ptr != NULL);
|
|
|
|
|
2018-02-22 20:19:18 +01:00
|
|
|
switch (msg) {
|
|
|
|
case U8X8_MSG_GPIO_AND_DELAY_INIT:
|
2019-01-03 00:39:56 +01:00
|
|
|
_enable_pins(u8x8_riot_ptr);
|
2018-02-22 20:19:18 +01:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_DELAY_MILLI:
|
|
|
|
xtimer_usleep(arg_int * 1000);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_DELAY_10MICRO:
|
|
|
|
xtimer_usleep(arg_int * 10);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_DELAY_100NANO:
|
|
|
|
xtimer_nanosleep(arg_int * 100);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_GPIO_CS:
|
2020-08-31 13:02:32 +02:00
|
|
|
if (u8x8_riot_ptr != NULL && gpio_is_valid(u8x8_riot_ptr->pin_cs)) {
|
2019-01-03 00:39:56 +01:00
|
|
|
gpio_write(u8x8_riot_ptr->pin_cs, arg_int);
|
2018-02-22 20:19:18 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_GPIO_DC:
|
2020-08-31 13:02:32 +02:00
|
|
|
if (u8x8_riot_ptr != NULL && gpio_is_valid(u8x8_riot_ptr->pin_dc)) {
|
2019-01-03 00:39:56 +01:00
|
|
|
gpio_write(u8x8_riot_ptr->pin_dc, arg_int);
|
2018-02-22 20:19:18 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_GPIO_RESET:
|
2020-08-31 13:02:32 +02:00
|
|
|
if (u8x8_riot_ptr != NULL && gpio_is_valid(u8x8_riot_ptr->pin_reset)) {
|
2019-01-03 00:39:56 +01:00
|
|
|
gpio_write(u8x8_riot_ptr->pin_reset, arg_int);
|
2018-02-22 20:19:18 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
#ifdef MODULE_PERIPH_SPI
|
|
|
|
uint8_t u8x8_byte_hw_spi_riotos(u8x8_t *u8g2, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
2018-02-22 20:19:18 +01:00
|
|
|
{
|
2019-01-03 00:39:56 +01:00
|
|
|
const u8x8_riotos_t *u8x8_riot_ptr = u8x8_GetUserPtr(u8g2);
|
|
|
|
|
|
|
|
/* assert that user_ptr is correctly set */
|
|
|
|
assert(u8x8_riot_ptr != NULL);
|
|
|
|
|
|
|
|
spi_t dev = SPI_DEV(u8x8_riot_ptr->device_index);
|
2018-02-22 20:19:18 +01:00
|
|
|
|
|
|
|
switch (msg) {
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
|
|
|
spi_transfer_bytes(dev, GPIO_UNDEF, true,
|
|
|
|
arg_ptr, NULL, (size_t)arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
u8x8_gpio_SetDC(u8g2, arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
spi_acquire(dev, GPIO_UNDEF,
|
|
|
|
u8x8_spi_mode_to_spi_conf(u8g2->display_info->spi_mode),
|
|
|
|
u8x8_pulse_width_to_spi_speed(u8g2->display_info->sck_pulse_width_ns));
|
|
|
|
|
|
|
|
u8x8_gpio_SetCS(u8g2, u8g2->display_info->chip_enable_level);
|
|
|
|
u8g2->gpio_and_delay_cb(u8g2, U8X8_MSG_DELAY_NANO, u8g2->display_info->post_chip_enable_wait_ns, NULL);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
u8g2->gpio_and_delay_cb(u8g2, U8X8_MSG_DELAY_NANO, u8g2->display_info->pre_chip_disable_wait_ns, NULL);
|
|
|
|
u8x8_gpio_SetCS(u8g2, u8g2->display_info->chip_disable_level);
|
|
|
|
|
|
|
|
spi_release(dev);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
2019-01-03 00:39:56 +01:00
|
|
|
#endif /* MODULE_PERIPH_SPI */
|
2018-02-22 20:19:18 +01:00
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
#ifdef MODULE_PERIPH_I2C
|
|
|
|
uint8_t u8x8_byte_hw_i2c_riotos(u8x8_t *u8g2, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
2018-02-22 20:19:18 +01:00
|
|
|
{
|
2019-01-03 00:39:56 +01:00
|
|
|
static uint8_t buffer[32]; /* u8x8 will never send more than 32 bytes
|
|
|
|
between START_TRANSFER and END_TRANSFER */
|
|
|
|
static size_t index;
|
|
|
|
|
|
|
|
const u8x8_riotos_t *u8x8_riot_ptr = u8x8_GetUserPtr(u8g2);
|
|
|
|
|
|
|
|
/* assert that user_ptr is correctly set */
|
|
|
|
assert(u8x8_riot_ptr != NULL);
|
2018-02-22 20:19:18 +01:00
|
|
|
|
2019-01-03 00:39:56 +01:00
|
|
|
i2c_t dev = I2C_DEV(u8x8_riot_ptr->device_index);
|
2018-02-22 20:19:18 +01:00
|
|
|
|
|
|
|
switch (msg) {
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
|
|
|
memcpy(&buffer[index], arg_ptr, arg_int);
|
|
|
|
index += arg_int;
|
2019-01-03 00:39:56 +01:00
|
|
|
assert(index <= sizeof(buffer));
|
2018-02-22 20:19:18 +01:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
i2c_acquire(dev);
|
|
|
|
index = 0;
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
2018-05-27 23:13:17 +02:00
|
|
|
i2c_write_bytes(dev, u8x8_GetI2CAddress(u8g2), buffer, index, 0);
|
2018-02-22 20:19:18 +01:00
|
|
|
i2c_release(dev);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
2019-01-03 00:39:56 +01:00
|
|
|
#endif /* MODULE_PERIPH_I2C */
|