mirror of
https://github.com/DCC-EX/CommandStation-EX.git
synced 2024-11-23 08:06:13 +01:00
Neil bugfixes. (#186)
* Re-enable native I2C driver. * Minor non-functional changes to native I2C Manager. * Minor changes to make variable types explicit in comparisons. * Fix IODevice::loop() to avoid null pointer dereference. Strange problems with LCD driver tracked down to being caused by a call to p->_loop() when p is NULL. * Correct sense of comparison in LCN support function Turnout::setClosedStateOnly() * Remove code (now unused) from LCD driver. * Add I2C textual error messages. * Add I2C textual error messages. * Fix compile error in 4809 I2C driver. * Remove init function call from SSD1306 driver.
This commit is contained in:
parent
5e30740c5b
commit
0a9fcf6ebc
|
@ -147,6 +147,25 @@ uint8_t I2CManagerClass::finishRB(I2CRB *rb, uint8_t status) {
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/***************************************************************************
|
||||||
|
* Get a message corresponding to the error status
|
||||||
|
***************************************************************************/
|
||||||
|
const FSH *I2CManagerClass::getErrorMessage(uint8_t status) {
|
||||||
|
switch (status) {
|
||||||
|
case I2C_STATUS_OK: return F("OK");
|
||||||
|
case I2C_STATUS_TRUNCATED: return F("Transmission truncated");
|
||||||
|
case I2C_STATUS_NEGATIVE_ACKNOWLEDGE: return F("No response from device (address NAK)");
|
||||||
|
case I2C_STATUS_TRANSMIT_ERROR: return F("Transmit error (data NAK)");
|
||||||
|
case I2C_STATUS_OTHER_TWI_ERROR: return F("Other Wire/TWI error");
|
||||||
|
case I2C_STATUS_TIMEOUT: return F("Timeout");
|
||||||
|
case I2C_STATUS_ARBITRATION_LOST: return F("Arbitration lost");
|
||||||
|
case I2C_STATUS_BUS_ERROR: return F("I2C bus error");
|
||||||
|
case I2C_STATUS_UNEXPECTED_ERROR: return F("Unexpected error");
|
||||||
|
case I2C_STATUS_PENDING: return F("Request pending");
|
||||||
|
default: return F("Error code not recognised");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/***************************************************************************
|
/***************************************************************************
|
||||||
* Declare singleton class instance.
|
* Declare singleton class instance.
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
@ -158,12 +177,25 @@ I2CManagerClass I2CManager = I2CManagerClass();
|
||||||
/////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/***************************************************************************
|
/***************************************************************************
|
||||||
* Block waiting for request block to complete, and return completion status
|
* Block waiting for request block to complete, and return completion status.
|
||||||
|
* Since such a loop could potentially last for ever if the RB status doesn't
|
||||||
|
* change, we set a high limit (0.1sec, 100ms) on the wait time and, if it
|
||||||
|
* hasn't changed by that time we assume it's not going to, and just return
|
||||||
|
* a timeout status. This means that CS will not lock up.
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
uint8_t I2CRB::wait() {
|
uint8_t I2CRB::wait() {
|
||||||
do
|
unsigned long waitStart = millis();
|
||||||
|
do {
|
||||||
I2CManager.loop();
|
I2CManager.loop();
|
||||||
while (status==I2C_STATUS_PENDING);
|
// Rather than looping indefinitely, let's set a very high timeout (100ms).
|
||||||
|
if ((millis() - waitStart) > 100UL) {
|
||||||
|
DIAG(F("I2C TIMEOUT I2C:x%x I2CRB:x%x"), i2cAddress, this);
|
||||||
|
status = I2C_STATUS_TIMEOUT;
|
||||||
|
// Note that, although the timeout is posted, the request may yet complete.
|
||||||
|
// TODO: Ideally we would like to cancel the request.
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
} while (status==I2C_STATUS_PENDING);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
25
I2CManager.h
25
I2CManager.h
|
@ -111,19 +111,22 @@
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define I2C_USE_WIRE
|
//#define I2C_USE_WIRE
|
||||||
#ifndef I2C_NO_INTERRUPTS
|
#ifndef I2C_NO_INTERRUPTS
|
||||||
#define I2C_USE_INTERRUPTS
|
#define I2C_USE_INTERRUPTS
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Status codes for I2CRB structures.
|
// Status codes for I2CRB structures.
|
||||||
enum : uint8_t {
|
enum : uint8_t {
|
||||||
|
// Codes used by Wire and by native drivers
|
||||||
I2C_STATUS_OK=0,
|
I2C_STATUS_OK=0,
|
||||||
I2C_STATUS_TRUNCATED=1,
|
I2C_STATUS_TRUNCATED=1,
|
||||||
I2C_STATUS_DEVICE_NOT_PRESENT=2,
|
I2C_STATUS_NEGATIVE_ACKNOWLEDGE=2,
|
||||||
I2C_STATUS_TRANSMIT_ERROR=3,
|
I2C_STATUS_TRANSMIT_ERROR=3,
|
||||||
I2C_STATUS_NEGATIVE_ACKNOWLEDGE=4,
|
|
||||||
I2C_STATUS_TIMEOUT=5,
|
I2C_STATUS_TIMEOUT=5,
|
||||||
|
// Code used by Wire only
|
||||||
|
I2C_STATUS_OTHER_TWI_ERROR=4, // catch-all error
|
||||||
|
// Codes used by native drivers only
|
||||||
I2C_STATUS_ARBITRATION_LOST=6,
|
I2C_STATUS_ARBITRATION_LOST=6,
|
||||||
I2C_STATUS_BUS_ERROR=7,
|
I2C_STATUS_BUS_ERROR=7,
|
||||||
I2C_STATUS_UNEXPECTED_ERROR=8,
|
I2C_STATUS_UNEXPECTED_ERROR=8,
|
||||||
|
@ -151,14 +154,16 @@ typedef enum : uint8_t
|
||||||
#define I2C_FREQ 400000L
|
#define I2C_FREQ 400000L
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Struct defining a request context for an I2C operation.
|
// Class defining a request context for an I2C operation.
|
||||||
struct I2CRB {
|
class I2CRB {
|
||||||
|
public:
|
||||||
volatile uint8_t status; // Completion status, or pending flag (updated from IRC)
|
volatile uint8_t status; // Completion status, or pending flag (updated from IRC)
|
||||||
volatile uint8_t nBytes; // Number of bytes read (updated from IRC)
|
volatile uint8_t nBytes; // Number of bytes read (updated from IRC)
|
||||||
|
|
||||||
|
inline I2CRB() { status = I2C_STATUS_OK; };
|
||||||
uint8_t wait();
|
uint8_t wait();
|
||||||
bool isBusy();
|
bool isBusy();
|
||||||
inline void init() { status = I2C_STATUS_OK; };
|
|
||||||
void setReadParams(uint8_t i2cAddress, uint8_t *readBuffer, uint8_t readLen);
|
void setReadParams(uint8_t i2cAddress, uint8_t *readBuffer, uint8_t readLen);
|
||||||
void setRequestParams(uint8_t i2cAddress, uint8_t *readBuffer, uint8_t readLen, const uint8_t *writeBuffer, uint8_t writeLen);
|
void setRequestParams(uint8_t i2cAddress, uint8_t *readBuffer, uint8_t readLen, const uint8_t *writeBuffer, uint8_t writeLen);
|
||||||
void setWriteParams(uint8_t i2cAddress, const uint8_t *writeBuffer, uint8_t writeLen);
|
void setWriteParams(uint8_t i2cAddress, const uint8_t *writeBuffer, uint8_t writeLen);
|
||||||
|
@ -213,6 +218,10 @@ public:
|
||||||
// Loop method
|
// Loop method
|
||||||
void loop();
|
void loop();
|
||||||
|
|
||||||
|
// Expand error codes into text. Note that they are in flash so
|
||||||
|
// need to be printed using FSH.
|
||||||
|
static const FSH *getErrorMessage(uint8_t status);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool _beginCompleted = false;
|
bool _beginCompleted = false;
|
||||||
bool _clockSpeedFixed = false;
|
bool _clockSpeedFixed = false;
|
||||||
|
@ -258,7 +267,9 @@ private:
|
||||||
static void I2C_close();
|
static void I2C_close();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void setTimeout(unsigned long value) { timeout = value;};
|
// setTimeout sets the timout value for I2C transactions.
|
||||||
|
// TODO: Get I2C timeout working before uncommenting the code below.
|
||||||
|
void setTimeout(unsigned long value) { (void)value; /* timeout = value; */ };
|
||||||
|
|
||||||
// handleInterrupt needs to be public to be called from the ISR function!
|
// handleInterrupt needs to be public to be called from the ISR function!
|
||||||
static void handleInterrupt();
|
static void handleInterrupt();
|
||||||
|
|
|
@ -105,14 +105,14 @@ void I2CManagerClass::I2C_handleInterrupt() {
|
||||||
I2C_sendStart(); // Reinitiate request
|
I2C_sendStart(); // Reinitiate request
|
||||||
} else if (currentStatus & TWI_BUSERR_bm) {
|
} else if (currentStatus & TWI_BUSERR_bm) {
|
||||||
// Bus error
|
// Bus error
|
||||||
status = I2C_STATUS_BUS_ERROR;
|
state = I2C_STATUS_BUS_ERROR;
|
||||||
TWI0.MSTATUS = currentStatus; // clear all flags
|
TWI0.MSTATUS = currentStatus; // clear all flags
|
||||||
} else if (currentStatus & TWI_WIF_bm) {
|
} else if (currentStatus & TWI_WIF_bm) {
|
||||||
// Master write completed
|
// Master write completed
|
||||||
if (currentStatus & TWI_RXACK_bm) {
|
if (currentStatus & TWI_RXACK_bm) {
|
||||||
// Nacked, send stop.
|
// Nacked, send stop.
|
||||||
TWI0.MCTRLB = TWI_MCMD_STOP_gc;
|
TWI0.MCTRLB = TWI_MCMD_STOP_gc;
|
||||||
status = I2C_STATUS_NEGATIVE_ACKNOWLEDGE;
|
state = I2C_STATUS_NEGATIVE_ACKNOWLEDGE;
|
||||||
} else if (bytesToSend) {
|
} else if (bytesToSend) {
|
||||||
// Acked, so send next byte
|
// Acked, so send next byte
|
||||||
if (currentRequest->operation == OPERATION_SEND_P)
|
if (currentRequest->operation == OPERATION_SEND_P)
|
||||||
|
@ -126,7 +126,7 @@ void I2CManagerClass::I2C_handleInterrupt() {
|
||||||
} else {
|
} else {
|
||||||
// No more data to send/receive. Initiate a STOP condition.
|
// No more data to send/receive. Initiate a STOP condition.
|
||||||
TWI0.MCTRLB = TWI_MCMD_STOP_gc;
|
TWI0.MCTRLB = TWI_MCMD_STOP_gc;
|
||||||
status = I2C_STATUS_OK; // Done
|
state = I2C_STATUS_OK; // Done
|
||||||
}
|
}
|
||||||
} else if (currentStatus & TWI_RIF_bm) {
|
} else if (currentStatus & TWI_RIF_bm) {
|
||||||
// Master read completed without errors
|
// Master read completed without errors
|
||||||
|
@ -136,7 +136,7 @@ void I2CManagerClass::I2C_handleInterrupt() {
|
||||||
} else {
|
} else {
|
||||||
// Buffer full, issue nack/stop
|
// Buffer full, issue nack/stop
|
||||||
TWI0.MCTRLB = TWI_ACKACT_bm | TWI_MCMD_STOP_gc;
|
TWI0.MCTRLB = TWI_ACKACT_bm | TWI_MCMD_STOP_gc;
|
||||||
status = I2C_STATUS_OK;
|
state = I2C_STATUS_OK;
|
||||||
}
|
}
|
||||||
if (bytesToReceive) {
|
if (bytesToReceive) {
|
||||||
// More bytes to receive, issue ack and start another read
|
// More bytes to receive, issue ack and start another read
|
||||||
|
@ -144,7 +144,7 @@ void I2CManagerClass::I2C_handleInterrupt() {
|
||||||
} else {
|
} else {
|
||||||
// Transaction finished, issue NACK and STOP.
|
// Transaction finished, issue NACK and STOP.
|
||||||
TWI0.MCTRLB = TWI_ACKACT_bm | TWI_MCMD_STOP_gc;
|
TWI0.MCTRLB = TWI_ACKACT_bm | TWI_MCMD_STOP_gc;
|
||||||
status = I2C_STATUS_OK;
|
state = I2C_STATUS_OK;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,10 +59,9 @@ void I2CManagerClass::_setClock(unsigned long i2cClockSpeed) {
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
void I2CManagerClass::startTransaction() {
|
void I2CManagerClass::startTransaction() {
|
||||||
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
||||||
I2CRB *t = queueHead;
|
if ((state == I2C_STATE_FREE) && (queueHead != NULL)) {
|
||||||
if ((state == I2C_STATE_FREE) && (t != NULL)) {
|
|
||||||
state = I2C_STATE_ACTIVE;
|
state = I2C_STATE_ACTIVE;
|
||||||
currentRequest = t;
|
currentRequest = queueHead;
|
||||||
rxCount = txCount = 0;
|
rxCount = txCount = 0;
|
||||||
// Copy key fields to static data for speed.
|
// Copy key fields to static data for speed.
|
||||||
operation = currentRequest->operation;
|
operation = currentRequest->operation;
|
||||||
|
@ -85,9 +84,9 @@ void I2CManagerClass::queueRequest(I2CRB *req) {
|
||||||
queueHead = queueTail = req; // Only item on queue
|
queueHead = queueTail = req; // Only item on queue
|
||||||
else
|
else
|
||||||
queueTail = queueTail->nextRequest = req; // Add to end
|
queueTail = queueTail->nextRequest = req; // Add to end
|
||||||
|
startTransaction();
|
||||||
}
|
}
|
||||||
|
|
||||||
startTransaction();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/***************************************************************************
|
/***************************************************************************
|
||||||
|
@ -135,7 +134,7 @@ void I2CManagerClass::checkForTimeout() {
|
||||||
unsigned long currentMicros = micros();
|
unsigned long currentMicros = micros();
|
||||||
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
||||||
I2CRB *t = queueHead;
|
I2CRB *t = queueHead;
|
||||||
if (state==I2C_STATE_ACTIVE && t!=0 && timeout > 0) {
|
if (state==I2C_STATE_ACTIVE && t!=0 && t==currentRequest && timeout > 0) {
|
||||||
// Check for timeout
|
// Check for timeout
|
||||||
if (currentMicros - startTime > timeout) {
|
if (currentMicros - startTime > timeout) {
|
||||||
// Excessive time. Dequeue request
|
// Excessive time. Dequeue request
|
||||||
|
@ -150,7 +149,7 @@ void I2CManagerClass::checkForTimeout() {
|
||||||
I2C_init();
|
I2C_init();
|
||||||
state = I2C_STATE_FREE;
|
state = I2C_STATE_FREE;
|
||||||
|
|
||||||
// Initiate next queued request
|
// Initiate next queued request if any.
|
||||||
startTransaction();
|
startTransaction();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -173,28 +172,30 @@ void I2CManagerClass::loop() {
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
void I2CManagerClass::handleInterrupt() {
|
void I2CManagerClass::handleInterrupt() {
|
||||||
|
|
||||||
|
// Update hardware state machine
|
||||||
I2C_handleInterrupt();
|
I2C_handleInterrupt();
|
||||||
|
|
||||||
// Experimental -- perform the post processing with interrupts enabled.
|
// Check if current request has completed. If there's a current request
|
||||||
//interrupts();
|
// and state isn't active then state contains the completion status of the request.
|
||||||
|
if (state != I2C_STATE_ACTIVE && currentRequest != NULL) {
|
||||||
if (state!=I2C_STATE_ACTIVE && state != I2C_STATE_FREE) {
|
|
||||||
// Remove completed request from head of queue
|
// Remove completed request from head of queue
|
||||||
I2CRB * t;
|
|
||||||
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
||||||
t = queueHead;
|
I2CRB * t = queueHead;
|
||||||
if (t != NULL) {
|
if (t == queueHead) {
|
||||||
queueHead = t->nextRequest;
|
queueHead = t->nextRequest;
|
||||||
if (!queueHead) queueTail = queueHead;
|
if (!queueHead) queueTail = queueHead;
|
||||||
t->nBytes = rxCount;
|
t->nBytes = rxCount;
|
||||||
t->status = state;
|
t->status = state;
|
||||||
}
|
|
||||||
// I2C state machine is now free for next request
|
// I2C state machine is now free for next request
|
||||||
|
currentRequest = NULL;
|
||||||
state = I2C_STATE_FREE;
|
state = I2C_STATE_FREE;
|
||||||
}
|
|
||||||
// Start next request (if any)
|
// Start next request (if any)
|
||||||
I2CManager.startTransaction();
|
I2CManager.startTransaction();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fields in I2CManager class specific to Non-blocking implementation.
|
// Fields in I2CManager class specific to Non-blocking implementation.
|
||||||
|
|
|
@ -69,8 +69,10 @@ void IODevice::loop() {
|
||||||
unsigned long currentMicros = micros();
|
unsigned long currentMicros = micros();
|
||||||
// Call every device's loop function in turn, one per entry.
|
// Call every device's loop function in turn, one per entry.
|
||||||
if (!_nextLoopDevice) _nextLoopDevice = _firstDevice;
|
if (!_nextLoopDevice) _nextLoopDevice = _firstDevice;
|
||||||
|
if (_nextLoopDevice) {
|
||||||
_nextLoopDevice->_loop(currentMicros);
|
_nextLoopDevice->_loop(currentMicros);
|
||||||
_nextLoopDevice = _nextLoopDevice->_nextDevice;
|
_nextLoopDevice = _nextLoopDevice->_nextDevice;
|
||||||
|
}
|
||||||
|
|
||||||
// Report loop time if diags enabled
|
// Report loop time if diags enabled
|
||||||
#if defined(DIAG_LOOPTIMES)
|
#if defined(DIAG_LOOPTIMES)
|
||||||
|
|
|
@ -145,7 +145,8 @@ void GPIOBase<T>::_loop(unsigned long currentMicros) {
|
||||||
_deviceState = DEVSTATE_NORMAL;
|
_deviceState = DEVSTATE_NORMAL;
|
||||||
} else {
|
} else {
|
||||||
_deviceState = DEVSTATE_FAILED;
|
_deviceState = DEVSTATE_FAILED;
|
||||||
DIAG(F("%S I2C:x%x Error:%d"), _deviceName, _I2CAddress, status);
|
DIAG(F("%S I2C:x%x Error:%d %S"), _deviceName, _I2CAddress, status,
|
||||||
|
I2CManager.getErrorMessage(status));
|
||||||
}
|
}
|
||||||
_processCompletion(status);
|
_processCompletion(status);
|
||||||
|
|
||||||
|
@ -174,7 +175,7 @@ void GPIOBase<T>::_loop(unsigned long currentMicros) {
|
||||||
if (digitalRead(_gpioInterruptPin)) return;
|
if (digitalRead(_gpioInterruptPin)) return;
|
||||||
} else
|
} else
|
||||||
// No interrupt pin. Check if tick has elapsed. If not, finish.
|
// No interrupt pin. Check if tick has elapsed. If not, finish.
|
||||||
if (currentMicros - _lastLoopEntry < _portTickTime) return;
|
if (currentMicros - _lastLoopEntry < (unsigned long)_portTickTime) return;
|
||||||
|
|
||||||
// TODO: Could suppress reads if there are no pins configured as inputs!
|
// TODO: Could suppress reads if there are no pins configured as inputs!
|
||||||
|
|
||||||
|
|
|
@ -102,7 +102,7 @@ protected:
|
||||||
|
|
||||||
// _loop function - read HC-SR04 once every 50 milliseconds.
|
// _loop function - read HC-SR04 once every 50 milliseconds.
|
||||||
void _loop(unsigned long currentMicros) override {
|
void _loop(unsigned long currentMicros) override {
|
||||||
if (currentMicros - _lastExecutionTime > 50000) {
|
if (currentMicros - _lastExecutionTime > 50000UL) {
|
||||||
_lastExecutionTime = currentMicros;
|
_lastExecutionTime = currentMicros;
|
||||||
|
|
||||||
_value = read_HCSR04device();
|
_value = read_HCSR04device();
|
||||||
|
|
|
@ -59,8 +59,6 @@ LiquidCrystal_I2C::LiquidCrystal_I2C(uint8_t lcd_Addr, uint8_t lcd_cols,
|
||||||
backlight();
|
backlight();
|
||||||
lcdDisplay = this;
|
lcdDisplay = this;
|
||||||
}
|
}
|
||||||
// Initialise request block for comms.
|
|
||||||
requestBlock.setWriteParams(lcd_Addr, outputBuffer, sizeof(outputBuffer));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void LiquidCrystal_I2C::begin() {
|
void LiquidCrystal_I2C::begin() {
|
||||||
|
@ -192,8 +190,6 @@ void LiquidCrystal_I2C::send(uint8_t value, uint8_t mode) {
|
||||||
mode |= _backlightval;
|
mode |= _backlightval;
|
||||||
uint8_t highnib = (((value >> 4) & 0x0f) << BACKPACK_DATA_BITS) | mode;
|
uint8_t highnib = (((value >> 4) & 0x0f) << BACKPACK_DATA_BITS) | mode;
|
||||||
uint8_t lownib = ((value & 0x0f) << BACKPACK_DATA_BITS) | mode;
|
uint8_t lownib = ((value & 0x0f) << BACKPACK_DATA_BITS) | mode;
|
||||||
// Wait for previous request to complete before writing to outputbuffer.
|
|
||||||
requestBlock.wait();
|
|
||||||
// Send both nibbles
|
// Send both nibbles
|
||||||
uint8_t len = 0;
|
uint8_t len = 0;
|
||||||
outputBuffer[len++] = highnib|En;
|
outputBuffer[len++] = highnib|En;
|
||||||
|
@ -209,8 +205,6 @@ void LiquidCrystal_I2C::write4bits(uint8_t value) {
|
||||||
// Enable must be set/reset for at least 450ns. This is well within the
|
// Enable must be set/reset for at least 450ns. This is well within the
|
||||||
// I2C clock cycle time of 2.5us at 400kHz. Data is clocked in to the
|
// I2C clock cycle time of 2.5us at 400kHz. Data is clocked in to the
|
||||||
// HD44780 on the trailing edge of the Enable pin.
|
// HD44780 on the trailing edge of the Enable pin.
|
||||||
// Wait for previous request to complete before writing to outputbuffer.
|
|
||||||
requestBlock.wait();
|
|
||||||
uint8_t len = 0;
|
uint8_t len = 0;
|
||||||
outputBuffer[len++] = _data|En;
|
outputBuffer[len++] = _data|En;
|
||||||
outputBuffer[len++] = _data;
|
outputBuffer[len++] = _data;
|
||||||
|
@ -220,8 +214,6 @@ void LiquidCrystal_I2C::write4bits(uint8_t value) {
|
||||||
// write a byte to the PCF8574 I2C interface. We don't need to set
|
// write a byte to the PCF8574 I2C interface. We don't need to set
|
||||||
// the enable pin for this.
|
// the enable pin for this.
|
||||||
void LiquidCrystal_I2C::expanderWrite(uint8_t value) {
|
void LiquidCrystal_I2C::expanderWrite(uint8_t value) {
|
||||||
// Wait for previous request to complete before writing to outputbuffer.
|
|
||||||
requestBlock.wait();
|
|
||||||
outputBuffer[0] = value | _backlightval;
|
outputBuffer[0] = value | _backlightval;
|
||||||
I2CManager.write(_Addr, outputBuffer, 1); // Write command synchronously
|
I2CManager.write(_Addr, outputBuffer, 1); // Write command synchronously
|
||||||
}
|
}
|
|
@ -75,10 +75,8 @@ public:
|
||||||
void backlight();
|
void backlight();
|
||||||
|
|
||||||
void command(uint8_t);
|
void command(uint8_t);
|
||||||
void init();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void init_priv();
|
|
||||||
void send(uint8_t, uint8_t);
|
void send(uint8_t, uint8_t);
|
||||||
void write4bits(uint8_t);
|
void write4bits(uint8_t);
|
||||||
void expanderWrite(uint8_t);
|
void expanderWrite(uint8_t);
|
||||||
|
@ -88,10 +86,9 @@ private:
|
||||||
uint8_t _displaymode;
|
uint8_t _displaymode;
|
||||||
uint8_t _backlightval;
|
uint8_t _backlightval;
|
||||||
|
|
||||||
I2CRB requestBlock;
|
|
||||||
uint8_t outputBuffer[4];
|
uint8_t outputBuffer[4];
|
||||||
// I/O is synchronous, so if this is called we're not busy!
|
// I/O is synchronous, so if this is called we're not busy!
|
||||||
bool isBusy() { return false; }
|
bool isBusy() override { return false; }
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -99,9 +99,6 @@ SSD1306AsciiWire::SSD1306AsciiWire(int width, int height) {
|
||||||
lcdRows = height / 8;
|
lcdRows = height / 8;
|
||||||
lcdCols = width / 6;
|
lcdCols = width / 6;
|
||||||
|
|
||||||
// Initialise request block for I2C
|
|
||||||
requestBlock.init();
|
|
||||||
|
|
||||||
I2CManager.begin();
|
I2CManager.begin();
|
||||||
I2CManager.setClock(400000L); // Set max supported I2C speed
|
I2CManager.setClock(400000L); // Set max supported I2C speed
|
||||||
for (byte address = 0x3c; address <= 0x3d; address++) {
|
for (byte address = 0x3c; address <= 0x3d; address++) {
|
||||||
|
|
|
@ -107,7 +107,7 @@
|
||||||
|
|
||||||
bool Turnout::setClosedStateOnly(uint16_t id, bool close) {
|
bool Turnout::setClosedStateOnly(uint16_t id, bool close) {
|
||||||
Turnout *tt = get(id);
|
Turnout *tt = get(id);
|
||||||
if (tt) return false;
|
if (!tt) return false;
|
||||||
tt->_turnoutData.closed = close;
|
tt->_turnoutData.closed = close;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user