MMDVMHost-Private/I2CPi.cpp

213 lines
4.7 KiB
C++

/*
* Copyright (C) 2020 by Jonathan Naylor G4KLX
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "I2CPi.h"
#include "Thread.h"
#include <bcm2835.h>
const unsigned char OLED_ADAFRUIT_SPI_128x32 = 0U;
const unsigned char OLED_ADAFRUIT_SPI_128x64 = 1U;
CI2CPi::CI2CPi() :
m_spi(false)
{
}
CI2CPi::~CI2CPi()
{
}
bool CI2CPi::open(unsigned char displayType)
{
if (displayType == OLED_ADAFRUIT_SPI_128x32 ||
displayType == OLED_ADAFRUIT_SPI_128x64) {
m_spi = true;
// Init & Configure Raspberry PI SPI
bcm2835_spi_begin(OLED_SPI_CS);
bcm2835_spi_setBitOrder(BCM2835_SPI_BIT_ORDER_MSBFIRST);
bcm2835_spi_setDataMode(BCM2835_SPI_MODE0);
// 16 MHz SPI bus, but Worked at 62 MHz also
bcm2835_spi_setClockDivider(BCM2835_SPI_CLOCK_DIVIDER_16);
// Set the pin that will control DC as output
bcm2835_gpio_fsel(OLED_SPI_DC, BCM2835_GPIO_FSEL_OUTP);
// Setup reset pin direction as output
bcm2835_gpio_fsel(OLED_SPI_RESET, BCM2835_GPIO_FSEL_OUTP);
// Setup reset pin direction (used by both SPI and I2C)
bcm2835_gpio_fsel(OLED_SPI_RESET, BCM2835_GPIO_FSEL_OUTP);
bcm2835_gpio_write(OLED_SPI_RESET, HIGH);
// VDD (3.3V) goes high at start, lets just chill for a ms
CThread::sleep(1U);
// bring reset low
bcm2835_gpio_write(OLED_SPI_RESET, LOW);
// wait 10ms
CThread::sleep(10U);
// bring out of reset
bcm2835_gpio_write(OLED_SPI_RESET, HIGH);
} else {
m_spi = false;
// Init Raspberry PI GPIO
if (!bcm2835_init())
return false;
// Init & Configure Raspberry PI I2C
if (!bcm2835_i2c_begin())
return false;
// Setup reset pin direction as output
bcm2835_gpio_fsel(OLED_I2C_RESET, BCM2835_GPIO_FSEL_OUTP);
// Setup reset pin direction (used by both SPI and I2C)
bcm2835_gpio_fsel(OLED_I2C_RESET, BCM2835_GPIO_FSEL_OUTP);
bcm2835_gpio_write(OLED_I2C_RESET, HIGH);
// VDD (3.3V) goes high at start, lets just chill for a ms
CThread::sleep(1U);
// bring reset low
bcm2835_gpio_write(OLED_I2C_RESET, LOW);
// wait 10ms
CThread::sleep(10U);
// bring out of reset
bcm2835_gpio_write(OLED_I2C_RESET, HIGH);
}
return true;
}
void CI2CPi::setAddress(unsigned char address)
{
if (!m_spi)
bcm2835_i2c_setSlaveAddress(address);
}
void CI2CPi::setDataMode()
{
if (m_spi) {
// Setup D/C line to high to switch to data mode
bcm2835_gpio_write(OLED_SPI_DC, HIGH);
}
}
void CI2CPi::sendCommand(uint8_t c0, uint8_t c1, uint8_t c2)
{
uint8_t buff[4U];
buff[1U] = c0;
buff[2U] = c1;
buff[3U] = c2;
// Is SPI
if (m_spi) {
// Setup D/C line to low to switch to command mode
bcm2835_gpio_write(OLED_SPI_DC, LOW);
// Write Data
bcm2835_spi_writenb(buff + 1U, 3U);
} else {
// Clear D/C to switch to command mode
buff[0U] = SSD_Command_Mode;
// Write Data on I2C
bcm2835_i2c_write(buff, 4U);
}
}
void CI2CPi::sendCommand(uint8_t c0, uint8_t c1)
{
uint8_t buff[3U];
buff[1U] = c0;
buff[2U] = c1;
// Is SPI
if (m_spi) {
// Setup D/C line to low to switch to command mode
bcm2835_gpio_write(OLED_SPI_DC, LOW);
// Write Data
bcm2835_spi_writenb(buff + 1U, 2U);
} else {
// Clear D/C to switch to command mode
buff[0U] = SSD_Command_Mode;
// Write Data on I2C
bcm2835_i2c_write(buff, 3U);
}
}
void CI2CPi::sendCommand(uint8_t c)
{
// Is SPI
if (m_spi) {
// Setup D/C line to low to switch to command mode
bcm2835_gpio_write(OLED_SPI_DC, LOW);
// Write Data on SPI
bcm2835_spi_transfer(c);
} else {
uint8_t buff[2U];
// Clear D/C to switch to command mode
buff[0] = SSD_Command_Mode;
buff[1] = c;
// Write Data on I2C
bcm2835_i2c_write(buff, 2U);
}
}
void CI2CPi::writeData(const uint8_t* data, uint16_t length)
{
if (m_spi)
bcm2835_spi_writenb(data, length);
else
bcm2835_i2c_write(data, length);
}
void CI2CPi::writeData(uint8_t c)
{
if (m_spi)
bcm2835_spi_transfer(c);
else
bcm2835_spi_transfer(c); // ???
}
void CI2CPi::close()
{
if (m_spi)
bcm2835_spi_end();
else
bcm2835_i2c_end();
// Release Raspberry I/O control
bcm2835_close();
}