2018-04-05 14:20:42 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 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_ucglib
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Ucglib driver to interact with RIOT-OS drivers.
|
|
|
|
*
|
|
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:58:21 +02:00
|
|
|
#include <assert.h>
|
2020-01-12 18:24:06 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "ucg_riotos.h"
|
2018-04-05 14:20:42 +02:00
|
|
|
|
2021-12-01 15:07:47 +01:00
|
|
|
#include "ztimer.h"
|
2018-04-05 14:20:42 +02:00
|
|
|
|
2020-01-12 18:24:06 +01:00
|
|
|
#ifdef MODULE_PERIPH_SPI
|
2018-04-05 14:20:42 +02:00
|
|
|
#include "periph/spi.h"
|
2020-01-12 18:24:06 +01:00
|
|
|
#endif
|
2018-04-05 14:20:42 +02:00
|
|
|
#include "periph/gpio.h"
|
|
|
|
|
2020-01-12 18:24:06 +01:00
|
|
|
#ifdef MODULE_PERIPH_SPI
|
2018-04-05 14:20:42 +02:00
|
|
|
static spi_clk_t ucg_serial_clk_speed_to_spi_speed(uint32_t serial_clk_speed)
|
|
|
|
{
|
|
|
|
if (serial_clk_speed < 100) {
|
|
|
|
return SPI_CLK_10MHZ;
|
|
|
|
}
|
|
|
|
else if (serial_clk_speed < 200) {
|
|
|
|
return SPI_CLK_5MHZ;
|
|
|
|
}
|
|
|
|
else if (serial_clk_speed < 1000) {
|
|
|
|
return SPI_CLK_1MHZ;
|
|
|
|
}
|
|
|
|
else if (serial_clk_speed < 2500) {
|
|
|
|
return SPI_CLK_400KHZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SPI_CLK_100KHZ;
|
|
|
|
}
|
2020-01-12 18:24:06 +01:00
|
|
|
#endif /* MODULE_PERIPH_SPI */
|
2018-04-05 14:20:42 +02:00
|
|
|
|
2020-01-12 18:24:06 +01:00
|
|
|
/**
|
|
|
|
* @brief Enable the selected pins in RIOT-OS.
|
|
|
|
*/
|
|
|
|
static void _enable_pins(const ucg_riotos_t *ucg_riot_ptr)
|
2018-04-05 14:20:42 +02:00
|
|
|
{
|
2020-01-12 18:24:06 +01:00
|
|
|
/* no hardware peripheral is being used, nothing to be done */
|
|
|
|
if (ucg_riot_ptr == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
2018-04-05 14:20:42 +02:00
|
|
|
|
2020-08-31 13:02:32 +02:00
|
|
|
if (gpio_is_valid(ucg_riot_ptr->pin_cs)) {
|
2020-01-12 18:24:06 +01:00
|
|
|
gpio_init(ucg_riot_ptr->pin_cs, GPIO_OUT);
|
|
|
|
}
|
|
|
|
|
2020-08-31 13:02:32 +02:00
|
|
|
if (gpio_is_valid(ucg_riot_ptr->pin_cd)) {
|
2020-01-12 18:24:06 +01:00
|
|
|
gpio_init(ucg_riot_ptr->pin_cd, GPIO_OUT);
|
|
|
|
}
|
|
|
|
|
2020-08-31 13:02:32 +02:00
|
|
|
if (gpio_is_valid(ucg_riot_ptr->pin_reset)) {
|
2020-01-12 18:24:06 +01:00
|
|
|
gpio_init(ucg_riot_ptr->pin_reset, GPIO_OUT);
|
2018-04-05 14:20:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-12 18:24:06 +01:00
|
|
|
#ifdef MODULE_PERIPH_SPI
|
|
|
|
int16_t ucg_com_hw_spi_riotos(ucg_t *ucg, int16_t msg, uint16_t arg, uint8_t *data)
|
2018-04-05 14:20:42 +02:00
|
|
|
{
|
2020-01-12 18:24:06 +01:00
|
|
|
const ucg_riotos_t *ucg_riot_ptr = ucg_GetUserPtr(ucg);
|
|
|
|
|
|
|
|
/* assert that user_ptr is correctly set */
|
|
|
|
assert(ucg_riot_ptr != NULL);
|
|
|
|
|
|
|
|
spi_t dev = SPI_DEV(ucg_riot_ptr->device_index);
|
2018-04-05 14:20:42 +02:00
|
|
|
|
|
|
|
switch (msg) {
|
|
|
|
case UCG_COM_MSG_POWER_UP:
|
|
|
|
/* setup pins */
|
2020-01-12 18:24:06 +01:00
|
|
|
_enable_pins(ucg_riot_ptr);
|
2018-04-05 14:20:42 +02:00
|
|
|
|
2020-01-12 18:24:06 +01:00
|
|
|
/* setup SPI */
|
2018-04-05 14:20:42 +02:00
|
|
|
spi_init_pins(dev);
|
2021-11-12 16:12:51 +01:00
|
|
|
/* correct alignment of data can be assumed, as in pkg callers use
|
|
|
|
* ucg_com_info_t to allocate memory */
|
|
|
|
ucg_com_info_t *info = (void *)(uintptr_t)data;
|
|
|
|
spi_clk_t speed = ucg_serial_clk_speed_to_spi_speed(info->serial_clk_speed);
|
|
|
|
spi_acquire(dev, GPIO_UNDEF, SPI_MODE_0, speed);
|
2018-04-05 14:20:42 +02:00
|
|
|
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_POWER_DOWN:
|
|
|
|
spi_release(dev);
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_DELAY:
|
2021-12-01 15:07:47 +01:00
|
|
|
ztimer_sleep(ZTIMER_USEC, arg);
|
2018-04-05 14:20:42 +02:00
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_CHANGE_RESET_LINE:
|
2020-08-31 13:02:32 +02:00
|
|
|
if (ucg_riot_ptr != NULL && gpio_is_valid(ucg_riot_ptr->pin_reset)) {
|
2020-01-12 18:24:06 +01:00
|
|
|
gpio_write(ucg_riot_ptr->pin_reset, arg);
|
2018-04-05 14:20:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_CHANGE_CS_LINE:
|
2020-08-31 13:02:32 +02:00
|
|
|
if (ucg_riot_ptr != NULL && gpio_is_valid(ucg_riot_ptr->pin_cs)) {
|
2020-01-12 18:24:06 +01:00
|
|
|
gpio_write(ucg_riot_ptr->pin_cs, arg);
|
2018-04-05 14:20:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_CHANGE_CD_LINE:
|
2020-08-31 13:02:32 +02:00
|
|
|
if (ucg_riot_ptr != NULL && gpio_is_valid(ucg_riot_ptr->pin_cd)) {
|
2020-01-12 18:24:06 +01:00
|
|
|
gpio_write(ucg_riot_ptr->pin_cd, arg);
|
2018-04-05 14:20:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_SEND_BYTE:
|
|
|
|
spi_transfer_byte(dev, GPIO_UNDEF, true, (uint8_t) arg);
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_REPEAT_1_BYTE:
|
|
|
|
while (arg--) {
|
|
|
|
spi_transfer_byte(dev, GPIO_UNDEF, true, ((uint8_t *) data)[0]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_REPEAT_2_BYTES:
|
|
|
|
while (arg--) {
|
|
|
|
spi_transfer_bytes(dev, GPIO_UNDEF, true, data, NULL, 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_REPEAT_3_BYTES:
|
|
|
|
while (arg--) {
|
|
|
|
spi_transfer_bytes(dev, GPIO_UNDEF, true, data, NULL, 3);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_SEND_STR:
|
|
|
|
spi_transfer_bytes(dev, GPIO_UNDEF, true, data, NULL, arg);
|
|
|
|
break;
|
|
|
|
case UCG_COM_MSG_SEND_CD_DATA_SEQUENCE:
|
|
|
|
while (arg--) {
|
|
|
|
if (*data != 0) {
|
2020-01-12 18:24:06 +01:00
|
|
|
if (ucg_riot_ptr != NULL && ucg_riot_ptr->pin_cd != GPIO_UNDEF) {
|
|
|
|
gpio_write(ucg_riot_ptr->pin_cd, *data);
|
2018-04-05 14:20:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
data++;
|
|
|
|
spi_transfer_bytes(dev, GPIO_UNDEF, true, data, NULL, 1);
|
|
|
|
data++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
2020-01-12 18:24:06 +01:00
|
|
|
#endif /* MODULE_PERIPH_SPI */
|
2018-04-05 14:20:42 +02:00
|
|
|
|
2020-01-12 18:24:06 +01:00
|
|
|
ucg_int_t ucg_dev_dummy_riotos(ucg_t *ucg, ucg_int_t msg, void *data)
|
2018-04-05 14:20:42 +02:00
|
|
|
{
|
|
|
|
static uint32_t pixels;
|
|
|
|
|
|
|
|
switch (msg) {
|
|
|
|
case UCG_MSG_DEV_POWER_UP:
|
|
|
|
puts("ucg: UCG_MSG_DEV_POWER_UP");
|
|
|
|
return 1;
|
|
|
|
case UCG_MSG_DEV_POWER_DOWN:
|
|
|
|
puts("ucg: UCG_MSG_DEV_POWER_DOWN");
|
|
|
|
return 1;
|
|
|
|
case UCG_MSG_GET_DIMENSION:
|
|
|
|
puts("ucg: UCG_MSG_GET_DIMENSION");
|
|
|
|
((ucg_wh_t *)data)->w = 128;
|
|
|
|
((ucg_wh_t *)data)->h = 128;
|
|
|
|
return 1;
|
|
|
|
case UCG_MSG_DRAW_PIXEL:
|
|
|
|
pixels++;
|
|
|
|
|
|
|
|
/* log each 128th draw */
|
|
|
|
if (pixels % 128 == 0) {
|
|
|
|
printf("ucg: UCG_MSG_DRAW_PIXEL (%" PRIu32 ")\n", pixels);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return ucg_dev_default_cb(ucg, msg, data);
|
|
|
|
}
|