Next set of split_common changes (#4974)
* Update split_common to use standard i2c drivers * Eliminate RGB_DIRTY/BACKLIT_DIRTY * Fix avr i2c_master error handling * Fix i2c_slave addressing * Remove unneeded timeout on i2c_stop() * Fix RGB I2C transfers * Remove incorrect comment
This commit is contained in:
parent
25bb059e4e
commit
37932c293c
24 changed files with 187 additions and 614 deletions
|
@ -308,16 +308,16 @@ ifeq ($(strip $(SPLIT_KEYBOARD)), yes)
|
||||||
OPT_DEFS += -DSPLIT_KEYBOARD
|
OPT_DEFS += -DSPLIT_KEYBOARD
|
||||||
|
|
||||||
# Include files used by all split keyboards
|
# Include files used by all split keyboards
|
||||||
QUANTUM_SRC += $(QUANTUM_DIR)/split_common/split_flags.c \
|
QUANTUM_SRC += $(QUANTUM_DIR)/split_common/split_util.c
|
||||||
$(QUANTUM_DIR)/split_common/split_util.c
|
|
||||||
|
|
||||||
# Determine which (if any) transport files are required
|
# Determine which (if any) transport files are required
|
||||||
ifneq ($(strip $(SPLIT_TRANSPORT)), custom)
|
ifneq ($(strip $(SPLIT_TRANSPORT)), custom)
|
||||||
QUANTUM_SRC += $(QUANTUM_DIR)/split_common/transport.c
|
QUANTUM_SRC += $(QUANTUM_DIR)/split_common/transport.c
|
||||||
# Functions added via QUANTUM_LIB_SRC are only included in the final binary if they're called.
|
# Functions added via QUANTUM_LIB_SRC are only included in the final binary if they're called.
|
||||||
# Unused functions are pruned away, which is why we can add both drivers here without bloat.
|
# Unused functions are pruned away, which is why we can add multiple drivers here without bloat.
|
||||||
QUANTUM_LIB_SRC += $(QUANTUM_DIR)/split_common/i2c.c \
|
QUANTUM_LIB_SRC += $(QUANTUM_DIR)/split_common/serial.c \
|
||||||
$(QUANTUM_DIR)/split_common/serial.c
|
i2c_master.c \
|
||||||
|
i2c_slave.c
|
||||||
endif
|
endif
|
||||||
COMMON_VPATH += $(QUANTUM_PATH)/split_common
|
COMMON_VPATH += $(QUANTUM_PATH)/split_common
|
||||||
endif
|
endif
|
||||||
|
|
|
@ -12,7 +12,7 @@ The I2C Master drivers used in QMK have a set of common functions to allow porta
|
||||||
|`uint8_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout);` |Receive data over I2C. Address is the 7-bit slave address without the direction. Saves number of bytes specified by `length` in `data` array. Returns status of transaction. |
|
|`uint8_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout);` |Receive data over I2C. Address is the 7-bit slave address without the direction. Saves number of bytes specified by `length` in `data` array. Returns status of transaction. |
|
||||||
|`uint8_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);` |Same as the `i2c_transmit` function but `regaddr` sets where in the slave the data will be written. |
|
|`uint8_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);` |Same as the `i2c_transmit` function but `regaddr` sets where in the slave the data will be written. |
|
||||||
|`uint8_t i2c_readReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);` |Same as the `i2c_receive` function but `regaddr` sets from where in the slave the data will be read. |
|
|`uint8_t i2c_readReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);` |Same as the `i2c_receive` function but `regaddr` sets from where in the slave the data will be read. |
|
||||||
|`uint8_t i2c_stop(uint16_t timeout);` |Stops the I2C driver. |
|
|`uint8_t i2c_stop(void);` |Ends an I2C transaction. |
|
||||||
|
|
||||||
### Function Return
|
### Function Return
|
||||||
|
|
||||||
|
|
|
@ -101,8 +101,7 @@ uint8_t i2c_readReg(uint8_t devaddr, uint8_t* regaddr, uint8_t* data, uint16_t l
|
||||||
return i2cMasterTransmitTimeout(&I2C_DRIVER, (i2c_address >> 1), regaddr, 1, data, length, MS2ST(timeout));
|
return i2cMasterTransmitTimeout(&I2C_DRIVER, (i2c_address >> 1), regaddr, 1, data, length, MS2ST(timeout));
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is usually not needed. It releases the driver to allow pins to become GPIO again.
|
uint8_t i2c_stop(void)
|
||||||
uint8_t i2c_stop(uint16_t timeout)
|
|
||||||
{
|
{
|
||||||
i2cStop(&I2C_DRIVER);
|
i2cStop(&I2C_DRIVER);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -47,4 +47,4 @@ uint8_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t ti
|
||||||
uint8_t i2c_transmit_receive(uint8_t address, uint8_t * tx_body, uint16_t tx_length, uint8_t * rx_body, uint16_t rx_length);
|
uint8_t i2c_transmit_receive(uint8_t address, uint8_t * tx_body, uint16_t tx_length, uint8_t * rx_body, uint16_t rx_length);
|
||||||
uint8_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
uint8_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
||||||
uint8_t i2c_readReg(uint8_t devaddr, uint8_t* regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
uint8_t i2c_readReg(uint8_t devaddr, uint8_t* regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
||||||
uint8_t i2c_stop(uint16_t timeout);
|
uint8_t i2c_stop(void);
|
||||||
|
|
|
@ -7,43 +7,44 @@
|
||||||
|
|
||||||
#include "i2c_master.h"
|
#include "i2c_master.h"
|
||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
#include "wait.h"
|
||||||
|
|
||||||
#ifndef F_SCL
|
#ifndef F_SCL
|
||||||
#define F_SCL 400000UL // SCL frequency
|
# define F_SCL 400000UL // SCL frequency
|
||||||
#endif
|
#endif
|
||||||
#define Prescaler 1
|
#define Prescaler 1
|
||||||
#define TWBR_val ((((F_CPU / F_SCL) / Prescaler) - 16 ) / 2)
|
#define TWBR_val ((((F_CPU / F_SCL) / Prescaler) - 16) / 2)
|
||||||
|
|
||||||
void i2c_init(void)
|
void i2c_init(void) {
|
||||||
{
|
|
||||||
TWSR = 0; /* no prescaler */
|
TWSR = 0; /* no prescaler */
|
||||||
TWBR = (uint8_t)TWBR_val;
|
TWBR = (uint8_t)TWBR_val;
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_start(uint8_t address, uint16_t timeout)
|
i2c_status_t i2c_start(uint8_t address, uint16_t timeout) {
|
||||||
{
|
|
||||||
// reset TWI control register
|
// reset TWI control register
|
||||||
TWCR = 0;
|
TWCR = 0;
|
||||||
// transmit START condition
|
// transmit START condition
|
||||||
TWCR = (1<<TWINT) | (1<<TWSTA) | (1<<TWEN);
|
TWCR = (1 << TWINT) | (1 << TWSTA) | (1 << TWEN);
|
||||||
|
|
||||||
uint16_t timeout_timer = timer_read();
|
uint16_t timeout_timer = timer_read();
|
||||||
while( !(TWCR & (1<<TWINT)) ) {
|
while (!(TWCR & (1 << TWINT))) {
|
||||||
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
||||||
return I2C_STATUS_TIMEOUT;
|
return I2C_STATUS_TIMEOUT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// check if the start condition was successfully transmitted
|
// check if the start condition was successfully transmitted
|
||||||
if(((TW_STATUS & 0xF8) != TW_START) && ((TW_STATUS & 0xF8) != TW_REP_START)){ return I2C_STATUS_ERROR; }
|
if (((TW_STATUS & 0xF8) != TW_START) && ((TW_STATUS & 0xF8) != TW_REP_START)) {
|
||||||
|
return I2C_STATUS_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
// load slave address into data register
|
// load slave address into data register
|
||||||
TWDR = address;
|
TWDR = address;
|
||||||
// start transmission of address
|
// start transmission of address
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN);
|
TWCR = (1 << TWINT) | (1 << TWEN);
|
||||||
|
|
||||||
timeout_timer = timer_read();
|
timeout_timer = timer_read();
|
||||||
while( !(TWCR & (1<<TWINT)) ) {
|
while (!(TWCR & (1 << TWINT))) {
|
||||||
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
||||||
return I2C_STATUS_TIMEOUT;
|
return I2C_STATUS_TIMEOUT;
|
||||||
}
|
}
|
||||||
|
@ -51,38 +52,39 @@ i2c_status_t i2c_start(uint8_t address, uint16_t timeout)
|
||||||
|
|
||||||
// check if the device has acknowledged the READ / WRITE mode
|
// check if the device has acknowledged the READ / WRITE mode
|
||||||
uint8_t twst = TW_STATUS & 0xF8;
|
uint8_t twst = TW_STATUS & 0xF8;
|
||||||
if ( (twst != TW_MT_SLA_ACK) && (twst != TW_MR_SLA_ACK) ) return I2C_STATUS_ERROR;
|
if ((twst != TW_MT_SLA_ACK) && (twst != TW_MR_SLA_ACK)) {
|
||||||
|
return I2C_STATUS_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
return I2C_STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_write(uint8_t data, uint16_t timeout)
|
i2c_status_t i2c_write(uint8_t data, uint16_t timeout) {
|
||||||
{
|
|
||||||
// load data into data register
|
// load data into data register
|
||||||
TWDR = data;
|
TWDR = data;
|
||||||
// start transmission of data
|
// start transmission of data
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN);
|
TWCR = (1 << TWINT) | (1 << TWEN);
|
||||||
|
|
||||||
uint16_t timeout_timer = timer_read();
|
uint16_t timeout_timer = timer_read();
|
||||||
while( !(TWCR & (1<<TWINT)) ) {
|
while (!(TWCR & (1 << TWINT))) {
|
||||||
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
||||||
return I2C_STATUS_TIMEOUT;
|
return I2C_STATUS_TIMEOUT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if( (TW_STATUS & 0xF8) != TW_MT_DATA_ACK ){ return I2C_STATUS_ERROR; }
|
if ((TW_STATUS & 0xF8) != TW_MT_DATA_ACK) {
|
||||||
|
return I2C_STATUS_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
return I2C_STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
int16_t i2c_read_ack(uint16_t timeout)
|
int16_t i2c_read_ack(uint16_t timeout) {
|
||||||
{
|
|
||||||
|
|
||||||
// start TWI module and acknowledge data after reception
|
// start TWI module and acknowledge data after reception
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWEA);
|
TWCR = (1 << TWINT) | (1 << TWEN) | (1 << TWEA);
|
||||||
|
|
||||||
uint16_t timeout_timer = timer_read();
|
uint16_t timeout_timer = timer_read();
|
||||||
while( !(TWCR & (1<<TWINT)) ) {
|
while (!(TWCR & (1 << TWINT))) {
|
||||||
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
||||||
return I2C_STATUS_TIMEOUT;
|
return I2C_STATUS_TIMEOUT;
|
||||||
}
|
}
|
||||||
|
@ -92,14 +94,12 @@ int16_t i2c_read_ack(uint16_t timeout)
|
||||||
return TWDR;
|
return TWDR;
|
||||||
}
|
}
|
||||||
|
|
||||||
int16_t i2c_read_nack(uint16_t timeout)
|
int16_t i2c_read_nack(uint16_t timeout) {
|
||||||
{
|
|
||||||
|
|
||||||
// start receiving without acknowledging reception
|
// start receiving without acknowledging reception
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN);
|
TWCR = (1 << TWINT) | (1 << TWEN);
|
||||||
|
|
||||||
uint16_t timeout_timer = timer_read();
|
uint16_t timeout_timer = timer_read();
|
||||||
while( !(TWCR & (1<<TWINT)) ) {
|
while (!(TWCR & (1 << TWINT))) {
|
||||||
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
||||||
return I2C_STATUS_TIMEOUT;
|
return I2C_STATUS_TIMEOUT;
|
||||||
}
|
}
|
||||||
|
@ -109,115 +109,89 @@ int16_t i2c_read_nack(uint16_t timeout)
|
||||||
return TWDR;
|
return TWDR;
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_transmit(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout)
|
i2c_status_t i2c_transmit(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout) {
|
||||||
{
|
|
||||||
i2c_status_t status = i2c_start(address | I2C_WRITE, timeout);
|
i2c_status_t status = i2c_start(address | I2C_WRITE, timeout);
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
for (uint16_t i = 0; i < length; i++) {
|
for (uint16_t i = 0; i < length && status >= 0; i++) {
|
||||||
status = i2c_write(data[i], timeout);
|
status = i2c_write(data[i], timeout);
|
||||||
if (status) return status;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
status = i2c_stop(timeout);
|
i2c_stop();
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout)
|
i2c_status_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout) {
|
||||||
{
|
|
||||||
i2c_status_t status = i2c_start(address | I2C_READ, timeout);
|
i2c_status_t status = i2c_start(address | I2C_READ, timeout);
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
for (uint16_t i = 0; i < (length-1); i++) {
|
for (uint16_t i = 0; i < (length - 1) && status >= 0; i++) {
|
||||||
status = i2c_read_ack(timeout);
|
status = i2c_read_ack(timeout);
|
||||||
if (status >= 0) {
|
if (status >= 0) {
|
||||||
data[i] = status;
|
data[i] = status;
|
||||||
} else {
|
|
||||||
return status;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (status >= 0) {
|
||||||
status = i2c_read_nack(timeout);
|
status = i2c_read_nack(timeout);
|
||||||
if (status >= 0 ) {
|
if (status >= 0) {
|
||||||
data[(length-1)] = status;
|
data[(length - 1)] = status;
|
||||||
} else {
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
i2c_stop();
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
|
||||||
|
|
||||||
status = i2c_stop(timeout);
|
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout)
|
i2c_status_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout) {
|
||||||
{
|
|
||||||
i2c_status_t status = i2c_start(devaddr | 0x00, timeout);
|
i2c_status_t status = i2c_start(devaddr | 0x00, timeout);
|
||||||
if (status) return status;
|
if (status >= 0) {
|
||||||
|
|
||||||
status = i2c_write(regaddr, timeout);
|
status = i2c_write(regaddr, timeout);
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
for (uint16_t i = 0; i < length; i++) {
|
for (uint16_t i = 0; i < length && status >= 0; i++) {
|
||||||
status = i2c_write(data[i], timeout);
|
status = i2c_write(data[i], timeout);
|
||||||
if (status) return status;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
status = i2c_stop(timeout);
|
i2c_stop();
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_readReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout)
|
i2c_status_t i2c_readReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout) {
|
||||||
{
|
|
||||||
i2c_status_t status = i2c_start(devaddr, timeout);
|
i2c_status_t status = i2c_start(devaddr, timeout);
|
||||||
if (status) return status;
|
if (status < 0) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
status = i2c_write(regaddr, timeout);
|
status = i2c_write(regaddr, timeout);
|
||||||
if (status) return status;
|
if (status < 0) {
|
||||||
|
goto error;
|
||||||
status = i2c_stop(timeout);
|
}
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
status = i2c_start(devaddr | 0x01, timeout);
|
status = i2c_start(devaddr | 0x01, timeout);
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
for (uint16_t i = 0; i < (length-1); i++) {
|
for (uint16_t i = 0; i < (length - 1) && status >= 0; i++) {
|
||||||
status = i2c_read_ack(timeout);
|
status = i2c_read_ack(timeout);
|
||||||
if (status >= 0) {
|
if (status >= 0) {
|
||||||
data[i] = status;
|
data[i] = status;
|
||||||
} else {
|
|
||||||
return status;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (status >= 0) {
|
||||||
status = i2c_read_nack(timeout);
|
status = i2c_read_nack(timeout);
|
||||||
if (status >= 0 ) {
|
if (status >= 0) {
|
||||||
data[(length-1)] = status;
|
data[(length - 1)] = status;
|
||||||
} else {
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
error:
|
||||||
|
i2c_stop();
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
|
||||||
|
|
||||||
status = i2c_stop(timeout);
|
|
||||||
if (status) return status;
|
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_status_t i2c_stop(uint16_t timeout)
|
void i2c_stop(void) {
|
||||||
{
|
|
||||||
// transmit STOP condition
|
// transmit STOP condition
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWSTO);
|
TWCR = (1 << TWINT) | (1 << TWEN) | (1 << TWSTO);
|
||||||
|
|
||||||
uint16_t timeout_timer = timer_read();
|
|
||||||
while(TWCR & (1<<TWSTO)) {
|
|
||||||
if ((timeout != I2C_TIMEOUT_INFINITE) && ((timer_read() - timeout_timer) >= timeout)) {
|
|
||||||
return I2C_STATUS_TIMEOUT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return I2C_STATUS_SUCCESS;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,6 @@ i2c_status_t i2c_transmit(uint8_t address, uint8_t* data, uint16_t length, uint1
|
||||||
i2c_status_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout);
|
i2c_status_t i2c_receive(uint8_t address, uint8_t* data, uint16_t length, uint16_t timeout);
|
||||||
i2c_status_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
i2c_status_t i2c_writeReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
||||||
i2c_status_t i2c_readReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
i2c_status_t i2c_readReg(uint8_t devaddr, uint8_t regaddr, uint8_t* data, uint16_t length, uint16_t timeout);
|
||||||
i2c_status_t i2c_stop(uint16_t timeout);
|
void i2c_stop(void);
|
||||||
|
|
||||||
#endif // I2C_MASTER_H
|
#endif // I2C_MASTER_H
|
|
@ -16,7 +16,7 @@ static volatile bool slave_has_register_set = false;
|
||||||
|
|
||||||
void i2c_slave_init(uint8_t address){
|
void i2c_slave_init(uint8_t address){
|
||||||
// load address into TWI address register
|
// load address into TWI address register
|
||||||
TWAR = (address << 1);
|
TWAR = address;
|
||||||
// set the TWCR to enable address matching and enable TWI, clear TWINT, enable TWI interrupt
|
// set the TWCR to enable address matching and enable TWI, clear TWINT, enable TWI interrupt
|
||||||
TWCR = (1 << TWIE) | (1 << TWEA) | (1 << TWINT) | (1 << TWEN);
|
TWCR = (1 << TWIE) | (1 << TWEA) | (1 << TWINT) | (1 << TWEN);
|
||||||
}
|
}
|
||||||
|
|
|
@ -109,7 +109,7 @@ uint8_t i2c_readReg(uint8_t devaddr, uint8_t* regaddr, uint8_t* data, uint16_t l
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is usually not needed. It releases the driver to allow pins to become GPIO again.
|
// This is usually not needed. It releases the driver to allow pins to become GPIO again.
|
||||||
uint8_t i2c_stop(uint16_t timeout)
|
uint8_t i2c_stop(void)
|
||||||
{
|
{
|
||||||
i2cStop(&I2C_DRIVER);
|
i2cStop(&I2C_DRIVER);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -455,10 +455,10 @@ i2c_status_t i2c_transaction(uint8_t address, uint32_t mask, uint8_t col_offset)
|
||||||
matrix[MATRIX_ROWS - 1] |= ((uint32_t)err << (MATRIX_COLS_SCANNED + col_offset)); //add new bits at the end
|
matrix[MATRIX_ROWS - 1] |= ((uint32_t)err << (MATRIX_COLS_SCANNED + col_offset)); //add new bits at the end
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
i2c_stop(10);
|
i2c_stop();
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_stop(10);
|
i2c_stop();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
|
@ -128,7 +128,7 @@ uint8_t init_mcp23018(void) {
|
||||||
mcp23018_status = i2c_write(IODIRA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(IODIRA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
mcp23018_status = i2c_write(0b00000000, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0b00000000, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
mcp23018_status = i2c_write(0b00111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0b00111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
|
|
||||||
// set pull-up
|
// set pull-up
|
||||||
// - unused : on : 1
|
// - unused : on : 1
|
||||||
|
@ -140,7 +140,7 @@ uint8_t init_mcp23018(void) {
|
||||||
mcp23018_status = i2c_write(0b00111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0b00111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
|
|
||||||
#ifdef LEFT_LEDS
|
#ifdef LEFT_LEDS
|
||||||
if (!mcp23018_status) mcp23018_status = ergodox_left_leds_update();
|
if (!mcp23018_status) mcp23018_status = ergodox_left_leds_update();
|
||||||
|
@ -179,7 +179,7 @@ uint8_t ergodox_left_leds_update(void) {
|
||||||
if (mcp23018_status) goto out;
|
if (mcp23018_status) goto out;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
return mcp23018_status;
|
return mcp23018_status;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -309,7 +309,7 @@ static matrix_row_t read_cols(uint8_t row)
|
||||||
data = ~((uint8_t)mcp23018_status);
|
data = ~((uint8_t)mcp23018_status);
|
||||||
mcp23018_status = I2C_STATUS_SUCCESS;
|
mcp23018_status = I2C_STATUS_SUCCESS;
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -362,7 +362,7 @@ static void select_row(uint8_t row)
|
||||||
mcp23018_status = i2c_write(GPIOA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(GPIOA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
mcp23018_status = i2c_write(0xFF & ~(1<<row), ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0xFF & ~(1<<row), ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// select on teensy
|
// select on teensy
|
||||||
|
|
|
@ -47,7 +47,7 @@ uint8_t init_mcp23018(void) {
|
||||||
mcp23018_status = i2c_write(IODIRA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(IODIRA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
mcp23018_status = i2c_write(0b10000000, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0b10000000, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
mcp23018_status = i2c_write(0b11111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0b11111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
|
|
||||||
// set pull-up
|
// set pull-up
|
||||||
// - unused : on : 1
|
// - unused : on : 1
|
||||||
|
@ -59,7 +59,7 @@ uint8_t init_mcp23018(void) {
|
||||||
mcp23018_status = i2c_write(0b11111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0b11111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
// SREG=sreg_prev;
|
// SREG=sreg_prev;
|
||||||
//uprintf("Init %x\n", mcp23018_status);
|
//uprintf("Init %x\n", mcp23018_status);
|
||||||
return mcp23018_status;
|
return mcp23018_status;
|
||||||
|
|
|
@ -394,7 +394,7 @@ static matrix_row_t read_cols(uint8_t row)
|
||||||
data = ~((uint8_t)mcp23018_status);
|
data = ~((uint8_t)mcp23018_status);
|
||||||
mcp23018_status = I2C_STATUS_SUCCESS;
|
mcp23018_status = I2C_STATUS_SUCCESS;
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
|
|
||||||
#ifdef DEBUG_MATRIX
|
#ifdef DEBUG_MATRIX
|
||||||
if (data != 0x00) xprintf("I2C: %d\n", data);
|
if (data != 0x00) xprintf("I2C: %d\n", data);
|
||||||
|
@ -444,7 +444,7 @@ static void select_row(uint8_t row)
|
||||||
mcp23018_status = i2c_write(GPIOA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(GPIOA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
mcp23018_status = i2c_write(0xFF & ~(1<<row), ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
mcp23018_status = i2c_write(0xFF & ~(1<<row), ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out;
|
||||||
out:
|
out:
|
||||||
i2c_stop(ERGODOX_EZ_I2C_TIMEOUT);
|
i2c_stop();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Output low(DDR:1, PORT:0) to select
|
// Output low(DDR:1, PORT:0) to select
|
||||||
|
|
|
@ -29,10 +29,6 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
#include "backlight.h"
|
#include "backlight.h"
|
||||||
#include "quantum.h"
|
#include "quantum.h"
|
||||||
|
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
#include "split_flags.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef MIDI_ENABLE
|
#ifdef MIDI_ENABLE
|
||||||
#include "process_midi.h"
|
#include "process_midi.h"
|
||||||
#endif
|
#endif
|
||||||
|
@ -138,39 +134,21 @@ action_t action_for_key(uint8_t layer, keypos_t key)
|
||||||
#ifdef BACKLIGHT_ENABLE
|
#ifdef BACKLIGHT_ENABLE
|
||||||
case BL_ON:
|
case BL_ON:
|
||||||
action.code = ACTION_BACKLIGHT_ON();
|
action.code = ACTION_BACKLIGHT_ON();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
case BL_OFF:
|
case BL_OFF:
|
||||||
action.code = ACTION_BACKLIGHT_OFF();
|
action.code = ACTION_BACKLIGHT_OFF();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
case BL_DEC:
|
case BL_DEC:
|
||||||
action.code = ACTION_BACKLIGHT_DECREASE();
|
action.code = ACTION_BACKLIGHT_DECREASE();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
case BL_INC:
|
case BL_INC:
|
||||||
action.code = ACTION_BACKLIGHT_INCREASE();
|
action.code = ACTION_BACKLIGHT_INCREASE();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
case BL_TOGG:
|
case BL_TOGG:
|
||||||
action.code = ACTION_BACKLIGHT_TOGGLE();
|
action.code = ACTION_BACKLIGHT_TOGGLE();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
case BL_STEP:
|
case BL_STEP:
|
||||||
action.code = ACTION_BACKLIGHT_STEP();
|
action.code = ACTION_BACKLIGHT_STEP();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
#ifdef SWAP_HANDS_ENABLE
|
#ifdef SWAP_HANDS_ENABLE
|
||||||
|
|
|
@ -360,9 +360,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_toggle();
|
rgblight_toggle();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_MODE_FORWARD:
|
case RGB_MODE_FORWARD:
|
||||||
|
@ -374,9 +371,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
else {
|
else {
|
||||||
rgblight_step();
|
rgblight_step();
|
||||||
}
|
}
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_MODE_REVERSE:
|
case RGB_MODE_REVERSE:
|
||||||
|
@ -388,9 +382,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
else {
|
else {
|
||||||
rgblight_step_reverse();
|
rgblight_step_reverse();
|
||||||
}
|
}
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_HUI:
|
case RGB_HUI:
|
||||||
|
@ -401,9 +392,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_increase_hue();
|
rgblight_increase_hue();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_HUD:
|
case RGB_HUD:
|
||||||
|
@ -414,9 +402,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_decrease_hue();
|
rgblight_decrease_hue();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_SAI:
|
case RGB_SAI:
|
||||||
|
@ -427,9 +412,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_increase_sat();
|
rgblight_increase_sat();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_SAD:
|
case RGB_SAD:
|
||||||
|
@ -440,9 +422,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_decrease_sat();
|
rgblight_decrease_sat();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_VAI:
|
case RGB_VAI:
|
||||||
|
@ -453,9 +432,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_increase_val();
|
rgblight_increase_val();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_VAD:
|
case RGB_VAD:
|
||||||
|
@ -466,9 +442,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
if (!record->event.pressed) {
|
if (!record->event.pressed) {
|
||||||
#endif
|
#endif
|
||||||
rgblight_decrease_val();
|
rgblight_decrease_val();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_SPI:
|
case RGB_SPI:
|
||||||
|
@ -484,9 +457,6 @@ bool process_record_quantum(keyrecord_t *record) {
|
||||||
case RGB_MODE_PLAIN:
|
case RGB_MODE_PLAIN:
|
||||||
if (record->event.pressed) {
|
if (record->event.pressed) {
|
||||||
rgblight_mode(RGBLIGHT_MODE_STATIC_LIGHT);
|
rgblight_mode(RGBLIGHT_MODE_STATIC_LIGHT);
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case RGB_MODE_BREATHE:
|
case RGB_MODE_BREATHE:
|
||||||
|
|
|
@ -44,10 +44,6 @@
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
#include "split_flags.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef RGB_MATRIX_ENABLE
|
#ifdef RGB_MATRIX_ENABLE
|
||||||
#include "rgb_matrix.h"
|
#include "rgb_matrix.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,184 +0,0 @@
|
||||||
#include <util/twi.h>
|
|
||||||
#include <avr/io.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <avr/interrupt.h>
|
|
||||||
#include <util/twi.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include "i2c.h"
|
|
||||||
#include "split_flags.h"
|
|
||||||
|
|
||||||
// Limits the amount of we wait for any one i2c transaction.
|
|
||||||
// Since were running SCL line 100kHz (=> 10μs/bit), and each transactions is
|
|
||||||
// 9 bits, a single transaction will take around 90μs to complete.
|
|
||||||
//
|
|
||||||
// (F_CPU/SCL_CLOCK) => # of μC cycles to transfer a bit
|
|
||||||
// poll loop takes at least 8 clock cycles to execute
|
|
||||||
#define I2C_LOOP_TIMEOUT (9+1)*(F_CPU/SCL_CLOCK)/8
|
|
||||||
|
|
||||||
#define BUFFER_POS_INC() (slave_buffer_pos = (slave_buffer_pos+1)%SLAVE_BUFFER_SIZE)
|
|
||||||
|
|
||||||
volatile uint8_t i2c_slave_buffer[SLAVE_BUFFER_SIZE];
|
|
||||||
|
|
||||||
static volatile uint8_t slave_buffer_pos;
|
|
||||||
static volatile bool slave_has_register_set = false;
|
|
||||||
|
|
||||||
// Wait for an i2c operation to finish
|
|
||||||
inline static
|
|
||||||
void i2c_delay(void) {
|
|
||||||
uint16_t lim = 0;
|
|
||||||
while(!(TWCR & (1<<TWINT)) && lim < I2C_LOOP_TIMEOUT)
|
|
||||||
lim++;
|
|
||||||
|
|
||||||
// easier way, but will wait slightly longer
|
|
||||||
// _delay_us(100);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Setup twi to run at 100kHz
|
|
||||||
void i2c_master_init(void) {
|
|
||||||
// no prescaler
|
|
||||||
TWSR = 0;
|
|
||||||
// Set TWI clock frequency to SCL_CLOCK. Need TWBR>10.
|
|
||||||
// Check datasheets for more info.
|
|
||||||
TWBR = ((F_CPU/SCL_CLOCK)-16)/2;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start a transaction with the given i2c slave address. The direction of the
|
|
||||||
// transfer is set with I2C_READ and I2C_WRITE.
|
|
||||||
// returns: 0 => success
|
|
||||||
// 1 => error
|
|
||||||
uint8_t i2c_master_start(uint8_t address) {
|
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWSTA);
|
|
||||||
|
|
||||||
i2c_delay();
|
|
||||||
|
|
||||||
// check that we started successfully
|
|
||||||
if ( (TW_STATUS != TW_START) && (TW_STATUS != TW_REP_START))
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
TWDR = address;
|
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN);
|
|
||||||
|
|
||||||
i2c_delay();
|
|
||||||
|
|
||||||
if ( (TW_STATUS != TW_MT_SLA_ACK) && (TW_STATUS != TW_MR_SLA_ACK) )
|
|
||||||
return 1; // slave did not acknowledge
|
|
||||||
else
|
|
||||||
return 0; // success
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Finish the i2c transaction.
|
|
||||||
void i2c_master_stop(void) {
|
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWSTO);
|
|
||||||
|
|
||||||
uint16_t lim = 0;
|
|
||||||
while(!(TWCR & (1<<TWSTO)) && lim < I2C_LOOP_TIMEOUT)
|
|
||||||
lim++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write one byte to the i2c slave.
|
|
||||||
// returns 0 => slave ACK
|
|
||||||
// 1 => slave NACK
|
|
||||||
uint8_t i2c_master_write(uint8_t data) {
|
|
||||||
TWDR = data;
|
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN);
|
|
||||||
|
|
||||||
i2c_delay();
|
|
||||||
|
|
||||||
// check if the slave acknowledged us
|
|
||||||
return (TW_STATUS == TW_MT_DATA_ACK) ? 0 : 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t i2c_master_write_data(void *const TXdata, uint8_t dataLen) {
|
|
||||||
|
|
||||||
uint8_t *data = (uint8_t *)TXdata;
|
|
||||||
int err = 0;
|
|
||||||
|
|
||||||
for (int i = 0; i < dataLen; i++) {
|
|
||||||
err = i2c_master_write(data[i]);
|
|
||||||
|
|
||||||
if ( err )
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
|
|
||||||
return err;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read one byte from the i2c slave. If ack=1 the slave is acknowledged,
|
|
||||||
// if ack=0 the acknowledge bit is not set.
|
|
||||||
// returns: byte read from i2c device
|
|
||||||
uint8_t i2c_master_read(int ack) {
|
|
||||||
TWCR = (1<<TWINT) | (1<<TWEN) | (ack<<TWEA);
|
|
||||||
|
|
||||||
i2c_delay();
|
|
||||||
return TWDR;
|
|
||||||
}
|
|
||||||
|
|
||||||
void i2c_reset_state(void) {
|
|
||||||
TWCR = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void i2c_slave_init(uint8_t address) {
|
|
||||||
TWAR = address << 0; // slave i2c address
|
|
||||||
// TWEN - twi enable
|
|
||||||
// TWEA - enable address acknowledgement
|
|
||||||
// TWINT - twi interrupt flag
|
|
||||||
// TWIE - enable the twi interrupt
|
|
||||||
TWCR = (1<<TWIE) | (1<<TWEA) | (1<<TWINT) | (1<<TWEN);
|
|
||||||
}
|
|
||||||
|
|
||||||
ISR(TWI_vect);
|
|
||||||
|
|
||||||
ISR(TWI_vect) {
|
|
||||||
uint8_t ack = 1;
|
|
||||||
switch(TW_STATUS) {
|
|
||||||
case TW_SR_SLA_ACK:
|
|
||||||
// this device has been addressed as a slave receiver
|
|
||||||
slave_has_register_set = false;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TW_SR_DATA_ACK:
|
|
||||||
// this device has received data as a slave receiver
|
|
||||||
// The first byte that we receive in this transaction sets the location
|
|
||||||
// of the read/write location of the slaves memory that it exposes over
|
|
||||||
// i2c. After that, bytes will be written at slave_buffer_pos, incrementing
|
|
||||||
// slave_buffer_pos after each write.
|
|
||||||
if(!slave_has_register_set) {
|
|
||||||
slave_buffer_pos = TWDR;
|
|
||||||
// don't acknowledge the master if this memory loctaion is out of bounds
|
|
||||||
if ( slave_buffer_pos >= SLAVE_BUFFER_SIZE ) {
|
|
||||||
ack = 0;
|
|
||||||
slave_buffer_pos = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
slave_has_register_set = true;
|
|
||||||
} else {
|
|
||||||
i2c_slave_buffer[slave_buffer_pos] = TWDR;
|
|
||||||
|
|
||||||
if ( slave_buffer_pos == I2C_BACKLIT_START) {
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
} else if ( slave_buffer_pos == (I2C_RGB_START+3)) {
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
BUFFER_POS_INC();
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TW_ST_SLA_ACK:
|
|
||||||
case TW_ST_DATA_ACK:
|
|
||||||
// master has addressed this device as a slave transmitter and is
|
|
||||||
// requesting data.
|
|
||||||
TWDR = i2c_slave_buffer[slave_buffer_pos];
|
|
||||||
BUFFER_POS_INC();
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TW_BUS_ERROR: // something went wrong, reset twi state
|
|
||||||
TWCR = 0;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
// Reset everything, so we are ready for the next TWI interrupt
|
|
||||||
TWCR |= (1<<TWIE) | (1<<TWINT) | (ack<<TWEA) | (1<<TWEN);
|
|
||||||
}
|
|
|
@ -1,59 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
#ifndef F_CPU
|
|
||||||
#define F_CPU 16000000UL
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define I2C_READ 1
|
|
||||||
#define I2C_WRITE 0
|
|
||||||
|
|
||||||
#define I2C_ACK 1
|
|
||||||
#define I2C_NACK 0
|
|
||||||
|
|
||||||
// Address location defines (Keymap should be last, as it's size is dynamic)
|
|
||||||
#define I2C_BACKLIT_START 0x00
|
|
||||||
// Need 4 bytes for RGB (32 bit)
|
|
||||||
#define I2C_RGB_START 0x01
|
|
||||||
#define I2C_KEYMAP_START 0x06
|
|
||||||
|
|
||||||
// Slave buffer (8bit per)
|
|
||||||
// Rows per hand + backlit space + rgb space
|
|
||||||
// TODO : Make this dynamically sized
|
|
||||||
#define SLAVE_BUFFER_SIZE 0x20
|
|
||||||
|
|
||||||
// i2c SCL clock frequency
|
|
||||||
#ifndef SCL_CLOCK
|
|
||||||
#define SCL_CLOCK 100000L
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Support 8bits right now (8 cols) will need to edit to take higher (code exists in delta split?)
|
|
||||||
extern volatile uint8_t i2c_slave_buffer[SLAVE_BUFFER_SIZE];
|
|
||||||
|
|
||||||
void i2c_master_init(void);
|
|
||||||
uint8_t i2c_master_start(uint8_t address);
|
|
||||||
void i2c_master_stop(void);
|
|
||||||
uint8_t i2c_master_write(uint8_t data);
|
|
||||||
uint8_t i2c_master_write_data(void *const TXdata, uint8_t dataLen);
|
|
||||||
uint8_t i2c_master_read(int);
|
|
||||||
void i2c_reset_state(void);
|
|
||||||
void i2c_slave_init(uint8_t address);
|
|
||||||
|
|
||||||
|
|
||||||
static inline unsigned char i2c_start_read(unsigned char addr) {
|
|
||||||
return i2c_master_start((addr << 1) | I2C_READ);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline unsigned char i2c_start_write(unsigned char addr) {
|
|
||||||
return i2c_master_start((addr << 1) | I2C_WRITE);
|
|
||||||
}
|
|
||||||
|
|
||||||
// from SSD1306 scrips
|
|
||||||
extern unsigned char i2c_rep_start(unsigned char addr);
|
|
||||||
extern void i2c_start_wait(unsigned char addr);
|
|
||||||
extern unsigned char i2c_readAck(void);
|
|
||||||
extern unsigned char i2c_readNak(void);
|
|
||||||
extern unsigned char i2c_read(unsigned char ack);
|
|
||||||
|
|
||||||
#define i2c_read(ack) (ack) ? i2c_readAck() : i2c_readNak();
|
|
|
@ -25,7 +25,6 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
#include "matrix.h"
|
#include "matrix.h"
|
||||||
#include "split_util.h"
|
#include "split_util.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "split_flags.h"
|
|
||||||
#include "quantum.h"
|
#include "quantum.h"
|
||||||
#include "debounce.h"
|
#include "debounce.h"
|
||||||
#include "transport.h"
|
#include "transport.h"
|
||||||
|
|
|
@ -1,5 +0,0 @@
|
||||||
#include "split_flags.h"
|
|
||||||
|
|
||||||
volatile bool RGB_DIRTY = false;
|
|
||||||
|
|
||||||
volatile bool BACKLIT_DIRTY = false;
|
|
|
@ -1,15 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Global Flags
|
|
||||||
**/
|
|
||||||
|
|
||||||
//RGB Stuff
|
|
||||||
extern volatile bool RGB_DIRTY;
|
|
||||||
|
|
||||||
|
|
||||||
//Backlight Stuff
|
|
||||||
extern volatile bool BACKLIT_DIRTY;
|
|
|
@ -3,7 +3,6 @@
|
||||||
#include "keyboard.h"
|
#include "keyboard.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
#include "split_flags.h"
|
|
||||||
#include "transport.h"
|
#include "transport.h"
|
||||||
#include "quantum.h"
|
#include "quantum.h"
|
||||||
|
|
||||||
|
@ -60,10 +59,6 @@ static void keyboard_master_setup(void) {
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
transport_master_init();
|
transport_master_init();
|
||||||
|
|
||||||
// For master the Backlight info needs to be sent on startup
|
|
||||||
// Otherwise the salve won't start with the proper info until an update
|
|
||||||
BACKLIT_DIRTY = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void keyboard_slave_setup(void)
|
static void keyboard_slave_setup(void)
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
#include "matrix.h"
|
#include "matrix.h"
|
||||||
#include "quantum.h"
|
#include "quantum.h"
|
||||||
|
|
||||||
#define ROWS_PER_HAND (MATRIX_ROWS/2)
|
#define ROWS_PER_HAND (MATRIX_ROWS / 2)
|
||||||
|
|
||||||
#ifdef RGBLIGHT_ENABLE
|
#ifdef RGBLIGHT_ENABLE
|
||||||
# include "rgblight.h"
|
# include "rgblight.h"
|
||||||
|
@ -11,138 +11,75 @@
|
||||||
|
|
||||||
#ifdef BACKLIGHT_ENABLE
|
#ifdef BACKLIGHT_ENABLE
|
||||||
# include "backlight.h"
|
# include "backlight.h"
|
||||||
extern backlight_config_t backlight_config;
|
extern backlight_config_t backlight_config;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if defined(USE_I2C) || defined(EH)
|
#if defined(USE_I2C) || defined(EH)
|
||||||
|
|
||||||
#include "i2c.h"
|
# include "i2c_master.h"
|
||||||
|
# include "i2c_slave.h"
|
||||||
|
|
||||||
#ifndef SLAVE_I2C_ADDRESS
|
# define I2C_BACKLIT_START 0x00
|
||||||
|
// Need 4 bytes for RGB (32 bit)
|
||||||
|
# define I2C_RGB_START 0x01
|
||||||
|
# define I2C_KEYMAP_START 0x05
|
||||||
|
|
||||||
|
# define TIMEOUT 100
|
||||||
|
|
||||||
|
# ifndef SLAVE_I2C_ADDRESS
|
||||||
# define SLAVE_I2C_ADDRESS 0x32
|
# define SLAVE_I2C_ADDRESS 0x32
|
||||||
#endif
|
# endif
|
||||||
|
|
||||||
#if (MATRIX_COLS > 8)
|
|
||||||
# error "Currently only supports 8 COLS"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Get rows from other half over i2c
|
// Get rows from other half over i2c
|
||||||
bool transport_master(matrix_row_t matrix[]) {
|
bool transport_master(matrix_row_t matrix[]) {
|
||||||
int err = 0;
|
i2c_readReg(SLAVE_I2C_ADDRESS, I2C_KEYMAP_START, (void *)matrix, ROWS_PER_HAND * sizeof(matrix_row_t), TIMEOUT);
|
||||||
|
|
||||||
// write backlight info
|
// write backlight info
|
||||||
#ifdef BACKLIGHT_ENABLE
|
# ifdef BACKLIGHT_ENABLE
|
||||||
if (BACKLIT_DIRTY) {
|
static uint8_t prev_level = ~0;
|
||||||
err = i2c_master_start(SLAVE_I2C_ADDRESS + I2C_WRITE);
|
uint8_t level = get_backlight_level();
|
||||||
if (err) { goto i2c_error; }
|
if (level != prev_level) {
|
||||||
|
i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_BACKLIT_START, (void *)&level, sizeof(level), TIMEOUT);
|
||||||
// Backlight location
|
prev_level = level;
|
||||||
err = i2c_master_write(I2C_BACKLIT_START);
|
|
||||||
if (err) { goto i2c_error; }
|
|
||||||
|
|
||||||
// Write backlight
|
|
||||||
i2c_master_write(get_backlight_level());
|
|
||||||
|
|
||||||
BACKLIT_DIRTY = false;
|
|
||||||
}
|
}
|
||||||
#endif
|
# endif
|
||||||
|
|
||||||
err = i2c_master_start(SLAVE_I2C_ADDRESS + I2C_WRITE);
|
# ifdef RGBLIGHT_ENABLE
|
||||||
if (err) { goto i2c_error; }
|
static uint32_t prev_rgb = ~0;
|
||||||
|
uint32_t rgb = eeconfig_read_rgblight();
|
||||||
// start of matrix stored at I2C_KEYMAP_START
|
if (rgb != prev_rgb) {
|
||||||
err = i2c_master_write(I2C_KEYMAP_START);
|
i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_RGB_START, (void *)&rgb, sizeof(rgb), TIMEOUT);
|
||||||
if (err) { goto i2c_error; }
|
prev_rgb = rgb;
|
||||||
|
|
||||||
// Start read
|
|
||||||
err = i2c_master_start(SLAVE_I2C_ADDRESS + I2C_READ);
|
|
||||||
if (err) { goto i2c_error; }
|
|
||||||
|
|
||||||
if (!err) {
|
|
||||||
int i;
|
|
||||||
for (i = 0; i < ROWS_PER_HAND-1; ++i) {
|
|
||||||
matrix[i] = i2c_master_read(I2C_ACK);
|
|
||||||
}
|
}
|
||||||
matrix[i] = i2c_master_read(I2C_NACK);
|
# endif
|
||||||
i2c_master_stop();
|
|
||||||
} else {
|
|
||||||
i2c_error: // the cable is disconnceted, or something else went wrong
|
|
||||||
i2c_reset_state();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef RGBLIGHT_ENABLE
|
|
||||||
if (RGB_DIRTY) {
|
|
||||||
err = i2c_master_start(SLAVE_I2C_ADDRESS + I2C_WRITE);
|
|
||||||
if (err) { goto i2c_error; }
|
|
||||||
|
|
||||||
// RGB Location
|
|
||||||
err = i2c_master_write(I2C_RGB_START);
|
|
||||||
if (err) { goto i2c_error; }
|
|
||||||
|
|
||||||
uint32_t dword = eeconfig_read_rgblight();
|
|
||||||
|
|
||||||
// Write RGB
|
|
||||||
err = i2c_master_write_data(&dword, 4);
|
|
||||||
if (err) { goto i2c_error; }
|
|
||||||
|
|
||||||
RGB_DIRTY = false;
|
|
||||||
i2c_master_stop();
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void transport_slave(matrix_row_t matrix[]) {
|
void transport_slave(matrix_row_t matrix[]) {
|
||||||
|
for (int i = 0; i < ROWS_PER_HAND * sizeof(matrix_row_t); ++i) {
|
||||||
for (int i = 0; i < ROWS_PER_HAND; ++i)
|
i2c_slave_reg[I2C_KEYMAP_START + i] = matrix[i];
|
||||||
{
|
|
||||||
i2c_slave_buffer[I2C_KEYMAP_START + i] = matrix[i];
|
|
||||||
}
|
|
||||||
// Read Backlight Info
|
|
||||||
#ifdef BACKLIGHT_ENABLE
|
|
||||||
if (BACKLIT_DIRTY)
|
|
||||||
{
|
|
||||||
backlight_set(i2c_slave_buffer[I2C_BACKLIT_START]);
|
|
||||||
BACKLIT_DIRTY = false;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef RGBLIGHT_ENABLE
|
|
||||||
if (RGB_DIRTY)
|
|
||||||
{
|
|
||||||
// Disable interupts (RGB data is big)
|
|
||||||
cli();
|
|
||||||
// Create new DWORD for RGB data
|
|
||||||
uint32_t dword;
|
|
||||||
|
|
||||||
// Fill the new DWORD with the data that was sent over
|
|
||||||
uint8_t * dword_dat = (uint8_t *)(&dword);
|
|
||||||
for (int i = 0; i < 4; i++)
|
|
||||||
{
|
|
||||||
dword_dat[i] = i2c_slave_buffer[I2C_RGB_START + i];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the RGB now with the new data and set RGB_DIRTY to false
|
// Read Backlight Info
|
||||||
rgblight_update_dword(dword);
|
# ifdef BACKLIGHT_ENABLE
|
||||||
RGB_DIRTY = false;
|
backlight_set(i2c_slave_reg[I2C_BACKLIT_START]);
|
||||||
// Re-enable interupts now that RGB is set
|
# endif
|
||||||
sei();
|
|
||||||
}
|
# ifdef RGBLIGHT_ENABLE
|
||||||
#endif
|
uint32_t rgb = *(uint32_t *)(i2c_slave_reg + I2C_RGB_START);
|
||||||
|
// Update the RGB with the new data
|
||||||
|
rgblight_update_dword(rgb);
|
||||||
|
# endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void transport_master_init(void) {
|
void transport_master_init(void) { i2c_init(); }
|
||||||
i2c_master_init();
|
|
||||||
}
|
|
||||||
|
|
||||||
void transport_slave_init(void) {
|
void transport_slave_init(void) { i2c_slave_init(SLAVE_I2C_ADDRESS); }
|
||||||
i2c_slave_init(SLAVE_I2C_ADDRESS);
|
|
||||||
}
|
|
||||||
|
|
||||||
#else // USE_SERIAL
|
#else // USE_SERIAL
|
||||||
|
|
||||||
#include "serial.h"
|
# include "serial.h"
|
||||||
|
|
||||||
typedef struct _Serial_s2m_buffer_t {
|
typedef struct _Serial_s2m_buffer_t {
|
||||||
// TODO: if MATRIX_COLS > 8 change to uint8_t packed_matrix[] for pack/unpack
|
// TODO: if MATRIX_COLS > 8 change to uint8_t packed_matrix[] for pack/unpack
|
||||||
|
@ -150,19 +87,19 @@ typedef struct _Serial_s2m_buffer_t {
|
||||||
} Serial_s2m_buffer_t;
|
} Serial_s2m_buffer_t;
|
||||||
|
|
||||||
typedef struct _Serial_m2s_buffer_t {
|
typedef struct _Serial_m2s_buffer_t {
|
||||||
#ifdef BACKLIGHT_ENABLE
|
# ifdef BACKLIGHT_ENABLE
|
||||||
uint8_t backlight_level;
|
uint8_t backlight_level;
|
||||||
#endif
|
# endif
|
||||||
#if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
|
# if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
|
||||||
rgblight_config_t rgblight_config; //not yet use
|
rgblight_config_t rgblight_config; // not yet use
|
||||||
//
|
//
|
||||||
// When MCUs on both sides drive their respective RGB LED chains,
|
// When MCUs on both sides drive their respective RGB LED chains,
|
||||||
// it is necessary to synchronize, so it is necessary to communicate RGB information.
|
// it is necessary to synchronize, so it is necessary to communicate RGB
|
||||||
// In that case, define the RGBLIGHT_SPLIT macro.
|
// information. In that case, define the RGBLIGHT_SPLIT macro.
|
||||||
//
|
//
|
||||||
// Otherwise, if the master side MCU drives both sides RGB LED chains,
|
// Otherwise, if the master side MCU drives both sides RGB LED chains,
|
||||||
// there is no need to communicate.
|
// there is no need to communicate.
|
||||||
#endif
|
# endif
|
||||||
} Serial_m2s_buffer_t;
|
} Serial_m2s_buffer_t;
|
||||||
|
|
||||||
volatile Serial_s2m_buffer_t serial_s2m_buffer = {};
|
volatile Serial_s2m_buffer_t serial_s2m_buffer = {};
|
||||||
|
@ -170,20 +107,20 @@ volatile Serial_m2s_buffer_t serial_m2s_buffer = {};
|
||||||
uint8_t volatile status0 = 0;
|
uint8_t volatile status0 = 0;
|
||||||
|
|
||||||
SSTD_t transactions[] = {
|
SSTD_t transactions[] = {
|
||||||
{ (uint8_t *)&status0,
|
{
|
||||||
sizeof(serial_m2s_buffer), (uint8_t *)&serial_m2s_buffer,
|
(uint8_t *)&status0,
|
||||||
sizeof(serial_s2m_buffer), (uint8_t *)&serial_s2m_buffer
|
sizeof(serial_m2s_buffer),
|
||||||
}
|
(uint8_t *)&serial_m2s_buffer,
|
||||||
|
sizeof(serial_s2m_buffer),
|
||||||
|
(uint8_t *)&serial_s2m_buffer,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
void transport_master_init(void)
|
void transport_master_init(void) { soft_serial_initiator_init(transactions, TID_LIMIT(transactions)); }
|
||||||
{ soft_serial_initiator_init(transactions, TID_LIMIT(transactions)); }
|
|
||||||
|
|
||||||
void transport_slave_init(void)
|
void transport_slave_init(void) { soft_serial_target_init(transactions, TID_LIMIT(transactions)); }
|
||||||
{ soft_serial_target_init(transactions, TID_LIMIT(transactions)); }
|
|
||||||
|
|
||||||
bool transport_master(matrix_row_t matrix[]) {
|
bool transport_master(matrix_row_t matrix[]) {
|
||||||
|
|
||||||
if (soft_serial_transaction()) {
|
if (soft_serial_transaction()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -193,32 +130,29 @@ bool transport_master(matrix_row_t matrix[]) {
|
||||||
matrix[i] = serial_s2m_buffer.smatrix[i];
|
matrix[i] = serial_s2m_buffer.smatrix[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
|
# if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
|
||||||
// Code to send RGB over serial goes here (not implemented yet)
|
// Code to send RGB over serial goes here (not implemented yet)
|
||||||
#endif
|
# endif
|
||||||
|
|
||||||
#ifdef BACKLIGHT_ENABLE
|
# ifdef BACKLIGHT_ENABLE
|
||||||
// Write backlight level for slave to read
|
// Write backlight level for slave to read
|
||||||
serial_m2s_buffer.backlight_level = backlight_config.enable ? backlight_config.level : 0;
|
serial_m2s_buffer.backlight_level = backlight_config.enable ? backlight_config.level : 0;
|
||||||
#endif
|
# endif
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void transport_slave(matrix_row_t matrix[]) {
|
void transport_slave(matrix_row_t matrix[]) {
|
||||||
|
|
||||||
// TODO: if MATRIX_COLS > 8 change to pack()
|
// TODO: if MATRIX_COLS > 8 change to pack()
|
||||||
for (int i = 0; i < ROWS_PER_HAND; ++i)
|
for (int i = 0; i < ROWS_PER_HAND; ++i) {
|
||||||
{
|
|
||||||
serial_s2m_buffer.smatrix[i] = matrix[i];
|
serial_s2m_buffer.smatrix[i] = matrix[i];
|
||||||
}
|
}
|
||||||
#ifdef BACKLIGHT_ENABLE
|
# ifdef BACKLIGHT_ENABLE
|
||||||
backlight_set(serial_m2s_buffer.backlight_level);
|
backlight_set(serial_m2s_buffer.backlight_level);
|
||||||
#endif
|
# endif
|
||||||
#if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
|
# if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
|
||||||
// Add serial implementation for RGB here
|
// Add serial implementation for RGB here
|
||||||
#endif
|
# endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -11,9 +11,6 @@
|
||||||
#include "led.h"
|
#include "led.h"
|
||||||
#include "host.h"
|
#include "host.h"
|
||||||
#include "rgblight_reconfig.h"
|
#include "rgblight_reconfig.h"
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
#include "split_flags.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef PROTOCOL_LUFA
|
#ifdef PROTOCOL_LUFA
|
||||||
#include "lufa.h"
|
#include "lufa.h"
|
||||||
|
@ -135,9 +132,6 @@ static void power_down(uint8_t wdto) {
|
||||||
is_suspended = true;
|
is_suspended = true;
|
||||||
rgblight_enabled = rgblight_config.enable;
|
rgblight_enabled = rgblight_config.enable;
|
||||||
rgblight_disable_noeeprom();
|
rgblight_disable_noeeprom();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
suspend_power_down_kb();
|
suspend_power_down_kb();
|
||||||
|
@ -216,9 +210,6 @@ void suspend_wakeup_init(void) {
|
||||||
wait_ms(10);
|
wait_ms(10);
|
||||||
#endif
|
#endif
|
||||||
rgblight_enable_noeeprom();
|
rgblight_enable_noeeprom();
|
||||||
#ifdef SPLIT_KEYBOARD
|
|
||||||
RGB_DIRTY = true;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
#ifdef RGBLIGHT_ANIMATIONS
|
#ifdef RGBLIGHT_ANIMATIONS
|
||||||
rgblight_timer_enable();
|
rgblight_timer_enable();
|
||||||
|
|
Loading…
Reference in a new issue