2016-01-01 16:27:17 +08:00
|
|
|
/*
|
|
|
|
|
|
|
|
U8x8lib.cpp
|
|
|
|
|
|
|
|
Arduino specific low level functions
|
|
|
|
|
|
|
|
|
2016-01-04 00:10:50 +08:00
|
|
|
Universal 8bit Graphics Library (https://github.com/olikraus/u8g2/)
|
2016-01-01 16:27:17 +08:00
|
|
|
|
|
|
|
Copyright (c) 2016, olikraus@gmail.com
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
are permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above copyright notice, this list
|
|
|
|
of conditions and the following disclaimer.
|
|
|
|
|
|
|
|
* Redistributions in binary form must reproduce the above copyright notice, this
|
|
|
|
list of conditions and the following disclaimer in the documentation and/or other
|
|
|
|
materials provided with the distribution.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
|
|
|
|
CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
|
|
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
|
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2015-12-13 00:26:10 +08:00
|
|
|
#include "U8x8lib.h"
|
2016-01-02 07:25:06 +08:00
|
|
|
#ifdef U8X8_HAVE_HW_SPI
|
2015-12-13 00:26:10 +08:00
|
|
|
#include <SPI.h>
|
2016-01-02 07:25:06 +08:00
|
|
|
#endif
|
2016-05-15 15:56:06 +08:00
|
|
|
#ifdef U8X8_HAVE_HW_I2C
|
2016-05-15 06:05:41 +08:00
|
|
|
#include <Wire.h>
|
|
|
|
#endif
|
2015-12-13 00:26:10 +08:00
|
|
|
|
2016-11-05 04:10:42 +08:00
|
|
|
/*=============================================*/
|
|
|
|
|
|
|
|
size_t U8X8::write(uint8_t v)
|
|
|
|
{
|
|
|
|
if ( v == '\n' )
|
|
|
|
{
|
|
|
|
ty++;
|
|
|
|
tx=0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
u8x8_DrawGlyph(&u8x8, tx, ty, v);
|
|
|
|
tx++;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-12-13 00:26:10 +08:00
|
|
|
/*=============================================*/
|
|
|
|
/* callbacks */
|
|
|
|
|
2017-08-23 16:32:28 +08:00
|
|
|
#ifdef U8X8_USE_PINS
|
2016-10-07 03:19:38 +08:00
|
|
|
extern "C" uint8_t u8x8_gpio_and_delay_arduino(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, U8X8_UNUSED void *arg_ptr)
|
2015-12-13 00:26:10 +08:00
|
|
|
{
|
|
|
|
uint8_t i;
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_GPIO_AND_DELAY_INIT:
|
|
|
|
|
|
|
|
for( i = 0; i < U8X8_PIN_CNT; i++ )
|
|
|
|
if ( u8x8->pins[i] != U8X8_PIN_NONE )
|
2016-05-17 05:13:09 +08:00
|
|
|
{
|
|
|
|
if ( i < U8X8_PIN_OUTPUT_CNT )
|
|
|
|
{
|
|
|
|
pinMode(u8x8->pins[i], OUTPUT);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
#ifdef INPUT_PULLUP
|
|
|
|
pinMode(u8x8->pins[i], INPUT_PULLUP);
|
|
|
|
#else
|
|
|
|
pinMode(u8x8->pins[i], OUTPUT);
|
|
|
|
digitalWrite(u8x8->pins[i], 1);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2015-12-13 00:26:10 +08:00
|
|
|
|
|
|
|
break;
|
2016-08-19 03:12:36 +08:00
|
|
|
|
|
|
|
#ifndef __AVR__
|
|
|
|
/* this case is not compiled for any AVR, because AVR uC are so slow */
|
|
|
|
/* that this delay does not matter */
|
|
|
|
case U8X8_MSG_DELAY_NANO:
|
|
|
|
delayMicroseconds(arg_int==0?0:1);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
case U8X8_MSG_DELAY_10MICRO:
|
|
|
|
/* not used at the moment */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_DELAY_100NANO:
|
|
|
|
/* not used at the moment */
|
|
|
|
break;
|
|
|
|
|
2015-12-13 00:26:10 +08:00
|
|
|
case U8X8_MSG_DELAY_MILLI:
|
|
|
|
delay(arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_DELAY_I2C:
|
|
|
|
/* arg_int is 1 or 4: 100KHz (5us) or 400KHz (1.25us) */
|
|
|
|
delayMicroseconds(arg_int<=2?5:2);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_GPIO_I2C_CLOCK:
|
|
|
|
case U8X8_MSG_GPIO_I2C_DATA:
|
|
|
|
if ( arg_int == 0 )
|
|
|
|
{
|
|
|
|
pinMode(u8x8_GetPinValue(u8x8, msg), OUTPUT);
|
|
|
|
digitalWrite(u8x8_GetPinValue(u8x8, msg), 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-01-02 07:25:06 +08:00
|
|
|
#ifdef INPUT_PULLUP
|
2015-12-13 00:26:10 +08:00
|
|
|
pinMode(u8x8_GetPinValue(u8x8, msg), INPUT_PULLUP);
|
2016-01-02 07:25:06 +08:00
|
|
|
#else
|
|
|
|
pinMode(u8x8_GetPinValue(u8x8, msg), OUTPUT);
|
|
|
|
digitalWrite(u8x8_GetPinValue(u8x8, msg), 1);
|
|
|
|
#endif
|
2015-12-13 00:26:10 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if ( msg >= U8X8_MSG_GPIO(0) )
|
|
|
|
{
|
|
|
|
i = u8x8_GetPinValue(u8x8, msg);
|
|
|
|
if ( i != U8X8_PIN_NONE )
|
2016-05-17 05:13:09 +08:00
|
|
|
{
|
|
|
|
if ( u8x8_GetPinIndex(u8x8, msg) < U8X8_PIN_OUTPUT_CNT )
|
|
|
|
{
|
|
|
|
digitalWrite(i, arg_int);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
u8x8_SetGPIOResult(u8x8, digitalRead(i) == 0 ? 0 : 1);
|
|
|
|
}
|
|
|
|
}
|
2015-12-13 00:26:10 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
2017-08-23 16:32:28 +08:00
|
|
|
#endif // U8X8_USE_PINS
|
2015-12-13 00:26:10 +08:00
|
|
|
|
|
|
|
|
|
|
|
/*=============================================*/
|
|
|
|
|
2016-05-16 15:16:54 +08:00
|
|
|
/*
|
|
|
|
replacement for a more faster u8x8_byte_4wire_sw_spi
|
|
|
|
in general u8x8_byte_4wire_sw_spi could be a fallback:
|
|
|
|
|
|
|
|
uint8_t u8x8_byte_arduino_4wire_sw_spi(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_4wire_sw_spi(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __AVR_ARCH__
|
|
|
|
#define __AVR_ARCH__ 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(U8X8_USE_PINS)
|
|
|
|
/* no pin information (very strange), so fallback */
|
|
|
|
uint8_t u8x8_byte_arduino_4wire_sw_spi(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_4wire_sw_spi(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif __AVR_ARCH__ == 4 || __AVR_ARCH__ == 5 || __AVR_ARCH__ == 51 || __AVR_ARCH__ == 6
|
|
|
|
|
|
|
|
/* this function completly replaces u8x8_byte_4wire_sw_spi*/
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_4wire_sw_spi(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
uint8_t i, b;
|
|
|
|
uint8_t *data;
|
2016-07-27 02:38:55 +08:00
|
|
|
uint8_t takeover_edge = u8x8_GetSPIClockPhase(u8x8);
|
2016-11-26 22:12:24 +08:00
|
|
|
//uint8_t not_takeover_edge = 1 - takeover_edge;
|
2016-05-16 15:16:54 +08:00
|
|
|
|
|
|
|
/* the following static vars are recalculated in U8X8_MSG_BYTE_START_TRANSFER */
|
|
|
|
/* so, it should be possible to used multiple displays with different pins */
|
|
|
|
|
|
|
|
static volatile uint8_t *arduino_clock_port;
|
2016-11-26 22:12:24 +08:00
|
|
|
|
|
|
|
static uint8_t arduino_clock_mask;
|
|
|
|
static uint8_t arduino_clock_n_mask;
|
2016-05-16 15:16:54 +08:00
|
|
|
|
|
|
|
static volatile uint8_t *arduino_data_port;
|
2016-11-26 22:12:24 +08:00
|
|
|
static uint8_t arduino_data_mask;
|
|
|
|
static uint8_t arduino_data_n_mask;
|
|
|
|
|
2016-05-16 15:16:54 +08:00
|
|
|
|
|
|
|
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
2016-11-28 06:35:12 +08:00
|
|
|
|
|
|
|
data = (uint8_t *)arg_ptr;
|
|
|
|
if ( takeover_edge == 0 )
|
|
|
|
{
|
|
|
|
while( arg_int > 0 )
|
|
|
|
{
|
|
|
|
b = *data;
|
|
|
|
data++;
|
|
|
|
arg_int--;
|
2017-01-15 03:01:41 +08:00
|
|
|
/* issue 156, check for speed */
|
|
|
|
#if F_CPU <= 17000000
|
2016-11-28 06:35:12 +08:00
|
|
|
if ( b == 0 )
|
|
|
|
{
|
|
|
|
*arduino_data_port &= arduino_data_n_mask;
|
2016-11-28 13:02:01 +08:00
|
|
|
for( i = 0; i < 8; i++ )
|
2016-11-28 06:35:12 +08:00
|
|
|
{
|
|
|
|
*arduino_clock_port |= arduino_clock_mask;
|
|
|
|
*arduino_clock_port &= arduino_clock_n_mask;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2017-01-15 03:01:41 +08:00
|
|
|
#endif
|
2016-11-28 06:35:12 +08:00
|
|
|
{
|
|
|
|
for( i = 0; i < 8; i++ )
|
|
|
|
{
|
|
|
|
if ( b & 128 )
|
|
|
|
*arduino_data_port |= arduino_data_mask;
|
|
|
|
else
|
|
|
|
*arduino_data_port &= arduino_data_n_mask;
|
|
|
|
|
|
|
|
*arduino_clock_port |= arduino_clock_mask;
|
2016-11-29 04:17:10 +08:00
|
|
|
b <<= 1;
|
2016-11-28 06:35:12 +08:00
|
|
|
*arduino_clock_port &= arduino_clock_n_mask;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while( arg_int > 0 )
|
|
|
|
{
|
|
|
|
b = *data;
|
|
|
|
data++;
|
|
|
|
arg_int--;
|
2017-01-15 03:01:41 +08:00
|
|
|
/* issue 156, check for speed */
|
|
|
|
#if F_CPU <= 17000000
|
2016-11-28 06:35:12 +08:00
|
|
|
if ( b == 0 )
|
|
|
|
{
|
|
|
|
*arduino_data_port &= arduino_data_n_mask;
|
2016-11-28 13:02:01 +08:00
|
|
|
for( i = 0; i < 8; i++ )
|
2016-11-28 06:35:12 +08:00
|
|
|
{
|
|
|
|
*arduino_clock_port &= arduino_clock_n_mask;
|
|
|
|
*arduino_clock_port |= arduino_clock_mask;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2017-01-15 03:01:41 +08:00
|
|
|
#endif
|
2016-11-28 06:35:12 +08:00
|
|
|
{
|
|
|
|
for( i = 0; i < 8; i++ )
|
|
|
|
{
|
|
|
|
if ( b & 128 )
|
|
|
|
*arduino_data_port |= arduino_data_mask;
|
|
|
|
else
|
|
|
|
*arduino_data_port &= arduino_data_n_mask;
|
|
|
|
|
|
|
|
*arduino_clock_port &= arduino_clock_n_mask;
|
2016-11-29 04:17:10 +08:00
|
|
|
b <<= 1;
|
2016-11-28 06:35:12 +08:00
|
|
|
*arduino_clock_port |= arduino_clock_mask;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-14 12:35:03 +08:00
|
|
|
}
|
2016-05-16 15:16:54 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
/* disable chipselect */
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
/* no wait required here */
|
|
|
|
|
|
|
|
/* for SPI: setup correct level of the clock signal */
|
2016-07-27 02:38:55 +08:00
|
|
|
u8x8_gpio_SetSPIClock(u8x8, u8x8_GetSPIClockPhase(u8x8));
|
2016-05-16 15:16:54 +08:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
u8x8_gpio_SetDC(u8x8, arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_enable_level);
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->post_chip_enable_wait_ns, NULL);
|
|
|
|
|
|
|
|
/* there is no consistency checking for u8x8->pins[U8X8_PIN_SPI_CLOCK] */
|
|
|
|
|
|
|
|
arduino_clock_port = portOutputRegister(digitalPinToPort(u8x8->pins[U8X8_PIN_SPI_CLOCK]));
|
|
|
|
arduino_clock_mask = digitalPinToBitMask(u8x8->pins[U8X8_PIN_SPI_CLOCK]);
|
|
|
|
arduino_clock_n_mask = ~arduino_clock_mask;
|
2016-11-26 22:12:24 +08:00
|
|
|
|
|
|
|
|
2016-05-16 15:16:54 +08:00
|
|
|
|
|
|
|
/* there is no consistency checking for u8x8->pins[U8X8_PIN_SPI_DATA] */
|
|
|
|
|
|
|
|
arduino_data_port = portOutputRegister(digitalPinToPort(u8x8->pins[U8X8_PIN_SPI_DATA]));
|
|
|
|
arduino_data_mask = digitalPinToBitMask(u8x8->pins[U8X8_PIN_SPI_DATA]);
|
|
|
|
arduino_data_n_mask = ~arduino_data_mask;
|
2017-02-05 18:25:47 +08:00
|
|
|
|
2016-05-16 15:16:54 +08:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->pre_chip_disable_wait_ns, NULL);
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
/* fallback */
|
|
|
|
uint8_t u8x8_byte_arduino_4wire_sw_spi(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_4wire_sw_spi(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/*=============================================*/
|
|
|
|
|
2017-08-23 16:32:28 +08:00
|
|
|
#ifdef U8X8_USE_PINS
|
2016-05-16 15:16:54 +08:00
|
|
|
|
2016-07-27 17:44:17 +08:00
|
|
|
extern "C" uint8_t u8x8_byte_arduino_hw_spi(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
2015-12-13 00:26:10 +08:00
|
|
|
{
|
2016-04-29 04:05:15 +08:00
|
|
|
#ifdef U8X8_HAVE_HW_SPI
|
2015-12-13 00:26:10 +08:00
|
|
|
uint8_t *data;
|
2016-08-21 17:46:01 +08:00
|
|
|
uint8_t internal_spi_mode;
|
2015-12-13 00:26:10 +08:00
|
|
|
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
2016-12-31 18:12:25 +08:00
|
|
|
|
|
|
|
// 1.6.5 offers a block transfer, but the problem is, that the
|
|
|
|
// buffer is overwritten with the incoming data
|
|
|
|
// so it can not be used...
|
|
|
|
// SPI.transfer((uint8_t *)arg_ptr, arg_int);
|
|
|
|
|
2015-12-13 00:26:10 +08:00
|
|
|
data = (uint8_t *)arg_ptr;
|
|
|
|
while( arg_int > 0 )
|
|
|
|
{
|
|
|
|
SPI.transfer((uint8_t)*data);
|
|
|
|
data++;
|
|
|
|
arg_int--;
|
|
|
|
}
|
2016-12-31 18:12:25 +08:00
|
|
|
|
2015-12-13 00:26:10 +08:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
/* disable chipselect */
|
2016-07-27 17:44:17 +08:00
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
2015-12-13 00:26:10 +08:00
|
|
|
/* no wait required here */
|
|
|
|
|
|
|
|
/* for SPI: setup correct level of the clock signal */
|
2016-08-18 03:01:03 +08:00
|
|
|
// removed, use SPI.begin() instead: pinMode(11, OUTPUT);
|
|
|
|
// removed, use SPI.begin() instead: pinMode(13, OUTPUT);
|
|
|
|
// removed, use SPI.begin() instead: digitalWrite(13, u8x8_GetSPIClockPhase(u8x8));
|
|
|
|
|
|
|
|
/* setup hardware with SPI.begin() instead of previous digitalWrite() and pinMode() calls */
|
|
|
|
SPI.begin();
|
|
|
|
|
2015-12-13 00:26:10 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
2016-07-27 17:44:17 +08:00
|
|
|
u8x8_gpio_SetDC(u8x8, arg_int);
|
2015-12-13 00:26:10 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
2016-08-21 17:46:01 +08:00
|
|
|
/* SPI mode has to be mapped to the mode of the current controller, at least Uno, Due, 101 have different SPI_MODEx values */
|
|
|
|
internal_spi_mode = 0;
|
|
|
|
switch(u8x8->display_info->spi_mode)
|
|
|
|
{
|
|
|
|
case 0: internal_spi_mode = SPI_MODE0; break;
|
|
|
|
case 1: internal_spi_mode = SPI_MODE1; break;
|
|
|
|
case 2: internal_spi_mode = SPI_MODE2; break;
|
|
|
|
case 3: internal_spi_mode = SPI_MODE3; break;
|
|
|
|
}
|
|
|
|
|
2016-08-18 03:01:03 +08:00
|
|
|
#if ARDUINO >= 10600
|
2016-08-21 17:46:01 +08:00
|
|
|
SPI.beginTransaction(SPISettings(u8x8->display_info->sck_clock_hz, MSBFIRST, internal_spi_mode));
|
2016-08-18 03:01:03 +08:00
|
|
|
#else
|
2015-12-13 00:26:10 +08:00
|
|
|
SPI.begin();
|
|
|
|
|
2016-07-27 17:44:17 +08:00
|
|
|
if ( u8x8->display_info->sck_pulse_width_ns < 70 )
|
2015-12-13 00:26:10 +08:00
|
|
|
SPI.setClockDivider( SPI_CLOCK_DIV2 );
|
2016-07-27 17:44:17 +08:00
|
|
|
else if ( u8x8->display_info->sck_pulse_width_ns < 140 )
|
2015-12-13 00:26:10 +08:00
|
|
|
SPI.setClockDivider( SPI_CLOCK_DIV4 );
|
2016-06-25 03:36:32 +08:00
|
|
|
else
|
|
|
|
SPI.setClockDivider( SPI_CLOCK_DIV8 );
|
2016-08-21 17:46:01 +08:00
|
|
|
SPI.setDataMode(internal_spi_mode);
|
2015-12-13 00:26:10 +08:00
|
|
|
SPI.setBitOrder(MSBFIRST);
|
2016-08-18 03:01:03 +08:00
|
|
|
#endif
|
2015-12-13 00:26:10 +08:00
|
|
|
|
2016-07-27 17:44:17 +08:00
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_enable_level);
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->post_chip_enable_wait_ns, NULL);
|
2015-12-13 00:26:10 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
2016-07-27 17:44:17 +08:00
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->pre_chip_disable_wait_ns, NULL);
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
2015-12-13 00:26:10 +08:00
|
|
|
|
2016-08-18 03:01:03 +08:00
|
|
|
#if ARDUINO >= 10600
|
|
|
|
SPI.endTransaction();
|
|
|
|
#else
|
2015-12-13 00:26:10 +08:00
|
|
|
SPI.end();
|
2016-08-18 03:01:03 +08:00
|
|
|
#endif
|
2015-12-13 00:26:10 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
2016-08-21 17:46:01 +08:00
|
|
|
|
|
|
|
#else
|
2016-01-02 07:25:06 +08:00
|
|
|
#endif
|
2015-12-13 00:26:10 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-12-07 05:48:57 +08:00
|
|
|
|
2017-05-20 13:59:48 +08:00
|
|
|
/* issue #244 */
|
2017-08-12 02:30:39 +08:00
|
|
|
extern "C" uint8_t u8x8_byte_arduino_2nd_hw_spi(U8X8_UNUSED u8x8_t *u8x8, U8X8_UNUSED uint8_t msg, U8X8_UNUSED uint8_t arg_int, U8X8_UNUSED void *arg_ptr)
|
2017-05-20 13:59:48 +08:00
|
|
|
{
|
|
|
|
#ifdef U8X8_HAVE_2ND_HW_SPI
|
|
|
|
uint8_t *data;
|
|
|
|
uint8_t internal_spi_mode;
|
|
|
|
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
|
|
|
|
|
|
|
// 1.6.5 offers a block transfer, but the problem is, that the
|
|
|
|
// buffer is overwritten with the incoming data
|
|
|
|
// so it can not be used...
|
|
|
|
// SPI.transfer((uint8_t *)arg_ptr, arg_int);
|
|
|
|
|
|
|
|
data = (uint8_t *)arg_ptr;
|
|
|
|
while( arg_int > 0 )
|
|
|
|
{
|
|
|
|
SPI1.transfer((uint8_t)*data);
|
|
|
|
data++;
|
|
|
|
arg_int--;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
/* disable chipselect */
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
/* no wait required here */
|
|
|
|
|
|
|
|
/* for SPI1: setup correct level of the clock signal */
|
|
|
|
// removed, use SPI.begin() instead: pinMode(11, OUTPUT);
|
|
|
|
// removed, use SPI.begin() instead: pinMode(13, OUTPUT);
|
|
|
|
// removed, use SPI.begin() instead: digitalWrite(13, u8x8_GetSPIClockPhase(u8x8));
|
|
|
|
|
|
|
|
/* setup hardware with SPI.begin() instead of previous digitalWrite() and pinMode() calls */
|
|
|
|
SPI1.begin();
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
u8x8_gpio_SetDC(u8x8, arg_int);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
/* SPI1 mode has to be mapped to the mode of the current controller, at least Uno, Due, 101 have different SPI_MODEx values */
|
|
|
|
internal_spi_mode = 0;
|
|
|
|
switch(u8x8->display_info->spi_mode)
|
|
|
|
{
|
|
|
|
case 0: internal_spi_mode = SPI_MODE0; break;
|
|
|
|
case 1: internal_spi_mode = SPI_MODE1; break;
|
|
|
|
case 2: internal_spi_mode = SPI_MODE2; break;
|
|
|
|
case 3: internal_spi_mode = SPI_MODE3; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ARDUINO >= 10600
|
|
|
|
SPI1.beginTransaction(SPISettings(u8x8->display_info->sck_clock_hz, MSBFIRST, internal_spi_mode));
|
|
|
|
#else
|
|
|
|
SPI1.begin();
|
|
|
|
|
|
|
|
if ( u8x8->display_info->sck_pulse_width_ns < 70 )
|
|
|
|
SPI1.setClockDivider( SPI_CLOCK_DIV2 );
|
|
|
|
else if ( u8x8->display_info->sck_pulse_width_ns < 140 )
|
|
|
|
SPI1.setClockDivider( SPI_CLOCK_DIV4 );
|
|
|
|
else
|
|
|
|
SPI1.setClockDivider( SPI_CLOCK_DIV8 );
|
|
|
|
SPI1.setDataMode(internal_spi_mode);
|
|
|
|
SPI1.setBitOrder(MSBFIRST);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_enable_level);
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->post_chip_enable_wait_ns, NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->pre_chip_disable_wait_ns, NULL);
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
|
|
|
|
#if ARDUINO >= 10600
|
|
|
|
SPI1.endTransaction();
|
|
|
|
#else
|
|
|
|
SPI1.end();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
#endif
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-15 06:05:41 +08:00
|
|
|
/*=============================================*/
|
|
|
|
|
2016-11-20 21:57:53 +08:00
|
|
|
extern "C" uint8_t u8x8_byte_arduino_hw_i2c(U8X8_UNUSED u8x8_t *u8x8, U8X8_UNUSED uint8_t msg, U8X8_UNUSED uint8_t arg_int, U8X8_UNUSED void *arg_ptr)
|
2016-05-15 06:05:41 +08:00
|
|
|
{
|
2016-05-15 15:56:06 +08:00
|
|
|
#ifdef U8X8_HAVE_HW_I2C
|
2016-05-15 06:05:41 +08:00
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
2016-05-15 15:56:06 +08:00
|
|
|
Wire.write((uint8_t *)arg_ptr, (int)arg_int);
|
2016-05-15 06:05:41 +08:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
2017-10-07 04:17:27 +08:00
|
|
|
#if defined(ESP8266) || defined(ARDUINO_ARCH_ESP8266) || defined(ESP_PLATFORM) || defined(ARDUINO_ARCH_ESP32)
|
|
|
|
/* for ESP8266/ESP32, Wire.begin has two more arguments: clock and data */
|
2017-02-05 18:25:47 +08:00
|
|
|
if ( u8x8->pins[U8X8_PIN_I2C_CLOCK] != U8X8_PIN_NONE && u8x8->pins[U8X8_PIN_I2C_DATA] != U8X8_PIN_NONE )
|
|
|
|
{
|
|
|
|
// second argument for the wire lib is the clock pin. In u8g2, the first argument of the clock pin in the clock/data pair
|
|
|
|
Wire.begin(u8x8->pins[U8X8_PIN_I2C_DATA] , u8x8->pins[U8X8_PIN_I2C_CLOCK]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Wire.begin();
|
|
|
|
}
|
|
|
|
#else
|
2016-05-15 06:05:41 +08:00
|
|
|
Wire.begin();
|
2017-02-05 18:25:47 +08:00
|
|
|
#endif
|
2016-05-15 06:05:41 +08:00
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
2016-11-02 22:25:29 +08:00
|
|
|
#if ARDUINO >= 10600
|
|
|
|
/* not sure when the setClock function was introduced, but it is there since 1.6.0 */
|
|
|
|
/* if there is any error with Wire.setClock() just remove this function call */
|
|
|
|
if ( u8x8->display_info->i2c_bus_clock_100kHz >= 4 )
|
|
|
|
{
|
2016-11-03 00:08:04 +08:00
|
|
|
Wire.setClock(400000L);
|
2016-11-02 22:25:29 +08:00
|
|
|
}
|
|
|
|
#endif
|
2016-05-15 06:05:41 +08:00
|
|
|
Wire.beginTransmission(u8x8_GetI2CAddress(u8x8)>>1);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
Wire.endTransmission();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-11-20 21:57:53 +08:00
|
|
|
extern "C" uint8_t u8x8_byte_arduino_2nd_hw_i2c(U8X8_UNUSED u8x8_t *u8x8, U8X8_UNUSED uint8_t msg, U8X8_UNUSED uint8_t arg_int, U8X8_UNUSED void *arg_ptr)
|
2016-11-13 00:30:14 +08:00
|
|
|
{
|
|
|
|
#ifdef U8X8_HAVE_2ND_HW_I2C
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
|
|
|
Wire1.write((uint8_t *)arg_ptr, (int)arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
Wire1.begin();
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
#if ARDUINO >= 10600
|
|
|
|
/* not sure when the setClock function was introduced, but it is there since 1.6.0 */
|
|
|
|
/* if there is any error with Wire.setClock() just remove this function call */
|
|
|
|
if ( u8x8->display_info->i2c_bus_clock_100kHz >= 4 )
|
|
|
|
{
|
|
|
|
Wire1.setClock(400000L);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
Wire1.beginTransmission(u8x8_GetI2CAddress(u8x8)>>1);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
Wire1.endTransmission();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-08-23 16:32:28 +08:00
|
|
|
#endif // U8X8_USE_PINS
|
|
|
|
|
2016-11-26 15:40:58 +08:00
|
|
|
/*=============================================*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
replacement for a more faster u8x8_byte_8bit_8080mode
|
|
|
|
in general u8x8_byte_8bit_8080mode could be a fallback:
|
|
|
|
|
|
|
|
uint8_t u8x8_byte_arduino_8bit_8080mode(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_8bit_8080mode(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __AVR_ARCH__
|
|
|
|
#define __AVR_ARCH__ 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(U8X8_USE_PINS)
|
|
|
|
/* no pin information (very strange), so fallback */
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_8bit_8080mode(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_8bit_8080mode(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif __AVR_ARCH__ == 4 || __AVR_ARCH__ == 5 || __AVR_ARCH__ == 51 || __AVR_ARCH__ == 6
|
|
|
|
|
|
|
|
/* this function completly replaces u8x8_byte_8bit_8080mode*/
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_8bit_8080mode(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
uint8_t i, b;
|
|
|
|
uint8_t *data;
|
|
|
|
|
|
|
|
/* the following static vars are recalculated in U8X8_MSG_BYTE_START_TRANSFER */
|
2016-12-14 12:35:03 +08:00
|
|
|
/* so, it should be possible to use multiple displays with different pins */
|
2016-11-26 15:40:58 +08:00
|
|
|
|
|
|
|
static volatile uint8_t *arduino_e_port;
|
|
|
|
static volatile uint8_t arduino_e_mask;
|
|
|
|
static volatile uint8_t arduino_e_n_mask;
|
|
|
|
|
|
|
|
static volatile uint8_t *arduino_data_port[8];
|
|
|
|
static volatile uint8_t arduino_data_mask[8];
|
|
|
|
static volatile uint8_t arduino_data_n_mask[8];
|
|
|
|
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
|
|
|
data = (uint8_t *)arg_ptr;
|
|
|
|
while( arg_int > 0 )
|
|
|
|
{
|
|
|
|
b = *data;
|
|
|
|
data++;
|
|
|
|
arg_int--;
|
|
|
|
for( i = 0; i < 8; i++ )
|
|
|
|
{
|
|
|
|
if ( b & 1 )
|
|
|
|
*arduino_data_port[i] |= arduino_data_mask[i];
|
|
|
|
else
|
|
|
|
*arduino_data_port[i] &= arduino_data_n_mask[i];
|
|
|
|
b >>= 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
*arduino_e_port &= arduino_e_n_mask;
|
|
|
|
|
|
|
|
|
|
|
|
/* AVR Architecture is very slow, extra call is not required */
|
2016-12-14 12:35:03 +08:00
|
|
|
//u8x8_gpio_Delay(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->sda_setup_time_ns);
|
|
|
|
u8x8_gpio_Delay(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->data_setup_time_ns);
|
2016-11-26 15:40:58 +08:00
|
|
|
|
|
|
|
*arduino_e_port |= arduino_e_mask;
|
|
|
|
|
|
|
|
/* AVR Architecture is very slow, extra call is not required */
|
2016-12-14 12:35:03 +08:00
|
|
|
//u8x8_gpio_Delay(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->sck_pulse_width_ns);
|
|
|
|
u8x8_gpio_Delay(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->write_pulse_width_ns);
|
2016-11-26 15:40:58 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
/* disable chipselect */
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
/* no wait required here */
|
|
|
|
|
|
|
|
/* ensure that the enable signal is high */
|
|
|
|
u8x8_gpio_call(u8x8, U8X8_MSG_GPIO_E, 1);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
u8x8_gpio_SetDC(u8x8, arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_enable_level);
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->post_chip_enable_wait_ns, NULL);
|
|
|
|
|
|
|
|
/* there is no consistency checking for u8x8->pins[U8X8_PIN_E] */
|
|
|
|
|
|
|
|
arduino_e_port = portOutputRegister(digitalPinToPort(u8x8->pins[U8X8_PIN_E]));
|
|
|
|
arduino_e_mask = digitalPinToBitMask(u8x8->pins[U8X8_PIN_E]);
|
|
|
|
arduino_e_n_mask = ~arduino_e_mask;
|
|
|
|
|
|
|
|
/* there is no consistency checking for u8x8->pins[U8X8_PIN_D0] */
|
|
|
|
|
|
|
|
for( i = 0; i < 8; i++ )
|
|
|
|
{
|
|
|
|
arduino_data_port[i] = portOutputRegister(digitalPinToPort(u8x8->pins[U8X8_PIN_D0+i]));
|
|
|
|
arduino_data_mask[i] = digitalPinToBitMask(u8x8->pins[U8X8_PIN_D0+i]);
|
|
|
|
arduino_data_n_mask[i] = ~arduino_data_mask[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->pre_chip_disable_wait_ns, NULL);
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
/* fallback */
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_8bit_8080mode(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_8bit_8080mode(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2016-12-14 12:35:03 +08:00
|
|
|
/*=============================================*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
replacement for a more faster u8x8_byte_ks0108
|
|
|
|
in general u8x8_byte_ks0108 could be a fallback:
|
|
|
|
|
|
|
|
uint8_t u8x8_byte_arduino_ks0108(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_ks0108(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __AVR_ARCH__
|
|
|
|
#define __AVR_ARCH__ 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(U8X8_USE_PINS)
|
|
|
|
/* no pin information (very strange), so fallback */
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_ks0108(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_ks0108(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif __AVR_ARCH__ == 4 || __AVR_ARCH__ == 5 || __AVR_ARCH__ == 51 || __AVR_ARCH__ == 6
|
|
|
|
|
|
|
|
/* this function completly replaces u8x8_byte_ks0108*/
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_ks0108(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
uint8_t i, b;
|
|
|
|
uint8_t *data;
|
|
|
|
|
|
|
|
/* the following static vars are recalculated in U8X8_MSG_BYTE_START_TRANSFER */
|
|
|
|
/* so, it should be possible to use multiple displays with different pins */
|
|
|
|
|
|
|
|
static volatile uint8_t *arduino_e_port;
|
|
|
|
static volatile uint8_t arduino_e_mask;
|
|
|
|
static volatile uint8_t arduino_e_n_mask;
|
|
|
|
|
|
|
|
static volatile uint8_t *arduino_data_port[8];
|
|
|
|
static volatile uint8_t arduino_data_mask[8];
|
|
|
|
static volatile uint8_t arduino_data_n_mask[8];
|
|
|
|
|
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case U8X8_MSG_BYTE_SEND:
|
|
|
|
data = (uint8_t *)arg_ptr;
|
|
|
|
while( arg_int > 0 )
|
|
|
|
{
|
|
|
|
b = *data;
|
|
|
|
data++;
|
|
|
|
arg_int--;
|
|
|
|
for( i = 0; i < 8; i++ )
|
|
|
|
{
|
|
|
|
if ( b & 1 )
|
|
|
|
*arduino_data_port[i] |= arduino_data_mask[i];
|
|
|
|
else
|
|
|
|
*arduino_data_port[i] &= arduino_data_n_mask[i];
|
|
|
|
b >>= 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
*arduino_e_port |= arduino_e_mask;
|
|
|
|
|
|
|
|
|
|
|
|
/* AVR Architecture is very slow, extra call is not required */
|
|
|
|
u8x8_gpio_Delay(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->data_setup_time_ns);
|
|
|
|
|
|
|
|
*arduino_e_port &= arduino_e_n_mask;
|
|
|
|
|
|
|
|
/* AVR Architecture is very slow, extra call is not required */
|
|
|
|
u8x8_gpio_Delay(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->write_pulse_width_ns);
|
|
|
|
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case U8X8_MSG_BYTE_INIT:
|
|
|
|
/* disable chipselect */
|
|
|
|
u8x8_gpio_SetCS(u8x8, u8x8->display_info->chip_disable_level);
|
|
|
|
/* no wait required here */
|
|
|
|
|
|
|
|
/* ensure that the enable signal is low */
|
|
|
|
u8x8_gpio_call(u8x8, U8X8_MSG_GPIO_E, 0);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_SET_DC:
|
|
|
|
u8x8_gpio_SetDC(u8x8, arg_int);
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_START_TRANSFER:
|
|
|
|
u8x8_byte_set_ks0108_cs(u8x8, arg_int);
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->post_chip_enable_wait_ns, NULL);
|
|
|
|
|
|
|
|
/* there is no consistency checking for u8x8->pins[U8X8_PIN_E] */
|
|
|
|
|
|
|
|
arduino_e_port = portOutputRegister(digitalPinToPort(u8x8->pins[U8X8_PIN_E]));
|
|
|
|
arduino_e_mask = digitalPinToBitMask(u8x8->pins[U8X8_PIN_E]);
|
|
|
|
arduino_e_n_mask = ~arduino_e_mask;
|
|
|
|
|
|
|
|
/* there is no consistency checking for u8x8->pins[U8X8_PIN_D0] */
|
|
|
|
|
|
|
|
for( i = 0; i < 8; i++ )
|
|
|
|
{
|
|
|
|
arduino_data_port[i] = portOutputRegister(digitalPinToPort(u8x8->pins[U8X8_PIN_D0+i]));
|
|
|
|
arduino_data_mask[i] = digitalPinToBitMask(u8x8->pins[U8X8_PIN_D0+i]);
|
|
|
|
arduino_data_n_mask[i] = ~arduino_data_mask[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case U8X8_MSG_BYTE_END_TRANSFER:
|
|
|
|
u8x8->gpio_and_delay_cb(u8x8, U8X8_MSG_DELAY_NANO, u8x8->display_info->pre_chip_disable_wait_ns, NULL);
|
|
|
|
u8x8_byte_set_ks0108_cs(u8x8, arg_int);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
/* fallback */
|
|
|
|
extern "C" uint8_t u8x8_byte_arduino_ks0108(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
|
|
|
|
{
|
|
|
|
return u8x8_byte_ks0108(u8x8, msg,arg_int, arg_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-11-26 15:40:58 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-08-23 16:32:28 +08:00
|
|
|
#ifdef U8X8_USE_PINS
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
use U8X8_PIN_NONE as value for "reset", if there is no reset line
|
|
|
|
*/
|
2015-12-23 16:04:24 +08:00
|
|
|
|
|
|
|
void u8x8_SetPin_4Wire_SW_SPI(u8x8_t *u8x8, uint8_t clock, uint8_t data, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_DATA, data);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
|
|
|
|
2016-05-01 16:01:37 +08:00
|
|
|
#ifdef _obsolete_com_specific_setup
|
2015-12-09 04:30:14 +08:00
|
|
|
void u8x8_Setup_4Wire_SW_SPI(u8x8_t *u8x8, u8x8_msg_cb display_cb, uint8_t clock, uint8_t data, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
2015-12-13 00:26:10 +08:00
|
|
|
u8x8_Setup(u8x8, display_cb, u8x8_cad_001, u8x8_byte_4wire_sw_spi, u8x8_gpio_and_delay_arduino);
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/* assign individual pin values (only for ARDUINO, if pin_list is available) */
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_DATA, data);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2016-05-01 16:01:37 +08:00
|
|
|
#endif /* obsolete com specific setup */
|
2015-12-09 04:30:14 +08:00
|
|
|
|
2015-12-23 16:04:24 +08:00
|
|
|
void u8x8_SetPin_3Wire_SW_SPI(u8x8_t *u8x8, uint8_t clock, uint8_t data, uint8_t cs, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_DATA, data);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2015-12-09 04:30:14 +08:00
|
|
|
|
2016-05-01 16:01:37 +08:00
|
|
|
#ifdef _obsolete_com_specific_setup
|
2015-12-09 04:30:14 +08:00
|
|
|
void u8x8_Setup_3Wire_SW_SPI(u8x8_t *u8x8, u8x8_msg_cb display_cb, uint8_t clock, uint8_t data, uint8_t cs, uint8_t reset)
|
|
|
|
{
|
2015-12-13 00:26:10 +08:00
|
|
|
u8x8_Setup(u8x8, display_cb, u8x8_cad_001, u8x8_byte_3wire_sw_spi, u8x8_gpio_and_delay_arduino);
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/* assign individual pin values (only for ARDUINO, if pin_list is available) */
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_SPI_DATA, data);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2016-05-01 16:01:37 +08:00
|
|
|
#endif /* obsolete com specific setup */
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
use U8X8_PIN_NONE as value for "reset", if there is no reset line
|
|
|
|
*/
|
2015-12-23 16:04:24 +08:00
|
|
|
void u8x8_SetPin_4Wire_HW_SPI(u8x8_t *u8x8, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
|
|
|
|
2016-06-24 06:58:17 +08:00
|
|
|
void u8x8_SetPin_ST7920_HW_SPI(u8x8_t *u8x8, uint8_t cs, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
|
|
|
|
2015-12-23 16:04:24 +08:00
|
|
|
|
2016-05-01 16:01:37 +08:00
|
|
|
#ifdef _obsolete_com_specific_setup
|
2015-12-09 04:30:14 +08:00
|
|
|
void u8x8_Setup_4Wire_HW_SPI(u8x8_t *u8x8, u8x8_msg_cb display_cb, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
2015-12-13 00:26:10 +08:00
|
|
|
u8x8_Setup(u8x8, display_cb, u8x8_cad_001, u8x8_byte_arduino_hw_spi, u8x8_gpio_and_delay_arduino);
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/* assign individual pin values (only for ARDUINO, if pin_list is available) */
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2016-05-01 16:01:37 +08:00
|
|
|
#endif /* obsolete com specific setup */
|
|
|
|
|
2015-12-09 04:30:14 +08:00
|
|
|
|
2016-06-23 05:58:14 +08:00
|
|
|
void u8x8_SetPin_SW_I2C(u8x8_t *u8x8, uint8_t clock, uint8_t data, uint8_t reset)
|
2015-12-23 16:04:24 +08:00
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_I2C_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_I2C_DATA, data);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
|
|
|
|
2016-05-01 16:01:37 +08:00
|
|
|
#ifdef _obsolete_com_specific_setup
|
2015-12-09 04:30:14 +08:00
|
|
|
void u8x8_Setup_SSD13xx_SW_I2C(u8x8_t *u8x8, u8x8_msg_cb display_cb, uint8_t clock, uint8_t data, uint8_t reset)
|
|
|
|
{
|
2015-12-13 00:26:10 +08:00
|
|
|
u8x8_Setup(u8x8, display_cb, u8x8_cad_001, u8x8_byte_ssd13xx_sw_i2c, u8x8_gpio_and_delay_arduino);
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/* assign individual pin values (only for ARDUINO, if pin_list is available) */
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_I2C_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_I2C_DATA, data);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2016-05-01 16:01:37 +08:00
|
|
|
#endif /* obsolete com specific setup */
|
2015-12-09 04:30:14 +08:00
|
|
|
|
2017-02-05 18:25:47 +08:00
|
|
|
void u8x8_SetPin_HW_I2C(u8x8_t *u8x8, uint8_t reset, uint8_t clock, uint8_t data)
|
2016-05-15 06:05:41 +08:00
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
2017-02-05 18:25:47 +08:00
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_I2C_CLOCK, clock);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_I2C_DATA, data);
|
2016-05-15 06:05:41 +08:00
|
|
|
}
|
|
|
|
|
2015-12-23 16:04:24 +08:00
|
|
|
void u8x8_SetPin_8Bit_6800(u8x8_t *u8x8, uint8_t d0, uint8_t d1, uint8_t d2, uint8_t d3, uint8_t d4, uint8_t d5, uint8_t d6, uint8_t d7, uint8_t enable, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D0, d0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D1, d1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D2, d2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D3, d3);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D4, d4);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D5, d5);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D6, d6);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D7, d7);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_E, enable);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
|
|
|
|
2016-05-01 16:01:37 +08:00
|
|
|
#ifdef _obsolete_com_specific_setup
|
2015-12-09 04:30:14 +08:00
|
|
|
void u8x8_Setup_8Bit_6800(u8x8_t *u8x8, u8x8_msg_cb display_cb, uint8_t d0, uint8_t d1, uint8_t d2, uint8_t d3, uint8_t d4, uint8_t d5, uint8_t d6, uint8_t d7, uint8_t enable, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
2015-12-13 00:26:10 +08:00
|
|
|
u8x8_Setup(u8x8, display_cb, u8x8_cad_001, u8x8_byte_8bit_6800mode, u8x8_gpio_and_delay_arduino);
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/* assign individual pin values (only for ARDUINO, if pin_list is available) */
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D0, d0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D1, d1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D2, d2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D3, d3);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D4, d4);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D5, d5);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D6, d6);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D7, d7);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_E, enable);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2016-05-01 16:01:37 +08:00
|
|
|
#endif /* obsolete com specific setup */
|
|
|
|
|
2015-12-09 04:30:14 +08:00
|
|
|
|
2015-12-23 16:04:24 +08:00
|
|
|
void u8x8_SetPin_8Bit_8080(u8x8_t *u8x8, uint8_t d0, uint8_t d1, uint8_t d2, uint8_t d3, uint8_t d4, uint8_t d5, uint8_t d6, uint8_t d7, uint8_t wr, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D0, d0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D1, d1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D2, d2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D3, d3);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D4, d4);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D5, d5);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D6, d6);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D7, d7);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_E, wr);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
|
|
|
|
2016-05-01 16:01:37 +08:00
|
|
|
|
|
|
|
#ifdef _obsolete_com_specific_setup
|
2015-12-09 04:30:14 +08:00
|
|
|
void u8x8_Setup_8Bit_8080(u8x8_t *u8x8, u8x8_msg_cb display_cb, uint8_t d0, uint8_t d1, uint8_t d2, uint8_t d3, uint8_t d4, uint8_t d5, uint8_t d6, uint8_t d7, uint8_t wr, uint8_t cs, uint8_t dc, uint8_t reset)
|
|
|
|
{
|
2015-12-13 00:26:10 +08:00
|
|
|
u8x8_Setup(u8x8, display_cb, u8x8_cad_001, u8x8_byte_8bit_8080mode, u8x8_gpio_and_delay_arduino);
|
2015-12-09 04:30:14 +08:00
|
|
|
|
|
|
|
/* assign individual pin values (only for ARDUINO, if pin_list is available) */
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D0, d0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D1, d1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D2, d2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D3, d3);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D4, d4);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D5, d5);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D6, d6);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D7, d7);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_E, wr);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2016-05-01 16:01:37 +08:00
|
|
|
#endif /* obsolete com specific setup */
|
|
|
|
|
2016-12-05 05:48:27 +08:00
|
|
|
void u8x8_SetPin_KS0108(u8x8_t *u8x8, uint8_t d0, uint8_t d1, uint8_t d2, uint8_t d3, uint8_t d4, uint8_t d5, uint8_t d6, uint8_t d7, uint8_t enable, uint8_t dc, uint8_t cs0, uint8_t cs1, uint8_t cs2, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D0, d0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D1, d1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D2, d2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D3, d3);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D4, d4);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D5, d5);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D6, d6);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D7, d7);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_E, enable);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, cs0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS1, cs1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS2, cs2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
|
|
|
}
|
2017-08-10 03:09:21 +08:00
|
|
|
|
|
|
|
void u8x8_SetPin_SED1520(u8x8_t *u8x8, uint8_t d0, uint8_t d1, uint8_t d2, uint8_t d3, uint8_t d4, uint8_t d5, uint8_t d6, uint8_t d7, uint8_t dc, uint8_t e1, uint8_t e2, uint8_t reset)
|
|
|
|
{
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D0, d0);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D1, d1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D2, d2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D3, d3);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D4, d4);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D5, d5);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D6, d6);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_D7, d7);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_E, e1);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_CS, e2);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_DC, dc);
|
|
|
|
u8x8_SetPin(u8x8, U8X8_PIN_RESET, reset);
|
2017-08-23 16:32:28 +08:00
|
|
|
}
|
|
|
|
#endif // U8X8_USE_PINS
|