2018-07-11 20:16:37 +01:00
/*
2019-10-27 10:13:24 +00:00
xsns_29_mcp230xx . ino - Support for I2C MCP23008 / MCP23017 GPIO Expander on Tasmota
2018-07-11 20:16:37 +01:00
2019-12-31 13:23:34 +00:00
Copyright ( C ) 2020 Andre Thomas and Theo Arends
2018-07-11 20:16:37 +01:00
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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# ifdef USE_I2C
# ifdef USE_MCP230xx
/*********************************************************************************************\
MCP23008 / 17 - I2C GPIO EXPANDER
Docs at https : //www.microchip.com/wwwproducts/en/MCP23008
https : //www.microchip.com/wwwproducts/en/MCP23017
2019-11-08 16:48:19 +00:00
I2C Address : 0x20 - 0x26 ( 0x27 is not supported )
2018-08-14 20:45:56 +01:00
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2018-07-11 20:16:37 +01:00
2018-07-12 22:12:01 +01:00
# define XSNS_29 29
2019-11-03 16:54:39 +00:00
# define XI2C_22 22 // See I2CDEVICES.md
2018-07-12 22:12:01 +01:00
2018-07-11 20:16:37 +01:00
/*
Default register locations for MCP23008 - They change for MCP23017 in default bank mode
*/
uint8_t MCP230xx_IODIR = 0x00 ;
uint8_t MCP230xx_GPINTEN = 0x02 ;
uint8_t MCP230xx_IOCON = 0x05 ;
uint8_t MCP230xx_GPPU = 0x06 ;
uint8_t MCP230xx_INTF = 0x07 ;
uint8_t MCP230xx_INTCAP = 0x08 ;
uint8_t MCP230xx_GPIO = 0x09 ;
uint8_t mcp230xx_type = 0 ;
2018-07-21 17:04:36 +01:00
uint8_t mcp230xx_pincount = 0 ;
2020-08-03 17:02:27 +01:00
uint8_t mcp230xx_oldoutpincount = 0 ;
2020-08-04 07:31:50 +01:00
# ifdef USE_MCP230xx_OUTPUT
2020-08-03 17:02:27 +01:00
uint8_t mcp230xx_outpinmapping [ 16 ] ;
2020-08-04 07:31:50 +01:00
# endif
2018-07-22 21:40:26 +01:00
uint8_t mcp230xx_int_en = 0 ;
2018-08-25 20:15:41 +01:00
uint8_t mcp230xx_int_prio_counter = 0 ;
uint8_t mcp230xx_int_counter_en = 0 ;
2018-12-06 22:55:44 +00:00
uint8_t mcp230xx_int_retainer_en = 0 ;
2018-08-25 20:15:41 +01:00
uint8_t mcp230xx_int_sec_counter = 0 ;
uint8_t mcp230xx_int_report_defer_counter [ 16 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ;
uint16_t mcp230xx_int_counter [ 16 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ;
2018-07-11 20:16:37 +01:00
2018-12-06 22:55:44 +00:00
uint8_t mcp230xx_int_retainer [ 16 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ; // Used to store if an interrupt occured that needs to be retained until teleperiod
2018-08-19 00:35:30 +01:00
unsigned long int_millis [ 16 ] ; // To keep track of millis() since last interrupt
2018-08-17 22:19:14 +01:00
const char MCP230XX_SENSOR_RESPONSE [ ] PROGMEM = " { \" Sensor29_D%i \" :{ \" MODE \" :%i, \" PULL_UP \" : \" %s \" , \" INT_MODE \" : \" %s \" , \" STATE \" : \" %s \" }} " ;
2018-07-13 09:04:08 +01:00
2018-08-25 20:15:41 +01:00
const char MCP230XX_INTCFG_RESPONSE [ ] PROGMEM = " { \" MCP230xx_INT%s \" :{ \" D_%i \" :%i}} " ;
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-14 16:45:06 +01:00
const char MCP230XX_CMND_RESPONSE [ ] PROGMEM = " { \" S29cmnd_D%i \" :{ \" COMMAND \" : \" %s \" , \" STATE \" : \" %s \" }} " ;
2018-08-09 10:46:14 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-08-25 20:15:41 +01:00
void MCP230xx_CheckForIntCounter ( void ) {
uint8_t en = 0 ;
2019-06-30 15:44:36 +01:00
for ( uint32_t ca = 0 ; ca < 16 ; ca + + ) {
2018-08-25 20:15:41 +01:00
if ( Settings . mcp230xx_config [ ca ] . int_count_en ) {
en = 1 ;
}
}
if ( ! Settings . mcp230xx_int_timer ) en = 0 ;
mcp230xx_int_counter_en = en ;
if ( ! mcp230xx_int_counter_en ) { // Interrupt counters are disabled, so we clear all the counters
2019-06-30 15:44:36 +01:00
for ( uint32_t ca = 0 ; ca < 16 ; ca + + ) {
2018-08-25 20:15:41 +01:00
mcp230xx_int_counter [ ca ] = 0 ;
}
}
}
2019-03-08 14:15:42 +00:00
2018-12-06 22:55:44 +00:00
void MCP230xx_CheckForIntRetainer ( void ) {
uint8_t en = 0 ;
2019-06-30 15:44:36 +01:00
for ( uint32_t ca = 0 ; ca < 16 ; ca + + ) {
2018-12-06 22:55:44 +00:00
if ( Settings . mcp230xx_config [ ca ] . int_retain_flag ) {
en = 1 ;
}
}
mcp230xx_int_retainer_en = en ;
if ( ! mcp230xx_int_retainer_en ) { // Interrupt counters are disabled, so we clear all the counters
2019-06-30 15:44:36 +01:00
for ( uint32_t ca = 0 ; ca < 16 ; ca + + ) {
2018-12-06 22:55:44 +00:00
mcp230xx_int_retainer [ ca ] = 0 ;
}
}
}
2018-08-25 20:15:41 +01:00
2018-08-14 00:13:56 +01:00
const char * ConvertNumTxt ( uint8_t statu , uint8_t pinmod = 0 ) {
# ifdef USE_MCP230xx_OUTPUT
2018-08-25 10:38:32 +01:00
if ( ( 6 = = pinmod ) & & ( statu < 2 ) ) { statu = abs ( statu - 1 ) ; }
2018-08-14 00:13:56 +01:00
# endif // USE_MCP230xx_OUTPUT
switch ( statu ) {
case 0 :
return " OFF " ;
break ;
case 1 :
return " ON " ;
break ;
# ifdef USE_MCP230xx_OUTPUT
case 2 :
return " TOGGLE " ;
break ;
# endif // USE_MCP230xx_OUTPUT
}
2018-08-26 12:38:43 +01:00
return " " ;
2018-08-14 00:13:56 +01:00
}
2018-08-16 23:18:45 +01:00
const char * IntModeTxt ( uint8_t intmo ) {
switch ( intmo ) {
case 0 :
return " ALL " ;
break ;
case 1 :
return " EVENT " ;
break ;
case 2 :
return " TELE " ;
break ;
case 3 :
return " DISABLED " ;
break ;
}
2018-08-26 12:38:43 +01:00
return " " ;
2018-08-16 23:18:45 +01:00
}
2018-07-11 20:16:37 +01:00
uint8_t MCP230xx_readGPIO ( uint8_t port ) {
2018-09-16 20:36:56 +01:00
return I2cRead8 ( USE_MCP230xx_ADDR , MCP230xx_GPIO + port ) ;
2018-07-11 20:16:37 +01:00
}
2019-11-11 16:32:44 +00:00
void MCP230xx_ApplySettings ( void )
{
2018-07-22 21:40:26 +01:00
uint8_t int_en = 0 ;
2019-11-11 16:32:44 +00:00
for ( uint32_t mcp230xx_port = 0 ; mcp230xx_port < mcp230xx_type ; mcp230xx_port + + ) {
2018-07-23 18:46:41 +01:00
uint8_t reg_gppu = 0 ;
uint8_t reg_gpinten = 0 ;
uint8_t reg_iodir = 0xFF ;
2018-08-14 00:13:56 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-09 10:46:14 +01:00
uint8_t reg_portpins = 0x00 ;
2018-08-15 21:07:25 +01:00
# endif // USE_MCP230xx_OUTPUT
2019-06-30 15:44:36 +01:00
for ( uint32_t idx = 0 ; idx < 8 ; idx + + ) {
2018-07-23 18:46:41 +01:00
switch ( Settings . mcp230xx_config [ idx + ( mcp230xx_port * 8 ) ] . pinmode ) {
2018-07-21 21:04:18 +01:00
case 0 . . . 1 :
2018-07-23 18:46:41 +01:00
reg_iodir | = ( 1 < < idx ) ;
2018-07-21 21:04:18 +01:00
break ;
case 2 . . . 4 :
2018-07-23 18:46:41 +01:00
reg_iodir | = ( 1 < < idx ) ;
reg_gpinten | = ( 1 < < idx ) ;
2018-08-25 10:38:32 +01:00
int_en = 1 ;
2018-07-21 21:04:18 +01:00
break ;
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-14 00:13:56 +01:00
case 5 . . . 6 :
2018-08-09 10:46:14 +01:00
reg_iodir & = ~ ( 1 < < idx ) ;
2019-11-03 11:33:36 +00:00
if ( Settings . flag . save_state ) { // SetOption0 - Save power state and use after restart - Firmware configuration wants us to use the last pin state
2018-08-11 23:32:56 +01:00
reg_portpins | = ( Settings . mcp230xx_config [ idx + ( mcp230xx_port * 8 ) ] . saved_state < < idx ) ;
2018-08-09 10:46:14 +01:00
} else {
if ( Settings . mcp230xx_config [ idx + ( mcp230xx_port * 8 ) ] . pullup ) {
reg_portpins | = ( 1 < < idx ) ;
}
}
break ;
2018-08-14 00:13:56 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-07-21 21:04:18 +01:00
default :
break ;
}
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-24 20:24:43 +01:00
if ( ( Settings . mcp230xx_config [ idx + ( mcp230xx_port * 8 ) ] . pullup ) & & ( Settings . mcp230xx_config [ idx + ( mcp230xx_port * 8 ) ] . pinmode < 5 ) ) {
2018-08-09 10:46:14 +01:00
reg_gppu | = ( 1 < < idx ) ;
}
2018-08-14 20:36:10 +01:00
# else // not USE_MCP230xx_OUTPUT
2018-07-23 18:46:41 +01:00
if ( Settings . mcp230xx_config [ idx + ( mcp230xx_port * 8 ) ] . pullup ) {
reg_gppu | = ( 1 < < idx ) ;
2018-07-11 20:16:37 +01:00
}
2018-08-14 20:36:10 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-07-11 20:16:37 +01:00
}
2018-09-16 20:36:56 +01:00
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_GPPU + mcp230xx_port , reg_gppu ) ;
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_GPINTEN + mcp230xx_port , reg_gpinten ) ;
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_IODIR + mcp230xx_port , reg_iodir ) ;
2018-08-14 00:13:56 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-09-16 20:36:56 +01:00
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_GPIO + mcp230xx_port , reg_portpins ) ;
2018-08-14 20:36:10 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-07-11 20:16:37 +01:00
}
2020-10-30 11:29:48 +00:00
TasmotaGlobal . devices_present - = mcp230xx_oldoutpincount ;
2020-08-03 17:02:27 +01:00
mcp230xx_oldoutpincount = 0 ;
2019-06-30 15:44:36 +01:00
for ( uint32_t idx = 0 ; idx < mcp230xx_pincount ; idx + + ) {
2020-08-03 17:02:27 +01:00
if ( Settings . mcp230xx_config [ idx ] . pinmode > = 5 ) {
mcp230xx_outpinmapping [ mcp230xx_oldoutpincount ] = idx ;
mcp230xx_oldoutpincount + + ;
}
2018-08-19 00:35:30 +01:00
int_millis [ idx ] = millis ( ) ;
}
2020-10-30 11:29:48 +00:00
TasmotaGlobal . devices_present + = mcp230xx_oldoutpincount ;
2018-08-25 10:38:32 +01:00
mcp230xx_int_en = int_en ;
2018-12-06 22:55:44 +00:00
MCP230xx_CheckForIntCounter ( ) ; // update register on whether or not we should be counting interrupts
MCP230xx_CheckForIntRetainer ( ) ; // update register on whether or not we should be retaining interrupt events for teleperiod
2018-07-11 20:16:37 +01:00
}
2018-09-20 20:03:33 +01:00
void MCP230xx_Detect ( void )
2018-07-11 20:16:37 +01:00
{
2019-11-11 16:32:44 +00:00
if ( I2cActive ( USE_MCP230xx_ADDR ) ) { return ; }
2018-08-20 23:01:42 +01:00
2018-08-14 20:45:56 +01:00
uint8_t buffer ;
2018-07-11 20:16:37 +01:00
2018-09-16 20:36:56 +01:00
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_IOCON , 0x80 ) ; // attempt to set bank mode - this will only work on MCP23017, so its the best way to detect the different chips 23008 vs 23017
if ( I2cValidRead8 ( & buffer , USE_MCP230xx_ADDR , MCP230xx_IOCON ) ) {
if ( 0x00 = = buffer ) {
mcp230xx_type = 1 ; // We have a MCP23008
2019-11-09 17:34:22 +00:00
I2cSetActiveFound ( USE_MCP230xx_ADDR , " MCP23008 " ) ;
2018-09-16 20:36:56 +01:00
mcp230xx_pincount = 8 ;
MCP230xx_ApplySettings ( ) ;
} else {
if ( 0x80 = = buffer ) {
mcp230xx_type = 2 ; // We have a MCP23017
2019-11-09 17:34:22 +00:00
I2cSetActiveFound ( USE_MCP230xx_ADDR , " MCP23017 " ) ;
2018-09-16 20:36:56 +01:00
mcp230xx_pincount = 16 ;
// Reset bank mode to 0
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_IOCON , 0x00 ) ;
// Update register locations for MCP23017
MCP230xx_GPINTEN = 0x04 ;
MCP230xx_GPPU = 0x0C ;
MCP230xx_INTF = 0x0E ;
MCP230xx_INTCAP = 0x10 ;
MCP230xx_GPIO = 0x12 ;
2018-07-11 20:16:37 +01:00
MCP230xx_ApplySettings ( ) ;
}
}
}
}
2018-08-26 12:38:43 +01:00
void MCP230xx_CheckForInterrupt ( void ) {
2018-07-11 20:16:37 +01:00
uint8_t intf ;
uint8_t mcp230xx_intcap = 0 ;
uint8_t report_int ;
2019-11-11 16:32:44 +00:00
for ( uint32_t mcp230xx_port = 0 ; mcp230xx_port < mcp230xx_type ; mcp230xx_port + + ) {
2018-09-16 20:36:56 +01:00
if ( I2cValidRead8 ( & intf , USE_MCP230xx_ADDR , MCP230xx_INTF + mcp230xx_port ) ) {
2018-07-11 20:16:37 +01:00
if ( intf > 0 ) {
2018-09-16 20:36:56 +01:00
if ( I2cValidRead8 ( & mcp230xx_intcap , USE_MCP230xx_ADDR , MCP230xx_INTCAP + mcp230xx_port ) ) {
2019-06-30 15:44:36 +01:00
for ( uint32_t intp = 0 ; intp < 8 ; intp + + ) {
2018-07-11 20:16:37 +01:00
if ( ( intf > > intp ) & 0x01 ) { // we know which pin caused interrupt
report_int = 0 ;
2018-07-23 18:46:41 +01:00
if ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . pinmode > 1 ) {
2018-07-29 14:25:00 +01:00
switch ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . pinmode ) {
2018-07-21 17:04:36 +01:00
case 2 :
report_int = 1 ;
break ;
case 3 :
if ( ( ( mcp230xx_intcap > > intp ) & 0x01 ) = = 0 ) report_int = 1 ; // Int on LOW
break ;
case 4 :
if ( ( ( mcp230xx_intcap > > intp ) & 0x01 ) = = 1 ) report_int = 1 ; // Int on HIGH
break ;
default :
break ;
}
2018-08-25 20:15:41 +01:00
// Check for interrupt counter
if ( ( mcp230xx_int_counter_en ) & & ( report_int ) ) { // We may have some counting to do
if ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . int_count_en ) { // Indeed, for this pin
mcp230xx_int_counter [ intp + ( mcp230xx_port * 8 ) ] + + ;
}
}
// check for interrupt defer on this pin
if ( report_int ) {
if ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . int_report_defer ) {
mcp230xx_int_report_defer_counter [ intp + ( mcp230xx_port * 8 ) ] + + ;
if ( mcp230xx_int_report_defer_counter [ intp + ( mcp230xx_port * 8 ) ] > = Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . int_report_defer ) {
mcp230xx_int_report_defer_counter [ intp + ( mcp230xx_port * 8 ) ] = 0 ;
} else {
report_int = 0 ; // defer int report for now
}
}
}
2018-12-06 22:55:44 +00:00
// check if interrupt retain is used, if it is for this pin then we do not report immediately as it will be reported in teleperiod
if ( report_int ) {
if ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . int_retain_flag ) {
mcp230xx_int_retainer [ intp + ( mcp230xx_port * 8 ) ] = 1 ;
report_int = 0 ; // do not report for now
}
}
2018-08-25 20:15:41 +01:00
if ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . int_count_en ) { // We do not want to report via tele or event if counting is enabled
report_int = 0 ;
}
2018-07-23 18:46:41 +01:00
if ( report_int ) {
2018-08-15 12:16:12 +01:00
bool int_tele = false ;
bool int_event = false ;
2018-09-15 17:59:45 +01:00
unsigned long millis_now = millis ( ) ;
unsigned long millis_since_last_int = millis_now - int_millis [ intp + ( mcp230xx_port * 8 ) ] ;
int_millis [ intp + ( mcp230xx_port * 8 ) ] = millis_now ;
2018-08-15 12:16:12 +01:00
switch ( Settings . mcp230xx_config [ intp + ( mcp230xx_port * 8 ) ] . int_report_mode ) {
case 0 :
int_tele = true ;
int_event = true ;
break ;
case 1 :
int_event = true ;
break ;
case 2 :
int_tele = true ;
break ;
}
if ( int_tele ) {
2019-09-04 17:06:34 +01:00
ResponseTime_P ( PSTR ( " , \" MCP230XX_INT \" :{ \" D%i \" :%i, \" MS \" :%lu}} " ) ,
2019-07-11 13:09:42 +01:00
intp + ( mcp230xx_port * 8 ) , ( ( mcp230xx_intcap > > intp ) & 0x01 ) , millis_since_last_int ) ;
2018-09-15 14:56:56 +01:00
MqttPublishPrefixTopic_P ( RESULT_OR_STAT , PSTR ( " MCP230XX_INT " ) ) ;
2020-05-16 16:30:43 +01:00
if ( Settings . flag3 . hass_tele_on_power ) { // SetOption59 - Send tele/%topic%/SENSOR in addition to stat/%topic%/RESULT
MqttPublishSensor ( ) ;
}
2018-08-15 12:16:12 +01:00
}
if ( int_event ) {
2018-09-10 17:32:41 +01:00
char command [ 19 ] ; // Theoretical max = 'event MCPINT_D16=1' so 18 + 1 (for the \n)
2018-08-15 12:16:12 +01:00
sprintf ( command , " event MCPINT_D%i=%i " , intp + ( mcp230xx_port * 8 ) , ( ( mcp230xx_intcap > > intp ) & 0x01 ) ) ;
ExecuteCommand ( command , SRC_RULE ) ;
}
2018-07-23 18:46:41 +01:00
}
2018-07-11 20:16:37 +01:00
}
}
}
}
}
}
}
}
2019-01-28 13:08:33 +00:00
void MCP230xx_Show ( bool json )
2018-07-11 20:16:37 +01:00
{
2019-11-11 16:32:44 +00:00
if ( json ) {
uint8_t gpio = MCP230xx_readGPIO ( 0 ) ;
ResponseAppend_P ( PSTR ( " , \" MCP230XX \" :{ \" D0 \" :%i, \" D1 \" :%i, \" D2 \" :%i, \" D3 \" :%i, \" D4 \" :%i, \" D5 \" :%i, \" D6 \" :%i, \" D7 \" :%i " ) ,
2020-08-03 17:02:27 +01:00
( gpio > > 0 ) & 1 , ( gpio > > 1 ) & 1 , ( gpio > > 2 ) & 1 , ( gpio > > 3 ) & 1 , ( gpio > > 4 ) & 1 , ( gpio > > 5 ) & 1 , ( gpio > > 6 ) & 1 , ( gpio > > 7 ) & 1 ) ;
2020-07-23 13:52:29 +01:00
uint8_t gpiob = 0 ;
2019-11-11 16:32:44 +00:00
if ( 2 = = mcp230xx_type ) {
2020-07-23 13:52:29 +01:00
gpiob = MCP230xx_readGPIO ( 1 ) ;
2019-11-11 16:32:44 +00:00
ResponseAppend_P ( PSTR ( " , \" D8 \" :%i, \" D9 \" :%i, \" D10 \" :%i, \" D11 \" :%i, \" D12 \" :%i, \" D13 \" :%i, \" D14 \" :%i, \" D15 \" :%i " ) ,
2020-07-23 13:52:29 +01:00
( gpiob > > 0 ) & 1 , ( gpiob > > 1 ) & 1 , ( gpiob > > 2 ) & 1 , ( gpiob > > 3 ) & 1 , ( gpiob > > 4 ) & 1 , ( gpiob > > 5 ) & 1 , ( gpiob > > 6 ) & 1 , ( gpiob > > 7 ) & 1 ) ;
2018-07-11 20:16:37 +01:00
}
2020-07-23 13:52:29 +01:00
2020-08-04 07:31:50 +01:00
# ifdef USE_MCP230xx_OUTPUT
uint8_t outputcount = 0 ;
for ( uint32_t pinx = 0 ; pinx < mcp230xx_pincount ; pinx + + ) {
if ( Settings . mcp230xx_config [ pinx ] . pinmode > = 5 ) { outputcount + + ; }
}
if ( outputcount ) {
uint16_t gpiototal = ( ( uint16_t ) gpiob < < 8 ) | gpio ;
ResponseAppend_P ( PSTR ( " , \" MCP230_OUT \" :{ " ) ) ;
char stt [ 7 ] ;
for ( uint32_t pinx = 0 ; pinx < mcp230xx_pincount ; pinx + + ) {
if ( Settings . mcp230xx_config [ pinx ] . pinmode > = 5 ) {
sprintf ( stt , ConvertNumTxt ( ( ( gpiototal > > pinx ) & 1 ) , Settings . mcp230xx_config [ pinx ] . pinmode ) ) ;
ResponseAppend_P ( PSTR ( " \" OUT_D%i \" : \" %s \" , " ) , pinx , stt ) ;
}
}
ResponseAppend_P ( PSTR ( " \" END \" :1} " ) ) ;
}
# endif // USE_MCP230xx_OUTPUT
2019-11-11 16:32:44 +00:00
ResponseJsonEnd ( ) ;
2018-07-11 20:16:37 +01:00
}
}
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
void MCP230xx_SetOutPin ( uint8_t pin , uint8_t pinstate ) {
uint8_t portpins ;
uint8_t port = 0 ;
2018-08-14 23:49:34 +01:00
uint8_t pinmo = Settings . mcp230xx_config [ pin ] . pinmode ;
2019-11-03 11:33:36 +00:00
uint8_t interlock = Settings . flag . interlock ; // CMND_INTERLOCK - Enable/disable interlock
2018-08-14 23:49:34 +01:00
int pinadd = ( pin % 2 ) + 1 - ( 3 * ( pin % 2 ) ) ; //check if pin is odd or even and convert to 1 (if even) or -1 (if odd)
2018-08-23 00:25:38 +01:00
char cmnd [ 7 ] , stt [ 4 ] ;
2018-08-25 10:38:32 +01:00
if ( pin > 7 ) { port = 1 ; }
2018-08-09 10:46:14 +01:00
portpins = MCP230xx_readGPIO ( port ) ;
2020-08-04 07:31:50 +01:00
if ( pinstate < 2 ) {
if ( pinstate ) portpins | = ( 1 < < ( pin - ( port * 8 ) ) ) ; else portpins & = ~ ( 1 < < ( pin - ( port * 8 ) ) ) ;
2018-08-15 09:23:02 +01:00
} else {
2020-08-04 07:31:50 +01:00
portpins ^ = ( 1 < < ( pin - ( port * 8 ) ) ) ;
2018-08-15 09:23:02 +01:00
}
2020-08-04 07:31:50 +01:00
2018-09-16 20:36:56 +01:00
I2cWrite8 ( USE_MCP230xx_ADDR , MCP230xx_GPIO + port , portpins ) ;
2019-11-03 11:33:36 +00:00
if ( Settings . flag . save_state ) { // SetOption0 - Save power state and use after restart - Firmware configured to save last known state in settings
2018-08-11 23:32:56 +01:00
Settings . mcp230xx_config [ pin ] . saved_state = portpins > > ( pin - ( port * 8 ) ) & 1 ;
2018-08-20 23:01:42 +01:00
Settings . mcp230xx_config [ pin + pinadd ] . saved_state = portpins > > ( pin + pinadd - ( port * 8 ) ) & 1 ;
2018-08-09 10:46:14 +01:00
}
2018-08-14 00:13:56 +01:00
sprintf ( cmnd , ConvertNumTxt ( pinstate , pinmo ) ) ;
sprintf ( stt , ConvertNumTxt ( ( portpins > > ( pin - ( port * 8 ) ) & 1 ) , pinmo ) ) ;
2018-08-25 10:38:32 +01:00
if ( interlock & & ( pinmo = = Settings . mcp230xx_config [ pin + pinadd ] . pinmode ) ) {
2018-08-20 23:01:42 +01:00
char stt1 [ 4 ] ;
sprintf ( stt1 , ConvertNumTxt ( ( portpins > > ( pin + pinadd - ( port * 8 ) ) & 1 ) , pinmo ) ) ;
2019-03-23 16:57:31 +00:00
Response_P ( PSTR ( " { \" S29cmnd_D%i \" :{ \" COMMAND \" : \" %s \" , \" STATE \" : \" %s \" }, \" S29cmnd_D%i \" :{ \" STATE \" : \" %s \" }} " ) , pin , cmnd , stt , pin + pinadd , stt1 ) ;
2018-08-20 23:01:42 +01:00
} else {
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_CMND_RESPONSE , pin , cmnd , stt ) ;
2018-08-20 23:01:42 +01:00
}
2018-08-09 10:46:14 +01:00
}
# endif // USE_MCP230xx_OUTPUT
2018-08-04 00:05:27 +01:00
void MCP230xx_Reset ( uint8_t pinmode ) {
uint8_t pullup = 0 ;
2018-08-25 10:38:32 +01:00
if ( ( pinmode > 1 ) & & ( pinmode < 5 ) ) { pullup = 1 ; }
2019-06-30 15:44:36 +01:00
for ( uint32_t pinx = 0 ; pinx < 16 ; pinx + + ) {
2018-08-04 00:05:27 +01:00
Settings . mcp230xx_config [ pinx ] . pinmode = pinmode ;
Settings . mcp230xx_config [ pinx ] . pullup = pullup ;
2018-08-11 23:32:56 +01:00
Settings . mcp230xx_config [ pinx ] . saved_state = 0 ;
2018-08-24 20:24:43 +01:00
if ( ( pinmode > 1 ) & & ( pinmode < 5 ) ) {
2018-08-18 15:28:25 +01:00
Settings . mcp230xx_config [ pinx ] . int_report_mode = 0 ; // Enabled for ALL by default
2018-08-18 15:34:41 +01:00
} else {
Settings . mcp230xx_config [ pinx ] . int_report_mode = 3 ; // Disabled for pinmode 1, 5 and 6 (No interrupts there)
2018-08-18 15:28:25 +01:00
}
2018-08-21 21:12:40 +01:00
Settings . mcp230xx_config [ pinx ] . int_report_defer = 0 ; // Disabled
2018-12-06 22:55:44 +00:00
Settings . mcp230xx_config [ pinx ] . int_count_en = 0 ; // Disabled by default
Settings . mcp230xx_config [ pinx ] . int_retain_flag = 0 ; // Disabled by default
2018-08-21 21:12:40 +01:00
Settings . mcp230xx_config [ pinx ] . spare13 = 0 ;
Settings . mcp230xx_config [ pinx ] . spare14 = 0 ;
Settings . mcp230xx_config [ pinx ] . spare15 = 0 ;
2018-08-04 00:05:27 +01:00
}
2018-08-25 20:15:41 +01:00
Settings . mcp230xx_int_prio = 0 ; // Once per FUNC_EVERY_50_MSECOND callback
Settings . mcp230xx_int_timer = 0 ;
2018-08-04 00:05:27 +01:00
MCP230xx_ApplySettings ( ) ;
2018-08-16 23:18:45 +01:00
char pulluptxt [ 7 ] ;
char intmodetxt [ 9 ] ;
2018-08-14 00:13:56 +01:00
sprintf ( pulluptxt , ConvertNumTxt ( pullup ) ) ;
2018-08-16 23:18:45 +01:00
uint8_t intmode = 3 ;
2018-08-25 10:38:32 +01:00
if ( ( pinmode > 1 ) & & ( pinmode < 5 ) ) { intmode = 0 ; }
2018-08-16 23:18:45 +01:00
sprintf ( intmodetxt , IntModeTxt ( intmode ) ) ;
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_SENSOR_RESPONSE , 99 , pinmode , pulluptxt , intmodetxt , " " ) ;
2018-08-04 00:05:27 +01:00
}
2019-11-09 17:34:22 +00:00
bool MCP230xx_Command ( void )
{
2019-01-28 13:08:33 +00:00
bool serviced = true ;
bool validpin = false ;
2018-08-24 19:58:19 +01:00
uint8_t paramcount = 0 ;
if ( XdrvMailbox . data_len > 0 ) {
paramcount = 1 ;
2018-09-06 18:26:52 +01:00
} else {
serviced = false ;
return serviced ;
2018-08-24 19:58:19 +01:00
}
2018-09-06 18:26:52 +01:00
char sub_string [ XdrvMailbox . data_len ] ;
2019-06-30 15:44:36 +01:00
for ( uint32_t ca = 0 ; ca < XdrvMailbox . data_len ; ca + + ) {
2018-08-25 10:38:32 +01:00
if ( ( ' ' = = XdrvMailbox . data [ ca ] ) | | ( ' = ' = = XdrvMailbox . data [ ca ] ) ) { XdrvMailbox . data [ ca ] = ' , ' ; }
if ( ' , ' = = XdrvMailbox . data [ ca ] ) { paramcount + + ; }
2018-08-24 19:58:19 +01:00
}
UpperCase ( XdrvMailbox . data , XdrvMailbox . data ) ;
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET " ) ) { MCP230xx_Reset ( 1 ) ; return serviced ; }
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET1 " ) ) { MCP230xx_Reset ( 1 ) ; return serviced ; }
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET2 " ) ) { MCP230xx_Reset ( 2 ) ; return serviced ; }
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET3 " ) ) { MCP230xx_Reset ( 3 ) ; return serviced ; }
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET4 " ) ) { MCP230xx_Reset ( 4 ) ; return serviced ; }
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-24 19:58:19 +01:00
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET5 " ) ) { MCP230xx_Reset ( 5 ) ; return serviced ; }
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " RESET6 " ) ) { MCP230xx_Reset ( 6 ) ; return serviced ; }
2018-08-14 20:36:10 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-08-25 20:15:41 +01:00
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " INTPRI " ) ) {
2018-09-06 18:26:52 +01:00
if ( paramcount > 1 ) {
2018-08-25 20:15:41 +01:00
uint8_t intpri = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) ) ;
if ( ( intpri > = 0 ) & & ( intpri < = 20 ) ) {
Settings . mcp230xx_int_prio = intpri ;
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " PRI " , 99 , Settings . mcp230xx_int_prio ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
}
} else { // No parameter was given for INTPRI so we return the current configured value
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " PRI " , 99 , Settings . mcp230xx_int_prio ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
}
}
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " INTTIMER " ) ) {
2018-09-06 18:26:52 +01:00
if ( paramcount > 1 ) {
2018-08-25 20:15:41 +01:00
uint8_t inttim = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) ) ;
if ( ( inttim > = 0 ) & & ( inttim < = 3600 ) ) {
Settings . mcp230xx_int_timer = inttim ;
MCP230xx_CheckForIntCounter ( ) ; // update register on whether or not we should be counting interrupts
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " TIMER " , 99 , Settings . mcp230xx_int_timer ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
}
} else { // No parameter was given for INTTIM so we return the current configured value
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " TIMER " , 99 , Settings . mcp230xx_int_timer ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
}
}
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " INTDEF " ) ) {
2018-09-06 18:26:52 +01:00
if ( paramcount > 1 ) {
2018-08-25 20:15:41 +01:00
uint8_t pin = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) ) ;
if ( pin < mcp230xx_pincount ) {
if ( pin = = 0 ) {
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " 0 " ) ) validpin = true ;
} else {
validpin = true ;
}
}
if ( validpin ) {
2018-09-06 18:26:52 +01:00
if ( paramcount > 2 ) {
2018-08-25 20:15:41 +01:00
uint8_t intdef = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 3 ) ) ;
if ( ( intdef > = 0 ) & & ( intdef < = 15 ) ) {
Settings . mcp230xx_config [ pin ] . int_report_defer = intdef ;
if ( Settings . mcp230xx_config [ pin ] . int_count_en ) {
Settings . mcp230xx_config [ pin ] . int_count_en = 0 ;
MCP230xx_CheckForIntCounter ( ) ;
2020-11-06 16:09:13 +00:00
AddLog_P ( LOG_LEVEL_INFO , PSTR ( " *** WARNING *** - Disabled INTCNT for pin D%i " ) , pin ) ;
2018-08-25 20:15:41 +01:00
}
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " DEF " , pin , Settings . mcp230xx_config [ pin ] . int_report_defer ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
} else {
serviced = false ;
return serviced ;
}
} else {
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " DEF " , pin , Settings . mcp230xx_config [ pin ] . int_report_defer ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
}
}
serviced = false ;
return serviced ;
} else {
serviced = false ;
return serviced ;
}
}
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " INTCNT " ) ) {
2018-09-06 18:26:52 +01:00
if ( paramcount > 1 ) {
2018-08-25 20:15:41 +01:00
uint8_t pin = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) ) ;
if ( pin < mcp230xx_pincount ) {
if ( pin = = 0 ) {
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " 0 " ) ) validpin = true ;
} else {
validpin = true ;
}
}
if ( validpin ) {
2018-09-06 18:26:52 +01:00
if ( paramcount > 2 ) {
2018-08-25 20:15:41 +01:00
uint8_t intcnt = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 3 ) ) ;
if ( ( intcnt > = 0 ) & & ( intcnt < = 1 ) ) {
Settings . mcp230xx_config [ pin ] . int_count_en = intcnt ;
if ( Settings . mcp230xx_config [ pin ] . int_report_defer ) {
Settings . mcp230xx_config [ pin ] . int_report_defer = 0 ;
2020-11-06 16:09:13 +00:00
AddLog_P ( LOG_LEVEL_INFO , PSTR ( " *** WARNING *** - Disabled INTDEF for pin D%i " ) , pin ) ;
2018-08-25 20:15:41 +01:00
}
if ( Settings . mcp230xx_config [ pin ] . int_report_mode < 3 ) {
Settings . mcp230xx_config [ pin ] . int_report_mode = 3 ;
2020-11-06 16:09:13 +00:00
AddLog_P ( LOG_LEVEL_INFO , PSTR ( " *** WARNING *** - Disabled immediate interrupt/telemetry reporting for pin D%i " ) , pin ) ;
2018-08-25 20:15:41 +01:00
}
if ( ( Settings . mcp230xx_config [ pin ] . int_count_en ) & & ( ! Settings . mcp230xx_int_timer ) ) {
2020-11-06 16:09:13 +00:00
AddLog_P ( LOG_LEVEL_INFO , PSTR ( " *** WARNING *** - INTCNT enabled for pin D%i but global INTTIMER is disabled! " ) , pin ) ;
2018-08-25 20:15:41 +01:00
}
MCP230xx_CheckForIntCounter ( ) ; // update register on whether or not we should be counting interrupts
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " CNT " , pin , Settings . mcp230xx_config [ pin ] . int_count_en ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
} else {
serviced = false ;
return serviced ;
}
} else {
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " CNT " , pin , Settings . mcp230xx_config [ pin ] . int_count_en ) ; // "{\"MCP230xx_INT%s\":{\"D_%i\":%i}}";
2018-08-25 20:15:41 +01:00
return serviced ;
}
}
serviced = false ;
return serviced ;
} else {
serviced = false ;
return serviced ;
}
}
2018-12-06 22:55:44 +00:00
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " INTRETAIN " ) ) {
if ( paramcount > 1 ) {
uint8_t pin = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) ) ;
if ( pin < mcp230xx_pincount ) {
if ( pin = = 0 ) {
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " 0 " ) ) validpin = true ;
} else {
validpin = true ;
}
}
if ( validpin ) {
if ( paramcount > 2 ) {
uint8_t int_retain = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 3 ) ) ;
if ( ( int_retain > = 0 ) & & ( int_retain < = 1 ) ) {
Settings . mcp230xx_config [ pin ] . int_retain_flag = int_retain ;
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " INT_RETAIN " , pin , Settings . mcp230xx_config [ pin ] . int_retain_flag ) ;
2018-12-06 22:55:44 +00:00
MCP230xx_CheckForIntRetainer ( ) ;
return serviced ;
} else {
serviced = false ;
return serviced ;
}
} else {
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_INTCFG_RESPONSE , " INT_RETAIN " , pin , Settings . mcp230xx_config [ pin ] . int_retain_flag ) ;
2018-12-06 22:55:44 +00:00
return serviced ;
}
}
serviced = false ;
return serviced ;
} else {
serviced = false ;
return serviced ;
}
}
2018-08-24 19:58:19 +01:00
uint8_t pin = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) ) ;
2019-03-08 14:15:42 +00:00
2018-07-21 17:04:36 +01:00
if ( pin < mcp230xx_pincount ) {
2018-08-25 10:38:32 +01:00
if ( 0 = = pin ) {
2018-08-24 19:58:19 +01:00
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 1 ) , " 0 " ) ) validpin = true ;
} else {
validpin = true ;
}
}
2018-09-06 18:26:52 +01:00
if ( validpin & & ( paramcount > 1 ) ) {
2018-08-24 19:58:19 +01:00
if ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " ? " ) ) {
2018-07-31 17:44:31 +01:00
uint8_t port = 0 ;
2018-08-25 10:38:32 +01:00
if ( pin > 7 ) { port = 1 ; }
2018-07-31 17:44:31 +01:00
uint8_t portdata = MCP230xx_readGPIO ( port ) ;
2018-08-16 23:18:45 +01:00
char pulluptxtr [ 7 ] , pinstatustxtr [ 7 ] ;
char intmodetxt [ 9 ] ;
sprintf ( intmodetxt , IntModeTxt ( Settings . mcp230xx_config [ pin ] . int_report_mode ) ) ;
2018-08-14 00:13:56 +01:00
sprintf ( pulluptxtr , ConvertNumTxt ( Settings . mcp230xx_config [ pin ] . pullup ) ) ;
# ifdef USE_MCP230xx_OUTPUT
uint8_t pinmod = Settings . mcp230xx_config [ pin ] . pinmode ;
sprintf ( pinstatustxtr , ConvertNumTxt ( portdata > > ( pin - ( port * 8 ) ) & 1 , pinmod ) ) ;
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_SENSOR_RESPONSE , pin , pinmod , pulluptxtr , intmodetxt , pinstatustxtr ) ;
2018-08-14 20:36:10 +01:00
# else // not USE_MCP230xx_OUTPUT
2018-08-14 00:13:56 +01:00
sprintf ( pinstatustxtr , ConvertNumTxt ( portdata > > ( pin - ( port * 8 ) ) & 1 ) ) ;
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_SENSOR_RESPONSE , pin , Settings . mcp230xx_config [ pin ] . pinmode , pulluptxtr , intmodetxt , pinstatustxtr ) ;
2018-08-14 00:13:56 +01:00
# endif //USE_MCP230xx_OUTPUT
2018-07-21 17:04:36 +01:00
return serviced ;
}
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-14 00:13:56 +01:00
if ( Settings . mcp230xx_config [ pin ] . pinmode > = 5 ) {
uint8_t pincmd = Settings . mcp230xx_config [ pin ] . pinmode - 5 ;
2020-08-04 07:31:50 +01:00
uint8_t relay_no = 0 ;
for ( relay_no = 0 ; relay_no < mcp230xx_pincount ; relay_no + + ) {
if ( mcp230xx_outpinmapping [ relay_no ] = = pin ) break ;
}
2020-10-30 11:29:48 +00:00
relay_no = TasmotaGlobal . devices_present - mcp230xx_oldoutpincount + relay_no + 1 ;
2019-09-30 16:19:15 +01:00
if ( ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " ON " ) ) | | ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " 1 " ) ) ) {
2020-08-04 07:31:50 +01:00
ExecuteCommandPower ( relay_no , 1 , SRC_IGNORE ) ;
2018-08-09 10:46:14 +01:00
return serviced ;
}
2019-09-30 16:19:15 +01:00
if ( ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " OFF " ) ) | | ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " 0 " ) ) ) {
2020-08-04 07:31:50 +01:00
ExecuteCommandPower ( relay_no , 0 , SRC_IGNORE ) ;
2018-08-09 10:46:14 +01:00
return serviced ;
}
2019-10-01 07:49:40 +01:00
if ( ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " T " ) ) | | ( ! strcmp ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) , " 2 " ) ) ) {
2020-08-04 07:31:50 +01:00
ExecuteCommandPower ( relay_no , 2 , SRC_IGNORE ) ;
2018-08-09 10:46:14 +01:00
return serviced ;
}
}
2018-08-14 00:13:56 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-08-24 19:58:19 +01:00
uint8_t pinmode = 0 ;
uint8_t pullup = 0 ;
uint8_t intmode = 0 ;
2018-09-06 18:26:52 +01:00
if ( paramcount > 1 ) {
2018-08-24 19:58:19 +01:00
pinmode = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 2 ) ) ;
}
2018-09-06 18:26:52 +01:00
if ( paramcount > 2 ) {
2018-08-24 19:58:19 +01:00
pullup = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 3 ) ) ;
}
2018-09-06 18:26:52 +01:00
if ( paramcount > 3 ) {
2018-08-24 19:58:19 +01:00
intmode = atoi ( subStr ( sub_string , XdrvMailbox . data , " , " , 4 ) ) ;
}
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2019-09-30 20:31:23 +01:00
if ( ( pin < mcp230xx_pincount ) & & ( pinmode > 0 ) & & ( pinmode < 7 ) & & ( pullup < 2 ) & & ( paramcount > 2 ) ) {
2018-08-24 19:58:19 +01:00
# else // not use OUTPUT
2019-09-30 20:31:23 +01:00
if ( ( pin < mcp230xx_pincount ) & & ( pinmode > 0 ) & & ( pinmode < 5 ) & & ( pullup < 2 ) & & ( paramcount > 2 ) ) {
2018-08-14 00:13:56 +01:00
# endif // USE_MCP230xx_OUTPUT
2018-08-24 19:58:19 +01:00
Settings . mcp230xx_config [ pin ] . pinmode = pinmode ;
Settings . mcp230xx_config [ pin ] . pullup = pullup ;
2018-08-24 20:24:43 +01:00
if ( ( pinmode > 1 ) & & ( pinmode < 5 ) ) {
if ( ( intmode > = 0 ) & & ( intmode < = 3 ) ) {
2018-08-24 19:58:19 +01:00
Settings . mcp230xx_config [ pin ] . int_report_mode = intmode ;
2018-08-16 23:18:45 +01:00
}
2018-08-24 19:58:19 +01:00
} else {
Settings . mcp230xx_config [ pin ] . int_report_mode = 3 ; // Int mode not valid for pinmodes other than 2 through 4
}
MCP230xx_ApplySettings ( ) ;
uint8_t port = 0 ;
2018-08-25 10:38:32 +01:00
if ( pin > 7 ) { port = 1 ; }
2018-08-24 19:58:19 +01:00
uint8_t portdata = MCP230xx_readGPIO ( port ) ;
char pulluptxtc [ 7 ] , pinstatustxtc [ 7 ] ;
char intmodetxt [ 9 ] ;
sprintf ( pulluptxtc , ConvertNumTxt ( pullup ) ) ;
sprintf ( intmodetxt , IntModeTxt ( Settings . mcp230xx_config [ pin ] . int_report_mode ) ) ;
2018-08-14 00:13:56 +01:00
# ifdef USE_MCP230xx_OUTPUT
2018-08-24 19:58:19 +01:00
sprintf ( pinstatustxtc , ConvertNumTxt ( portdata > > ( pin - ( port * 8 ) ) & 1 , Settings . mcp230xx_config [ pin ] . pinmode ) ) ;
2018-08-14 00:13:56 +01:00
# else // not USE_MCP230xx_OUTPUT
2018-08-24 19:58:19 +01:00
sprintf ( pinstatustxtc , ConvertNumTxt ( portdata > > ( pin - ( port * 8 ) ) & 1 ) ) ;
2018-08-14 00:13:56 +01:00
# endif // USE_MCP230xx_OUTPUT
2019-03-23 16:57:31 +00:00
Response_P ( MCP230XX_SENSOR_RESPONSE , pin , pinmode , pulluptxtc , intmodetxt , pinstatustxtc ) ;
2018-08-24 19:58:19 +01:00
return serviced ;
2018-07-12 22:12:01 +01:00
}
} else {
2018-08-24 19:58:19 +01:00
serviced = false ; // no valid pin was used
return serviced ;
2018-07-12 22:12:01 +01:00
}
return serviced ;
}
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_DISPLAYOUTPUT
2019-03-19 16:31:43 +00:00
const char HTTP_SNS_MCP230xx_OUTPUT [ ] PROGMEM = " {s}MCP230XX D%d{m}%s{e} " ; // {s} = <tr><th>, {m} = </th><td>, {e} = </td></tr>
2018-08-09 10:46:14 +01:00
2019-11-09 17:34:22 +00:00
void MCP230xx_UpdateWebData ( void )
{
2018-08-09 10:46:14 +01:00
uint8_t gpio1 = MCP230xx_readGPIO ( 0 ) ;
uint8_t gpio2 = 0 ;
2018-08-25 10:38:32 +01:00
if ( 2 = = mcp230xx_type ) {
gpio2 = MCP230xx_readGPIO ( 1 ) ;
2018-08-09 10:46:14 +01:00
}
uint16_t gpio = ( gpio2 < < 8 ) + gpio1 ;
2019-06-30 15:44:36 +01:00
for ( uint32_t pin = 0 ; pin < mcp230xx_pincount ; pin + + ) {
2018-08-14 00:13:56 +01:00
if ( Settings . mcp230xx_config [ pin ] . pinmode > = 5 ) {
2018-08-16 23:18:45 +01:00
char stt [ 7 ] ;
2018-08-14 00:13:56 +01:00
sprintf ( stt , ConvertNumTxt ( ( gpio > > pin ) & 1 , Settings . mcp230xx_config [ pin ] . pinmode ) ) ;
2019-03-19 16:31:43 +00:00
WSContentSend_PD ( HTTP_SNS_MCP230xx_OUTPUT , pin , stt ) ;
2018-08-09 10:46:14 +01:00
}
}
}
# endif // USE_MCP230xx_DISPLAYOUTPUT
2020-07-23 13:52:29 +01:00
/*
2018-08-09 10:46:14 +01:00
# ifdef USE_MCP230xx_OUTPUT
2019-11-11 16:32:44 +00:00
void MCP230xx_OutputTelemetry ( void )
{
2018-08-09 10:46:14 +01:00
uint8_t outputcount = 0 ;
uint16_t gpiototal = 0 ;
uint8_t gpioa = 0 ;
uint8_t gpiob = 0 ;
gpioa = MCP230xx_readGPIO ( 0 ) ;
2018-08-25 10:38:32 +01:00
if ( 2 = = mcp230xx_type ) { gpiob = MCP230xx_readGPIO ( 1 ) ; }
2018-08-24 20:24:43 +01:00
gpiototal = ( ( uint16_t ) gpiob < < 8 ) | gpioa ;
2019-06-30 15:44:36 +01:00
for ( uint32_t pinx = 0 ; pinx < mcp230xx_pincount ; pinx + + ) {
2018-08-14 00:13:56 +01:00
if ( Settings . mcp230xx_config [ pinx ] . pinmode > = 5 ) outputcount + + ;
2018-08-09 10:46:14 +01:00
}
if ( outputcount ) {
2018-08-16 23:18:45 +01:00
char stt [ 7 ] ;
2019-09-04 17:06:34 +01:00
ResponseTime_P ( PSTR ( " , \" MCP230_OUT \" :{ " ) ) ;
2019-06-30 15:44:36 +01:00
for ( uint32_t pinx = 0 ; pinx < mcp230xx_pincount ; pinx + + ) {
2018-08-14 00:13:56 +01:00
if ( Settings . mcp230xx_config [ pinx ] . pinmode > = 5 ) {
sprintf ( stt , ConvertNumTxt ( ( ( gpiototal > > pinx ) & 1 ) , Settings . mcp230xx_config [ pinx ] . pinmode ) ) ;
2019-03-23 16:57:31 +00:00
ResponseAppend_P ( PSTR ( " \" OUT_D%i \" : \" %s \" , " ) , pinx , stt ) ;
2018-08-09 10:46:14 +01:00
}
}
2019-03-23 16:57:31 +00:00
ResponseAppend_P ( PSTR ( " \" END \" :1}} " ) ) ;
2019-11-10 16:40:37 +00:00
MqttPublishTeleSensor ( ) ;
2018-08-09 10:46:14 +01:00
}
}
2018-08-14 20:36:10 +01:00
# endif // USE_MCP230xx_OUTPUT
2020-07-23 13:52:29 +01:00
*/
2018-08-09 10:46:14 +01:00
2018-09-23 23:30:07 +01:00
void MCP230xx_Interrupt_Counter_Report ( void ) {
2019-09-04 17:06:34 +01:00
ResponseTime_P ( PSTR ( " , \" MCP230_INTTIMER \" :{ " ) ) ;
2019-06-30 15:44:36 +01:00
for ( uint32_t pinx = 0 ; pinx < mcp230xx_pincount ; pinx + + ) {
2018-08-25 20:15:41 +01:00
if ( Settings . mcp230xx_config [ pinx ] . int_count_en ) { // Counting is enabled for this pin so we add to report
2019-03-23 16:57:31 +00:00
ResponseAppend_P ( PSTR ( " \" INTCNT_D%i \" :%i, " ) , pinx , mcp230xx_int_counter [ pinx ] ) ;
2018-08-25 20:15:41 +01:00
mcp230xx_int_counter [ pinx ] = 0 ;
}
}
2019-03-23 16:57:31 +00:00
ResponseAppend_P ( PSTR ( " \" END \" :1}} " ) ) ;
2019-11-10 16:40:37 +00:00
MqttPublishTeleSensor ( ) ;
2018-08-25 20:15:41 +01:00
mcp230xx_int_sec_counter = 0 ;
}
2018-12-06 22:55:44 +00:00
void MCP230xx_Interrupt_Retain_Report ( void ) {
uint16_t retainresult = 0 ;
2019-09-04 17:06:34 +01:00
ResponseTime_P ( PSTR ( " , \" MCP_INTRETAIN \" :{ " ) ) ;
2019-06-30 15:44:36 +01:00
for ( uint32_t pinx = 0 ; pinx < mcp230xx_pincount ; pinx + + ) {
2019-03-08 14:15:42 +00:00
if ( Settings . mcp230xx_config [ pinx ] . int_retain_flag ) {
2019-03-23 16:57:31 +00:00
ResponseAppend_P ( PSTR ( " \" D%i \" :%i, " ) , pinx , mcp230xx_int_retainer [ pinx ] ) ;
2018-12-06 22:55:44 +00:00
retainresult | = ( ( ( mcp230xx_int_retainer [ pinx ] ) & 1 ) < < pinx ) ;
mcp230xx_int_retainer [ pinx ] = 0 ;
}
}
2019-03-23 16:57:31 +00:00
ResponseAppend_P ( PSTR ( " \" Value \" :%u}} " ) , retainresult ) ;
2019-11-10 16:40:37 +00:00
MqttPublishTeleSensor ( ) ;
2018-12-06 22:55:44 +00:00
}
2018-08-09 10:46:14 +01:00
2020-08-04 07:31:50 +01:00
# ifdef USE_MCP230xx_OUTPUT
2020-08-03 17:02:27 +01:00
void MCP230xx_SwitchRelay ( ) {
2020-10-30 11:29:48 +00:00
for ( uint32_t i = TasmotaGlobal . devices_present - mcp230xx_oldoutpincount ; i < TasmotaGlobal . devices_present ; i + + ) {
uint8_t pin = mcp230xx_outpinmapping [ i - ( TasmotaGlobal . devices_present - mcp230xx_oldoutpincount ) ] ;
2020-08-03 17:02:27 +01:00
uint8_t pincmd = Settings . mcp230xx_config [ pin ] . pinmode - 5 ;
uint8_t relay_state = bitRead ( XdrvMailbox . index , i ) ;
2020-11-06 16:09:13 +00:00
AddLog_P ( LOG_LEVEL_DEBUG , PSTR ( " MCP: relay %d pin_no %d state %d " ) , i , pin , relay_state ) ;
2020-08-03 17:02:27 +01:00
switch ( relay_state ) {
case 1 :
MCP230xx_SetOutPin ( pin , abs ( pincmd - 1 ) ) ;
break ;
case 0 :
MCP230xx_SetOutPin ( pin , pincmd ) ;
break ;
}
}
}
2020-08-04 07:31:50 +01:00
# endif // USE_MCP230xx_OUTPUT
2020-08-03 17:02:27 +01:00
2018-07-11 20:16:37 +01:00
/*********************************************************************************************\
Interface
2018-08-14 00:13:56 +01:00
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2018-07-11 20:16:37 +01:00
2019-01-28 13:08:33 +00:00
bool Xsns29 ( uint8_t function )
2018-07-11 20:16:37 +01:00
{
2019-11-04 09:38:05 +00:00
if ( ! I2cEnabled ( XI2C_22 ) ) { return false ; }
2019-11-03 16:54:39 +00:00
2019-01-28 13:08:33 +00:00
bool result = false ;
2018-07-11 20:16:37 +01:00
2020-08-03 17:02:27 +01:00
if ( FUNC_PRE_INIT = = function ) {
2019-11-09 17:34:22 +00:00
MCP230xx_Detect ( ) ;
2019-11-11 16:32:44 +00:00
}
else if ( mcp230xx_type ) {
switch ( function ) {
case FUNC_EVERY_50_MSECOND :
if ( mcp230xx_int_en ) { // Only check for interrupts if its enabled on one of the pins
mcp230xx_int_prio_counter + + ;
if ( ( mcp230xx_int_prio_counter ) > = ( Settings . mcp230xx_int_prio ) ) {
MCP230xx_CheckForInterrupt ( ) ;
mcp230xx_int_prio_counter = 0 ;
}
}
break ;
case FUNC_EVERY_SECOND :
2019-11-09 17:34:22 +00:00
if ( mcp230xx_int_counter_en ) {
mcp230xx_int_sec_counter + + ;
if ( mcp230xx_int_sec_counter > = Settings . mcp230xx_int_timer ) { // Interrupt counter interval reached, lets report
MCP230xx_Interrupt_Counter_Report ( ) ;
}
}
2020-10-29 12:37:09 +00:00
if ( TasmotaGlobal . tele_period = = 0 ) {
2019-11-09 17:34:22 +00:00
if ( mcp230xx_int_retainer_en ) { // We have pins configured for interrupt retain reporting
MCP230xx_Interrupt_Retain_Report ( ) ;
}
2020-07-23 13:52:29 +01:00
/*
2019-11-09 17:34:22 +00:00
# ifdef USE_MCP230xx_OUTPUT
MCP230xx_OutputTelemetry ( ) ;
# endif // USE_MCP230xx_OUTPUT
2020-07-23 13:52:29 +01:00
*/
2019-11-09 17:34:22 +00:00
}
2019-11-11 16:32:44 +00:00
break ;
2020-08-04 07:31:50 +01:00
# ifdef USE_MCP230xx_OUTPUT
2020-08-03 17:02:27 +01:00
case FUNC_SET_POWER :
MCP230xx_SwitchRelay ( ) ;
break ;
2020-08-04 07:31:50 +01:00
# endif // USE_MCP230xx_OUTPUT
2019-11-11 16:32:44 +00:00
case FUNC_JSON_APPEND :
MCP230xx_Show ( 1 ) ;
break ;
case FUNC_COMMAND_SENSOR :
if ( XSNS_29 = = XdrvMailbox . index ) {
result = MCP230xx_Command ( ) ;
}
break ;
2018-08-09 10:46:14 +01:00
# ifdef USE_WEBSERVER
# ifdef USE_MCP230xx_OUTPUT
# ifdef USE_MCP230xx_DISPLAYOUTPUT
2019-11-11 16:32:44 +00:00
case FUNC_WEB_SENSOR :
MCP230xx_UpdateWebData ( ) ;
break ;
2018-08-09 10:46:14 +01:00
# endif // USE_MCP230xx_DISPLAYOUTPUT
# endif // USE_MCP230xx_OUTPUT
# endif // USE_WEBSERVER
2019-11-11 16:32:44 +00:00
}
2018-07-11 20:16:37 +01:00
}
return result ;
}
# endif // USE_MCP230xx
# endif // USE_I2C