2015-04-25 12:04:10 +01:00
# include "OneWireSlave.h"
2015-08-09 08:15:32 +01:00
// uncomment this line to enable sending messages along with errors (but takes more program memory)
//#define ERROR_MESSAGES
2015-04-27 20:53:31 +01:00
# ifdef ERROR_MESSAGES
# define ERROR(msg) error_(msg)
# else
# define ERROR(msg) error_(0)
2015-04-25 20:40:38 +01:00
# endif
namespace
2015-04-25 12:04:10 +01:00
{
2015-04-25 20:40:38 +01:00
const unsigned long ResetMinDuration = 480 ;
const unsigned long ResetMaxDuration = 900 ;
2017-04-30 21:45:32 +01:00
const unsigned long PresenceWaitDuration = 15 ;
const unsigned long PresenceDuration = 200 ;
2015-04-25 12:04:10 +01:00
2015-04-26 21:01:30 +01:00
const unsigned long ReadBitSamplingTime = 25 ;
2015-04-25 20:40:38 +01:00
const unsigned long SendBitDuration = 35 ;
2015-04-26 00:08:37 +01:00
const byte ReceiveCommand = ( byte ) - 1 ;
2015-04-25 20:40:38 +01:00
void ( * timerEvent ) ( ) = 0 ;
}
2015-12-13 20:33:43 +00:00
OneWireSlave OWSlave ;
2015-04-26 19:24:11 +01:00
byte OneWireSlave : : rom_ [ 8 ] ;
2015-04-26 21:01:30 +01:00
byte OneWireSlave : : scratchpad_ [ 8 ] ;
2015-04-26 19:24:11 +01:00
Pin OneWireSlave : : pin_ ;
unsigned long OneWireSlave : : resetStart_ ;
unsigned long OneWireSlave : : lastReset_ ;
void ( * OneWireSlave : : receiveBitCallback_ ) ( bool bit , bool error ) ;
void ( * OneWireSlave : : bitSentCallback_ ) ( bool error ) ;
2015-04-27 20:53:31 +01:00
void ( * OneWireSlave : : clientReceiveCallback_ ) ( ReceiveEvent evt , byte data ) ;
2015-12-14 22:00:27 +00:00
void ( * OneWireSlave : : clientReceiveBitCallback_ ) ( bool bit ) ;
2015-04-26 19:24:11 +01:00
byte OneWireSlave : : receivingByte_ ;
byte OneWireSlave : : searchRomBytePos_ ;
byte OneWireSlave : : searchRomBitPos_ ;
bool OneWireSlave : : searchRomInverse_ ;
2016-02-26 09:06:59 +00:00
bool OneWireSlave : : resumeCommandFlag_ ;
2016-02-27 10:35:59 +00:00
bool OneWireSlave : : alarmedFlag_ ;
2015-04-25 20:40:38 +01:00
2015-08-11 20:18:35 +01:00
const byte * OneWireSlave : : sendBuffer_ ;
byte * OneWireSlave : : recvBuffer_ ;
2015-04-27 20:53:31 +01:00
short OneWireSlave : : bufferLength_ ;
byte OneWireSlave : : bufferBitPos_ ;
short OneWireSlave : : bufferPos_ ;
2015-04-26 21:01:30 +01:00
void ( * OneWireSlave : : receiveBytesCallback_ ) ( bool error ) ;
2015-04-27 20:53:31 +01:00
void ( * OneWireSlave : : sendBytesCallback_ ) ( bool error ) ;
2015-12-18 22:25:02 +00:00
volatile bool OneWireSlave : : waitingSynchronousWriteToComplete_ ;
volatile bool OneWireSlave : : synchronousWriteError_ ;
2015-12-18 20:13:46 +00:00
bool OneWireSlave : : sendingClientBytes_ ;
2015-12-14 22:00:27 +00:00
bool OneWireSlave : : singleBit_ ;
bool OneWireSlave : : singleBitRepeat_ ;
void ( * OneWireSlave : : singleBitSentCallback_ ) ( bool error ) ;
2015-12-18 22:25:02 +00:00
void ( * OneWireSlave : : logCallback_ ) ( const char * message ) ;
2015-04-26 21:01:30 +01:00
2016-03-10 17:02:04 +00:00
ISR ( USERTIMER_COMPA_vect ) // timer1 interrupt
2015-04-25 20:40:38 +01:00
{
2016-03-10 17:02:04 +00:00
UserTimer_Stop ( ) ; // disable clock
2015-04-25 20:40:38 +01:00
void ( * event ) ( ) = timerEvent ;
timerEvent = 0 ;
event ( ) ;
}
2015-08-11 20:18:35 +01:00
void OneWireSlave : : begin ( const byte * rom , byte pinNumber )
2015-04-25 20:40:38 +01:00
{
2015-04-26 19:24:11 +01:00
pin_ = Pin ( pinNumber ) ;
resetStart_ = ( unsigned long ) - 1 ;
lastReset_ = 0 ;
2015-04-25 20:40:38 +01:00
memcpy ( rom_ , rom , 7 ) ;
2015-08-09 08:15:32 +01:00
rom_ [ 7 ] = crc8 ( rom_ , 7 ) ;
2015-04-25 12:04:10 +01:00
2016-02-26 09:06:59 +00:00
resumeCommandFlag_ = false ;
2016-02-27 10:35:59 +00:00
alarmedFlag_ = false ;
2016-02-26 09:06:59 +00:00
2015-12-14 22:00:27 +00:00
clientReceiveBitCallback_ = 0 ;
2015-12-18 20:13:46 +00:00
sendingClientBytes_ = false ;
2015-12-14 22:00:27 +00:00
2015-08-09 08:15:32 +01:00
// log("Enabling 1-wire library")
2015-04-25 20:40:38 +01:00
cli ( ) ; // disable interrupts
pin_ . inputMode ( ) ;
2015-04-26 19:24:11 +01:00
pin_ . writeLow ( ) ; // make sure the internal pull-up resistor is disabled
2015-04-25 20:40:38 +01:00
// prepare hardware timer
2016-03-10 17:02:04 +00:00
UserTimer_Init ( ) ;
2015-04-26 19:24:11 +01:00
// start 1-wire activity
2015-04-25 20:40:38 +01:00
beginWaitReset_ ( ) ;
sei ( ) ; // enable interrupts
2015-04-25 12:04:10 +01:00
}
2015-04-26 19:24:11 +01:00
void OneWireSlave : : end ( )
2015-04-25 12:04:10 +01:00
{
2015-08-11 19:40:57 +01:00
// log("Disabling 1-wire library");
2015-04-25 12:04:10 +01:00
2015-04-25 20:40:38 +01:00
cli ( ) ;
disableTimer_ ( ) ;
pin_ . detachInterrupt ( ) ;
releaseBus_ ( ) ;
sei ( ) ;
2015-04-25 12:04:10 +01:00
}
2015-12-18 20:13:46 +00:00
bool OneWireSlave : : write ( const byte * bytes , short numBytes )
{
// TODO: put the arduino to sleep between interrupts to save power?
waitingSynchronousWriteToComplete_ = true ;
beginWrite ( bytes , numBytes , & OneWireSlave : : onSynchronousWriteComplete_ ) ;
while ( waitingSynchronousWriteToComplete_ )
delay ( 1 ) ;
2015-12-18 22:25:02 +00:00
return ! synchronousWriteError_ ;
2015-12-18 20:13:46 +00:00
}
void OneWireSlave : : onSynchronousWriteComplete_ ( bool error )
{
synchronousWriteError_ = error ;
waitingSynchronousWriteToComplete_ = false ;
}
void OneWireSlave : : beginWrite ( const byte * bytes , short numBytes , void ( * complete ) ( bool error ) )
2015-04-25 12:04:10 +01:00
{
2015-04-27 20:53:31 +01:00
cli ( ) ;
2015-12-20 20:40:25 +00:00
endWrite_ ( true ) ;
2015-12-18 20:13:46 +00:00
sendingClientBytes_ = true ;
2015-04-27 20:53:31 +01:00
beginWriteBytes_ ( bytes , numBytes , complete = = 0 ? noOpCallback_ : complete ) ;
sei ( ) ;
2015-04-25 12:04:10 +01:00
}
2015-04-25 20:40:38 +01:00
2015-12-20 20:40:25 +00:00
void OneWireSlave : : endWrite_ ( bool error , bool resetInterrupts )
2015-12-18 20:13:46 +00:00
{
2015-12-20 20:40:25 +00:00
if ( resetInterrupts )
beginWaitReset_ ( ) ;
2015-12-18 20:13:46 +00:00
if ( sendingClientBytes_ )
{
sendingClientBytes_ = false ;
if ( sendBytesCallback_ ! = 0 )
{
void ( * callback ) ( bool error ) = sendBytesCallback_ ;
sendBytesCallback_ = noOpCallback_ ;
callback ( error ) ;
}
}
else if ( singleBitSentCallback_ ! = 0 )
{
void ( * callback ) ( bool ) = singleBitSentCallback_ ;
singleBitSentCallback_ = 0 ;
callback ( error ) ;
}
}
2015-12-20 20:40:25 +00:00
bool OneWireSlave : : writeBit ( bool value )
{
// TODO: put the arduino to sleep between interrupts to save power?
waitingSynchronousWriteToComplete_ = true ;
beginWriteBit ( value , false , & OneWireSlave : : onSynchronousWriteComplete_ ) ;
while ( waitingSynchronousWriteToComplete_ )
delay ( 1 ) ;
return ! synchronousWriteError_ ;
}
void OneWireSlave : : beginWriteBit ( bool value , bool repeat , void ( * bitSent ) ( bool ) )
2015-12-14 21:32:26 +00:00
{
cli ( ) ;
2015-12-20 20:40:25 +00:00
endWrite_ ( true ) ;
2015-12-18 20:13:46 +00:00
2015-12-14 21:32:26 +00:00
singleBit_ = value ;
singleBitRepeat_ = repeat ;
singleBitSentCallback_ = bitSent ;
beginSendBit_ ( value , & OneWireSlave : : onSingleBitSent_ ) ;
sei ( ) ;
}
void OneWireSlave : : onSingleBitSent_ ( bool error )
{
if ( ! error & & singleBitRepeat_ )
{
beginSendBit_ ( singleBit_ , & OneWireSlave : : onSingleBitSent_ ) ;
}
else
{
2015-12-14 22:00:27 +00:00
beginReceiveBytes_ ( scratchpad_ , 1 , & OneWireSlave : : notifyClientByteReceived_ ) ;
2015-12-14 21:32:26 +00:00
}
if ( singleBitSentCallback_ ! = 0 )
{
2015-12-18 20:13:46 +00:00
void ( * callback ) ( bool ) = singleBitSentCallback_ ;
singleBitSentCallback_ = 0 ;
callback ( error ) ;
2015-12-14 21:32:26 +00:00
}
}
void OneWireSlave : : stopWrite ( )
{
2015-12-18 20:13:46 +00:00
beginWrite ( 0 , 0 , 0 ) ;
2015-12-14 21:32:26 +00:00
}
2016-02-27 10:35:59 +00:00
void OneWireSlave : : alarmed ( bool value )
{
alarmedFlag_ = value ;
}
2015-08-11 20:18:35 +01:00
byte OneWireSlave : : crc8 ( const byte * data , short numBytes )
2015-04-25 20:40:38 +01:00
{
byte crc = 0 ;
while ( numBytes - - ) {
byte inbyte = * data + + ;
for ( byte i = 8 ; i ; i - - ) {
byte mix = ( crc ^ inbyte ) & 0x01 ;
crc > > = 1 ;
if ( mix ) crc ^ = 0x8C ;
inbyte > > = 1 ;
}
}
return crc ;
}
void OneWireSlave : : setTimerEvent_ ( short delayMicroSeconds , void ( * handler ) ( ) )
{
delayMicroSeconds - = 10 ; // remove overhead (tuned on Arduino Uno)
short skipTicks = ( delayMicroSeconds - 3 ) / 4 ; // round the micro seconds delay to a number of ticks to skip (4us per tick, so 4us must skip 0 tick, 8us must skip 1 tick, etc.)
if ( skipTicks < 1 ) skipTicks = 1 ;
timerEvent = handler ;
2016-03-10 17:02:04 +00:00
UserTimer_Run ( skipTicks ) ;
2015-04-25 20:40:38 +01:00
}
void OneWireSlave : : disableTimer_ ( )
{
2016-03-10 17:02:04 +00:00
UserTimer_Stop ( ) ;
2015-04-25 20:40:38 +01:00
}
void OneWireSlave : : onEnterInterrupt_ ( )
{
}
void OneWireSlave : : onLeaveInterrupt_ ( )
{
}
void OneWireSlave : : error_ ( const char * message )
{
2015-12-18 22:25:02 +00:00
if ( logCallback_ ! = 0 )
logCallback_ ( message ) ;
2015-12-20 20:40:25 +00:00
endWrite_ ( true ) ;
2015-12-18 20:13:46 +00:00
if ( clientReceiveCallback_ ! = 0 )
clientReceiveCallback_ ( RE_Error , 0 ) ;
2015-04-25 20:40:38 +01:00
}
void OneWireSlave : : pullLow_ ( )
{
pin_ . outputMode ( ) ;
pin_ . writeLow ( ) ;
}
void OneWireSlave : : releaseBus_ ( )
{
pin_ . inputMode ( ) ;
}
2015-04-26 20:09:30 +01:00
void OneWireSlave : : beginResetDetection_ ( )
{
setTimerEvent_ ( ResetMinDuration - 50 , & OneWireSlave : : resetCheck_ ) ;
resetStart_ = micros ( ) - 50 ;
}
2016-02-23 16:23:54 +00:00
void OneWireSlave : : beginResetDetectionSendZero_ ( )
{
setTimerEvent_ ( ResetMinDuration - SendBitDuration - 50 , & OneWireSlave : : resetCheck_ ) ;
resetStart_ = micros ( ) - SendBitDuration - 50 ;
}
2015-04-26 20:09:30 +01:00
void OneWireSlave : : cancelResetDetection_ ( )
2015-04-25 20:40:38 +01:00
{
disableTimer_ ( ) ;
2015-04-26 20:09:30 +01:00
resetStart_ = ( unsigned long ) - 1 ;
2015-04-25 20:40:38 +01:00
}
2015-04-26 20:09:30 +01:00
void OneWireSlave : : resetCheck_ ( )
2015-04-25 20:40:38 +01:00
{
onEnterInterrupt_ ( ) ;
2015-04-26 20:09:30 +01:00
if ( ! pin_ . read ( ) )
2015-04-25 20:40:38 +01:00
{
2015-04-26 20:09:30 +01:00
pin_ . attachInterrupt ( & OneWireSlave : : waitReset_ , CHANGE ) ;
2015-08-09 08:15:32 +01:00
// log("Reset detected during another operation");
2015-04-25 20:40:38 +01:00
}
onLeaveInterrupt_ ( ) ;
}
2015-04-26 18:46:52 +01:00
void OneWireSlave : : beginReceiveBit_ ( void ( * completeCallback ) ( bool bit , bool error ) )
{
receiveBitCallback_ = completeCallback ;
2015-04-26 19:24:11 +01:00
pin_ . attachInterrupt ( & OneWireSlave : : receive_ , FALLING ) ;
2015-04-25 20:40:38 +01:00
}
2015-04-26 00:08:37 +01:00
void OneWireSlave : : receive_ ( )
{
onEnterInterrupt_ ( ) ;
2015-04-26 19:24:11 +01:00
pin_ . detachInterrupt ( ) ;
setTimerEvent_ ( ReadBitSamplingTime , & OneWireSlave : : readBit_ ) ;
2015-04-26 18:46:52 +01:00
onLeaveInterrupt_ ( ) ;
}
2015-04-26 20:09:30 +01:00
void OneWireSlave : : readBit_ ( )
{
onEnterInterrupt_ ( ) ;
bool bit = pin_ . read ( ) ;
if ( bit )
cancelResetDetection_ ( ) ;
else
beginResetDetection_ ( ) ;
receiveBitCallback_ ( bit , false ) ;
//dbgOutput.writeLow();
//dbgOutput.writeHigh();
onLeaveInterrupt_ ( ) ;
}
2015-04-26 18:46:52 +01:00
void OneWireSlave : : beginSendBit_ ( bool bit , void ( * completeCallback ) ( bool error ) )
{
bitSentCallback_ = completeCallback ;
if ( bit )
{
2015-04-26 19:24:11 +01:00
pin_ . attachInterrupt ( & OneWireSlave : : sendBitOne_ , FALLING ) ;
2015-04-26 18:46:52 +01:00
}
else
{
2015-04-26 19:24:11 +01:00
pin_ . attachInterrupt ( & OneWireSlave : : sendBitZero_ , FALLING ) ;
2015-04-26 18:46:52 +01:00
}
}
void OneWireSlave : : sendBitOne_ ( )
{
onEnterInterrupt_ ( ) ;
2015-04-26 00:08:37 +01:00
2015-04-26 20:09:30 +01:00
beginResetDetection_ ( ) ;
2015-04-26 19:24:11 +01:00
bitSentCallback_ ( false ) ;
2015-04-26 18:46:52 +01:00
onLeaveInterrupt_ ( ) ;
}
void OneWireSlave : : sendBitZero_ ( )
{
2015-04-26 19:24:11 +01:00
pullLow_ ( ) ; // this must be executed first because the timing is very tight with some master devices
2015-04-26 18:46:52 +01:00
2015-04-26 19:24:11 +01:00
onEnterInterrupt_ ( ) ;
2015-04-26 18:46:52 +01:00
pin_ . detachInterrupt ( ) ;
2015-04-26 19:24:11 +01:00
setTimerEvent_ ( SendBitDuration , & OneWireSlave : : endSendBitZero_ ) ;
2015-04-26 18:46:52 +01:00
2015-04-26 00:08:37 +01:00
onLeaveInterrupt_ ( ) ;
}
2015-04-26 18:46:52 +01:00
void OneWireSlave : : endSendBitZero_ ( )
{
onEnterInterrupt_ ( ) ;
releaseBus_ ( ) ;
2016-02-23 16:23:54 +00:00
beginResetDetectionSendZero_ ( ) ;
2015-04-26 18:46:52 +01:00
bitSentCallback_ ( false ) ;
onLeaveInterrupt_ ( ) ;
}
2015-04-26 20:09:30 +01:00
void OneWireSlave : : beginWaitReset_ ( )
{
disableTimer_ ( ) ;
2017-04-25 20:11:19 +01:00
pin_ . inputMode ( ) ;
2015-04-26 20:09:30 +01:00
pin_ . attachInterrupt ( & OneWireSlave : : waitReset_ , CHANGE ) ;
resetStart_ = ( unsigned int ) - 1 ;
}
void OneWireSlave : : waitReset_ ( )
2015-04-26 00:08:37 +01:00
{
onEnterInterrupt_ ( ) ;
2015-04-26 20:09:30 +01:00
bool state = pin_ . read ( ) ;
unsigned long now = micros ( ) ;
if ( state )
{
if ( resetStart_ = = ( unsigned int ) - 1 )
{
onLeaveInterrupt_ ( ) ;
return ;
}
2015-04-26 00:08:37 +01:00
2015-04-26 20:09:30 +01:00
unsigned long resetDuration = now - resetStart_ ;
resetStart_ = ( unsigned int ) - 1 ;
if ( resetDuration > = ResetMinDuration )
{
if ( resetDuration > ResetMaxDuration )
{
2015-04-27 20:53:31 +01:00
ERROR ( " Reset too long " ) ;
2015-04-26 20:09:30 +01:00
onLeaveInterrupt_ ( ) ;
return ;
}
2015-04-26 00:08:37 +01:00
2015-04-26 20:09:30 +01:00
lastReset_ = now ;
pin_ . detachInterrupt ( ) ;
2017-04-30 21:45:32 +01:00
unsigned long alreadyElapsedTime = micros ( ) - now ;
setTimerEvent_ ( alreadyElapsedTime < PresenceWaitDuration ? PresenceWaitDuration - alreadyElapsedTime : 0 , & OneWireSlave : : beginPresence_ ) ;
2015-12-20 20:40:25 +00:00
endWrite_ ( true , false ) ;
2015-04-27 20:53:31 +01:00
if ( clientReceiveCallback_ ! = 0 )
clientReceiveCallback_ ( RE_Reset , 0 ) ;
2015-04-26 20:09:30 +01:00
}
}
else
{
resetStart_ = now ;
}
2015-04-26 18:46:52 +01:00
onLeaveInterrupt_ ( ) ;
}
2015-04-26 20:09:30 +01:00
void OneWireSlave : : beginPresence_ ( )
{
pullLow_ ( ) ;
setTimerEvent_ ( PresenceDuration , & OneWireSlave : : endPresence_ ) ;
}
void OneWireSlave : : endPresence_ ( )
{
releaseBus_ ( ) ;
beginWaitCommand_ ( ) ;
}
void OneWireSlave : : beginWaitCommand_ ( )
{
2015-04-27 20:53:31 +01:00
bufferPos_ = ReceiveCommand ;
2015-04-26 20:09:30 +01:00
beginReceive_ ( ) ;
}
void OneWireSlave : : beginReceive_ ( )
{
receivingByte_ = 0 ;
2015-04-27 20:53:31 +01:00
bufferBitPos_ = 0 ;
2015-04-26 20:09:30 +01:00
beginReceiveBit_ ( & OneWireSlave : : onBitReceived_ ) ;
}
2015-04-26 18:46:52 +01:00
void OneWireSlave : : onBitReceived_ ( bool bit , bool error )
{
if ( error )
{
2015-04-27 20:53:31 +01:00
ERROR ( " Invalid bit " ) ;
if ( bufferPos_ > = 0 )
2015-04-26 21:01:30 +01:00
receiveBytesCallback_ ( true ) ;
2015-04-26 18:46:52 +01:00
return ;
}
2015-04-27 20:53:31 +01:00
receivingByte_ | = ( ( bit ? 1 : 0 ) < < bufferBitPos_ ) ;
+ + bufferBitPos_ ;
2015-04-26 00:08:37 +01:00
2015-12-14 22:00:27 +00:00
if ( clientReceiveBitCallback_ ! = 0 & & bufferPos_ ! = ReceiveCommand )
clientReceiveBitCallback_ ( bit ) ;
2015-04-27 20:53:31 +01:00
if ( bufferBitPos_ = = 8 )
2015-04-26 00:08:37 +01:00
{
2015-08-09 08:15:32 +01:00
// log("received byte", (long)receivingByte_);
2015-04-27 20:53:31 +01:00
if ( bufferPos_ = = ReceiveCommand )
2015-04-26 00:08:37 +01:00
{
2015-04-27 20:53:31 +01:00
bufferPos_ = 0 ;
2015-04-26 21:01:30 +01:00
switch ( receivingByte_ )
2015-04-26 00:08:37 +01:00
{
2015-04-26 21:01:30 +01:00
case 0xF0 : // SEARCH ROM
2016-02-27 10:35:59 +00:00
resumeCommandFlag_ = false ;
2015-04-26 00:08:37 +01:00
beginSearchRom_ ( ) ;
2015-04-26 18:46:52 +01:00
return ;
2016-02-27 10:35:59 +00:00
case 0xEC : // CONDITIONAL SEARCH ROM
resumeCommandFlag_ = false ;
if ( alarmedFlag_ )
{
beginSearchRom_ ( ) ;
}
else
{
beginWaitReset_ ( ) ;
}
return ;
2015-04-26 21:01:30 +01:00
case 0x33 : // READ ROM
2016-02-27 10:35:59 +00:00
resumeCommandFlag_ = false ;
2015-04-26 21:01:30 +01:00
beginWriteBytes_ ( rom_ , 8 , & OneWireSlave : : noOpCallback_ ) ;
return ;
case 0x55 : // MATCH ROM
2016-02-27 10:35:59 +00:00
resumeCommandFlag_ = false ;
2015-04-26 21:01:30 +01:00
beginReceiveBytes_ ( scratchpad_ , 8 , & OneWireSlave : : matchRomBytesReceived_ ) ;
return ;
case 0xCC : // SKIP ROM
2016-02-26 09:06:59 +00:00
resumeCommandFlag_ = false ;
beginReceiveBytes_ ( scratchpad_ , 1 , & OneWireSlave : : notifyClientByteReceived_ ) ;
return ;
case 0xA5 : // RESUME
if ( resumeCommandFlag_ )
{
beginReceiveBytes_ ( scratchpad_ , 1 , & OneWireSlave : : notifyClientByteReceived_ ) ;
}
else
{
beginWaitReset_ ( ) ;
}
2015-04-26 21:01:30 +01:00
return ;
default :
2015-04-27 20:53:31 +01:00
ERROR ( " Unknown command " ) ;
2015-04-26 18:46:52 +01:00
return ;
2015-04-26 00:08:37 +01:00
}
}
else
{
2015-08-11 20:18:35 +01:00
recvBuffer_ [ bufferPos_ + + ] = receivingByte_ ;
2015-04-26 21:01:30 +01:00
receivingByte_ = 0 ;
2015-04-27 20:53:31 +01:00
bufferBitPos_ = 0 ;
if ( bufferPos_ = = bufferLength_ )
2015-04-26 21:01:30 +01:00
{
beginWaitReset_ ( ) ;
receiveBytesCallback_ ( false ) ;
return ;
}
2015-04-26 00:08:37 +01:00
}
}
2015-04-26 18:46:52 +01:00
2015-04-26 19:24:11 +01:00
beginReceiveBit_ ( & OneWireSlave : : onBitReceived_ ) ;
2015-04-26 00:08:37 +01:00
}
void OneWireSlave : : beginSearchRom_ ( )
{
2015-04-26 18:46:52 +01:00
searchRomBytePos_ = 0 ;
searchRomBitPos_ = 0 ;
searchRomInverse_ = false ;
beginSearchRomSendBit_ ( ) ;
}
void OneWireSlave : : beginSearchRomSendBit_ ( )
{
byte currentByte = rom_ [ searchRomBytePos_ ] ;
bool currentBit = bitRead ( currentByte , searchRomBitPos_ ) ;
bool bitToSend = searchRomInverse_ ? ! currentBit : currentBit ;
2015-04-26 19:24:11 +01:00
beginSendBit_ ( bitToSend , & OneWireSlave : : continueSearchRom_ ) ;
2015-04-26 18:46:52 +01:00
}
void OneWireSlave : : continueSearchRom_ ( bool error )
{
if ( error )
{
2015-04-27 20:53:31 +01:00
ERROR ( " Failed to send bit " ) ;
2015-04-26 18:46:52 +01:00
return ;
}
searchRomInverse_ = ! searchRomInverse_ ;
if ( searchRomInverse_ )
{
beginSearchRomSendBit_ ( ) ;
}
else
{
2015-04-26 19:24:11 +01:00
beginReceiveBit_ ( & OneWireSlave : : searchRomOnBitReceived_ ) ;
2015-04-26 18:46:52 +01:00
}
}
void OneWireSlave : : searchRomOnBitReceived_ ( bool bit , bool error )
{
if ( error )
{
2015-04-27 20:53:31 +01:00
ERROR ( " Bit read error during ROM search " ) ;
2015-04-26 18:46:52 +01:00
return ;
}
byte currentByte = rom_ [ searchRomBytePos_ ] ;
bool currentBit = bitRead ( currentByte , searchRomBitPos_ ) ;
if ( bit = = currentBit )
{
+ + searchRomBitPos_ ;
if ( searchRomBitPos_ = = 8 )
{
searchRomBitPos_ = 0 ;
+ + searchRomBytePos_ ;
}
if ( searchRomBytePos_ = = 8 )
{
2015-08-09 08:15:32 +01:00
// log("ROM sent entirely");
2015-04-26 18:46:52 +01:00
beginWaitReset_ ( ) ;
}
else
{
beginSearchRomSendBit_ ( ) ;
}
}
else
{
2015-08-09 08:15:32 +01:00
// log("Leaving ROM search");
2015-04-26 18:46:52 +01:00
beginWaitReset_ ( ) ;
}
}
2015-04-26 21:01:30 +01:00
2015-08-11 20:18:35 +01:00
void OneWireSlave : : beginWriteBytes_ ( const byte * data , short numBytes , void ( * complete ) ( bool error ) )
2015-04-26 21:01:30 +01:00
{
2015-08-11 20:18:35 +01:00
sendBuffer_ = data ;
2015-04-27 20:53:31 +01:00
bufferLength_ = numBytes ;
bufferPos_ = 0 ;
bufferBitPos_ = 0 ;
sendBytesCallback_ = complete ;
2015-12-14 21:32:26 +00:00
if ( sendBuffer_ ! = 0 & & bufferLength_ > 0 )
{
bool bit = bitRead ( sendBuffer_ [ 0 ] , 0 ) ;
beginSendBit_ ( bit , & OneWireSlave : : bitSent_ ) ;
}
else
{
2015-12-20 20:40:25 +00:00
endWrite_ ( true ) ;
2015-12-14 22:00:27 +00:00
beginReceiveBytes_ ( scratchpad_ , 1 , & OneWireSlave : : notifyClientByteReceived_ ) ;
2015-12-14 21:32:26 +00:00
}
2015-04-27 20:53:31 +01:00
}
void OneWireSlave : : bitSent_ ( bool error )
{
if ( error )
{
ERROR ( " error sending a bit " ) ;
sendBytesCallback_ ( true ) ;
return ;
}
+ + bufferBitPos_ ;
if ( bufferBitPos_ = = 8 )
{
bufferBitPos_ = 0 ;
+ + bufferPos_ ;
}
if ( bufferPos_ = = bufferLength_ )
{
2015-12-20 20:40:25 +00:00
endWrite_ ( false ) ;
2015-04-27 20:53:31 +01:00
sendBytesCallback_ ( false ) ;
return ;
}
2015-08-11 20:18:35 +01:00
bool bit = bitRead ( sendBuffer_ [ bufferPos_ ] , bufferBitPos_ ) ;
2015-04-27 20:53:31 +01:00
beginSendBit_ ( bit , & OneWireSlave : : bitSent_ ) ;
2015-04-26 21:01:30 +01:00
}
void OneWireSlave : : beginReceiveBytes_ ( byte * buffer , short numBytes , void ( * complete ) ( bool error ) )
{
2015-08-11 20:18:35 +01:00
recvBuffer_ = buffer ;
2015-04-27 20:53:31 +01:00
bufferLength_ = numBytes ;
bufferPos_ = 0 ;
2015-04-26 21:01:30 +01:00
receiveBytesCallback_ = complete ;
beginReceive_ ( ) ;
}
void OneWireSlave : : noOpCallback_ ( bool error )
{
if ( error )
2015-04-27 20:53:31 +01:00
ERROR ( " error during an internal 1-wire operation " ) ;
2015-04-26 21:01:30 +01:00
}
void OneWireSlave : : matchRomBytesReceived_ ( bool error )
{
if ( error )
{
2016-02-26 09:06:59 +00:00
resumeCommandFlag_ = false ;
2015-04-27 20:53:31 +01:00
ERROR ( " error receiving match rom bytes " ) ;
2015-04-26 21:01:30 +01:00
return ;
}
if ( memcmp ( rom_ , scratchpad_ , 8 ) = = 0 )
{
2015-08-09 08:15:32 +01:00
// log("ROM matched");
2016-02-26 09:06:59 +00:00
resumeCommandFlag_ = true ;
2015-04-26 21:01:30 +01:00
beginReceiveBytes_ ( scratchpad_ , 1 , & OneWireSlave : : notifyClientByteReceived_ ) ;
}
else
{
2015-08-09 08:15:32 +01:00
// log("ROM not matched");
2016-02-26 09:06:59 +00:00
resumeCommandFlag_ = false ;
2015-04-26 21:01:30 +01:00
beginWaitReset_ ( ) ;
}
}
void OneWireSlave : : notifyClientByteReceived_ ( bool error )
{
if ( error )
{
2015-04-27 20:53:31 +01:00
if ( clientReceiveCallback_ ! = 0 )
clientReceiveCallback_ ( RE_Error , 0 ) ;
ERROR ( " error receiving custom bytes " ) ;
2015-04-26 21:01:30 +01:00
return ;
}
2015-04-27 20:53:31 +01:00
beginReceiveBytes_ ( scratchpad_ , 1 , & OneWireSlave : : notifyClientByteReceived_ ) ;
if ( clientReceiveCallback_ ! = 0 )
clientReceiveCallback_ ( RE_Byte , scratchpad_ [ 0 ] ) ;
2015-04-26 21:01:30 +01:00
}