mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 23:32:59 +01:00
d63d58ac0b
The expandable GPIO API requires the comparison of structured GPIO types. This means that inline functions must be used instead of direct comparisons. For the migration process, packages must first be changed so that they use the inline comparison functions.
211 lines
5.7 KiB
C
211 lines
5.7 KiB
C
/*
|
|
* 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
|
|
* @brief U8g2 driver for interacting with RIOT-OS peripherals
|
|
*
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include "u8x8_riotos.h"
|
|
|
|
#include "xtimer.h"
|
|
|
|
#ifdef MODULE_PERIPH_SPI
|
|
#include "periph/spi.h"
|
|
#endif
|
|
#ifdef MODULE_PERIPH_I2C
|
|
#include "periph/i2c.h"
|
|
#endif
|
|
|
|
#ifdef MODULE_PERIPH_SPI
|
|
static spi_clk_t u8x8_pulse_width_to_spi_speed(uint32_t pulse_width)
|
|
{
|
|
const uint32_t cycle_time = 2 * pulse_width;
|
|
|
|
if (cycle_time < 100) {
|
|
return SPI_CLK_10MHZ;
|
|
}
|
|
else if (cycle_time < 200) {
|
|
return SPI_CLK_5MHZ;
|
|
}
|
|
else if (cycle_time < 1000) {
|
|
return SPI_CLK_1MHZ;
|
|
}
|
|
else if (cycle_time < 2500) {
|
|
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;
|
|
}
|
|
#endif /* MODULE_PERIPH_SPI */
|
|
|
|
/**
|
|
* @brief Enable the selected pins in RIOT-OS.
|
|
*/
|
|
static void _enable_pins(const u8x8_riotos_t *u8x8_riot_ptr)
|
|
{
|
|
/* no hardware peripheral is being used, nothing to be done */
|
|
if (u8x8_riot_ptr == NULL) {
|
|
return;
|
|
}
|
|
|
|
if (gpio_is_valid(u8x8_riot_ptr->pin_cs)) {
|
|
gpio_init(u8x8_riot_ptr->pin_cs, GPIO_OUT);
|
|
}
|
|
|
|
if (gpio_is_valid(u8x8_riot_ptr->pin_dc)) {
|
|
gpio_init(u8x8_riot_ptr->pin_dc, GPIO_OUT);
|
|
}
|
|
|
|
if (gpio_is_valid(u8x8_riot_ptr->pin_reset)) {
|
|
gpio_init(u8x8_riot_ptr->pin_reset, GPIO_OUT);
|
|
}
|
|
}
|
|
|
|
uint8_t u8x8_gpio_and_delay_riotos(u8x8_t *u8g2, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
{
|
|
(void) arg_ptr;
|
|
|
|
const u8x8_riotos_t *u8x8_riot_ptr = u8x8_GetUserPtr(u8g2);
|
|
|
|
/* assert that user_ptr is correctly set */
|
|
assert(u8x8_riot_ptr != NULL);
|
|
|
|
switch (msg) {
|
|
case U8X8_MSG_GPIO_AND_DELAY_INIT:
|
|
_enable_pins(u8x8_riot_ptr);
|
|
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:
|
|
if (u8x8_riot_ptr != NULL && gpio_is_valid(u8x8_riot_ptr->pin_cs)) {
|
|
gpio_write(u8x8_riot_ptr->pin_cs, arg_int);
|
|
}
|
|
break;
|
|
case U8X8_MSG_GPIO_DC:
|
|
if (u8x8_riot_ptr != NULL && gpio_is_valid(u8x8_riot_ptr->pin_dc)) {
|
|
gpio_write(u8x8_riot_ptr->pin_dc, arg_int);
|
|
}
|
|
break;
|
|
case U8X8_MSG_GPIO_RESET:
|
|
if (u8x8_riot_ptr != NULL && gpio_is_valid(u8x8_riot_ptr->pin_reset)) {
|
|
gpio_write(u8x8_riot_ptr->pin_reset, arg_int);
|
|
}
|
|
break;
|
|
default:
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
#ifdef MODULE_PERIPH_SPI
|
|
uint8_t u8x8_byte_hw_spi_riotos(u8x8_t *u8g2, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
{
|
|
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);
|
|
|
|
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;
|
|
}
|
|
#endif /* MODULE_PERIPH_SPI */
|
|
|
|
#ifdef MODULE_PERIPH_I2C
|
|
uint8_t u8x8_byte_hw_i2c_riotos(u8x8_t *u8g2, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
{
|
|
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);
|
|
|
|
i2c_t dev = I2C_DEV(u8x8_riot_ptr->device_index);
|
|
|
|
switch (msg) {
|
|
case U8X8_MSG_BYTE_SEND:
|
|
memcpy(&buffer[index], arg_ptr, arg_int);
|
|
index += arg_int;
|
|
assert(index <= sizeof(buffer));
|
|
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:
|
|
i2c_write_bytes(dev, u8x8_GetI2CAddress(u8g2), buffer, index, 0);
|
|
i2c_release(dev);
|
|
break;
|
|
default:
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
#endif /* MODULE_PERIPH_I2C */
|