|
|
View previous topic :: View next topic |
Author |
Message |
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Tue Dec 01, 2020 2:46 pm |
|
|
Ttelmah wrote: | Slightly 'worried' by your speed remarks.
Are you trying to run I2C in fast mode+ (over 400Kbps)?.
If so, you do realise that Fast Mode*+ (over 400K), and high speed mode
(over 1Mbps) _both require_ active (current source) pull-ups?. |
I am having the hardware designer take a look at this. Since we designed it for this speed and it’s been in use for years before it was slowed down last year for problems that don’t seem to be related to the speed, I hope they hardware folks got it implemented correct. Out system uses five custom designed PIC24 boards and it’s the primary chipset they use here. _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Tue Dec 01, 2020 6:45 pm |
|
|
Should I be able to read the P (stop bit) from inside an ISR?
Doing some testing (polling on P) never sees the P set to 1. Something like this "should" work (and works in my bootloader code) unless something about the ISR is an issue:
Code: |
#word I2C2STAT = getenv("SFR:I2C2STAT")
#bit P = I2C2STAT.4 // bit 4 - stop
#word IFS3 = getenv("SFR:IFS3")
#bit I2C_SYSTEM_BUS_IRQ_PENDING_BIT = IFS3.1 // bit 1 - slave I2C2 IRQ status
...
// Wait for either a stop bit, or another incoming data byte.
while ((P == 0) && (I2C_SYSTEM_BUS_IRQ_PENDING_BIT == 0));
if (P != 0)
{
messageReceivedFlag = TRUE;
}
|
I do see the I2C interrupt bit set (after I manually clear it before polling it) but I never see the P.
This PIC24 has issues where the first S start bit and interrupt are not seen, always losing the first message on the wire (workaround described in the errata document) so I do wonder if I am running in to another issue. Someone told me this was a 10+ year old part (?). _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19587
|
|
Posted: Wed Dec 02, 2020 2:42 am |
|
|
As I have already said multiple time, no.
The point is you only arrive in the interrupt after a byte is received. Your
last byte received in the packet, is _before_ the stop bit is received, so the
P bit is not set. The first byte of the next packet is received after the
start bit is sent, so the P bit is turned off, and instead you have the flag
saying a start bit has been received. Since in I2C, you can never send
STOP, DATA, DATA,,,,, You should never arrive in the 'data'
interrupt (which the I2C interrupt is on this chip), with the stop flag set. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19587
|
|
Posted: Wed Dec 02, 2020 8:32 am |
|
|
Actually almost 13 years old. However still deemed 'current'.
The more modern replacement, is the PIC33EP64GS804.
That does have the PCIE bit. |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Wed Dec 02, 2020 10:01 am |
|
|
Ttelmah wrote: | As I have already said multiple time, no.
The point is you only arrive in the interrupt after a byte is received. Your
last byte received in the packet, is _before_ the stop bit is received, so the
P bit is not set. The first byte of the next packet is received after the
start bit is sent, so the P bit is turned off, and instead you have the flag
saying a start bit has been received. Since in I2C, you can never send
STOP, DATA, DATA,,,,, You should never arrive in the 'data'
interrupt (which the I2C interrupt is on this chip), with the stop flag set. |
Understood.
But I am in the ISR from a data byte, then polling looking for the P bit. I never see it on this PIC. Basically, something like this:
Code: |
...inside the I2C ISR...
else if (state > TRANSMIT_I2C)
{
// Send the data in the TX buffer to the master.
int byteToSend;
if (slaveIndexTX < txMsgDataSlave.numBytes)
{
byteToSend = txSlaveBuffer[slaveIndexTX++];
}
else
{
// Pad extra bytes if the master reads more than we are writing.
// This can happen if we are NAKing something and the master
// expects a longer message back.
byteToSend = 0xff;
}
i2c_write (SYSTEM_BUS, byteToSend);
// HACK ==============================================================
// Wait for either a stop bit, or another incoming data byte.
while ((P == 0) && (I2C_SYSTEM_BUS_IRQ_PENDING_BIT == 0));
if (P != 0)
{
runTimeVariables.msgStatus = TX_MESSAGE_SENT;
}
}
|
Above, I can block at the while() and when the next byte comes in, it releases, then the ISR gets called again to process the pending byte. But I never see the P at the end of the message.
This morning I am doing the same test on a different PIC variant. I have my bootloader I2C polled code working on this chip, so the P works that way. _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19587
|
|
Posted: Wed Dec 02, 2020 10:49 am |
|
|
OK. So does the master NACK the last byte it wants, and then send
stop?.
Point is that there won't be another data transmission request after
the NACKED byte.
You can just test the ACK status to detect this for the end of packet. |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Wed Dec 02, 2020 11:00 am |
|
|
Yes, an I2C capture shows the last bye NACK'd then the stop bit shortly after. But the P bit never flips from 0 inside the ISR, apparently.
I am now testing on a 24EP256GP202 but having unrelated issues with the debugger not showing console output reliable. Fun day. _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Wed Dec 02, 2020 1:04 pm |
|
|
I tried this on a different PIC24 variant and it isn't working there, either, so I guess I can't stick around in an ISR looking for the P bit to be set (at least not the way I am trying to do it).
I can disable the interrupt and do a brute force bit of code in my main loop and that works.
I didn't see anything in the datasheet to say there was anything special about looking for the P bit in regards to interrupts, but perhaps there is some masking that goes on that impacts that bit being set. _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19587
|
|
Posted: Wed Dec 02, 2020 1:21 pm |
|
|
One suggestion.
Set your ISR to NOCLEAR, and clear the interrupt bit yourself immediately
after you have read I2C_ISR_STATE.
It may be that the presence of the non cleared interrupt bit masks off the
stop bit detection.
Would actually make sense, since the assumption is that this will only
be used outside the ISR.
Hionestly if polling for the bit like this, add a counter and exit when this
reaches some value. Otherwise you are asking for the bus to get hung
is a bit is missed.... |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Wed Dec 02, 2020 2:43 pm |
|
|
Ttelmah wrote: | One suggestion.
Set your ISR to NOCLEAR, and clear the interrupt bit yourself immediately
after you have read I2C_ISR_STATE.
It may be that the presence of the non cleared interrupt bit masks off the
stop bit detection.
Would actually make sense, since the assumption is that this will only
be used outside the ISR.
|
Perfect timing. I was reading the Help file last night and saw NOCLEAR and had that on my list. I have to manually clear the bit before I can detect the next one inside the ISR anyway, so I will give that a shot.
Ttelmah wrote: | One suggestion.
Hionestly if polling for the bit like this, add a counter and exit when this
reaches some value. Otherwise you are asking for the bus to get hung
is a bit is missed.... |
Yeah, I have some "i2c is stuck" code but that's only useful if it's a case where comm gets out-of-sync and the Master is reading more than what the firmware has to reply.
I went down the "I2C bus reset" rabbit hole a year ago, seeing it mentioned and documented with ways to un-stick the lines. But, sadly, FTDI provides no way to do this from their Windows driver/API so we could never implement it :( _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Wed Dec 02, 2020 2:58 pm |
|
|
Ttelmah wrote: | One suggestion.
Set your ISR to NOCLEAR, and clear the interrupt bit yourself immediately
after you have read I2C_ISR_STATE.
It may be that the presence of the non cleared interrupt bit masks off the
stop bit detection.
Would actually make sense, since the assumption is that this will only
be used outside the ISR. |
This does appear to work! Thanks for the nudge! I'll do some testing, but I am finally seeing my "P" code running. _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19587
|
|
Posted: Thu Dec 03, 2020 2:02 am |
|
|
Good.
I2C has two possible 'stuck' states. The first is when the SDA line stops
responding. The clear for this is for the master to send 9 or more clock pulses
until SDA goes high. The second is when a slave holds SCL low. The reset
for this is to reset the slave device(s). However you can also implement a
default bus timeout as SMBUS does. With the slave devices starting a timer
when the 'start' is seen, resetting it for every received byte, and if it times
out, resetting the I2C peripheral. The minimum for this should be 10mSec
and a value like 50mSec s a very nice safe figure. This ensures the slaves
can't remain stuck holding SCL low.
As a comment, if you are making INT_I2C 'hang' waiting for the P bit
in the interrupt, I'd probably suggest you enable nested interrupts, and
set this to a priority below the other interrupt handlers, otherwise you
risk this interfering with these..... |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Thu Dec 03, 2020 9:26 am |
|
|
Ttelmah wrote: |
As a comment, if you are making INT_I2C 'hang' waiting for the P bit
in the interrupt, I'd probably suggest you enable nested interrupts, and
set this to a priority below the other interrupt handlers, otherwise you
risk this interfering with these..... |
Always a good idea to me. "Program as if someone's life depended on it" is my motto. (https://subethasoftware.com/2020/01/16/code-as-if-someones-life-depended-on-it/).
But we always inherit code that was written assuming success, instead of failure. I've been coding since 1982, and I'm still not confident anything I write is going to work 100% of the time ;-)
The PIC firmware I work on doesn't make use of the watchdog timer, either, which I want to add. That, along with a bootloader (so you can always recover in case of a bricked update process) will go a long way to taking care of unexpected issues. _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
allenhuffman
Joined: 17 Jun 2019 Posts: 588 Location: Des Moines, Iowa, USA
|
|
Posted: Fri Dec 04, 2020 11:32 am |
|
|
Ttelmah wrote: | Set your ISR to NOCLEAR, and clear the interrupt bit yourself immediately after you have read I2C_ISR_STATE. |
The alpha software for the Salea logic analyzer is awesome. It scrolls the data realtime.
I have a situation where the Master (PC using FTDI chipset and drivers) tries to write to a PIC24FJ64GA002 device. This PIC operates as a Slave, but also runs as Master for a sub-bus to talk to other PIC24 devices (multiplexer).
I see the Master does a write to 0x6e (the MUX address) and it is ACK's. The PIC must have seen it.
Then, all following bytes are NAK'd.
After the message, the Master tries to Read a response (it should not, but apparently there is either a bug in the Master code or no error is being returned from the FTDI drivers) and the bus locks (slave side).
In my PIC ISR, I turn on an LED before any read or write, and then turn it off at the stop bit. That LED is never on, which makes it look like my ISR was never called:
(Note: this code includes an infinite while loop with the emergency escape code removes so I can use a satus LED to see if it ever hangs while debugging.)
Code: |
#word I2C2STAT = getenv("SFR:I2C2STAT")
#bit P = I2C2STAT.4 // bit 4 - stop
#word IFS3 = getenv("SFR:IFS3")
#bit I2C_SYSTEM_BUS_IRQ_PENDING_BIT = IFS3.1 // bit 1 - slave I2C2 IRQ status
#INT_SI2C2 NOCLEAR LEVEL=7
void si2c2_isr (void)
{
int state = i2c_isr_state (SYSTEM_BUS);
I2C_SYSTEM_BUS_IRQ_PENDING_BIT = 0; // NOCLEAR, so we manually clear.
if (state == RECEIVE_I2C)
{
slaveIndexRX = ZERO;
LED_SystemOn ();
// Get a byte of data which is the address of the board.
boardAddress = i2c_read (SYSTEM_BUS);
}
...etc...
|
For some reason, my code never ran this to read the address byte. I am investigating places where interrupts are masked.
Since the LED is off at this point, that means the previous message went through fine, since I clear at the end of a message when the stop bit is seen:
Code: |
...snip...
// Receives a message from the master.
else if (state > RECEIVE_I2C)
{
// Store the message in the RX buffer.
rxSlaveBuffer[slaveIndexRX++] = i2c_read (SYSTEM_BUS);
// Wait for either a stop bit, or another incoming data byte.
while ((I2C_SYSTEM_BUS_IRQ_PENDING_BIT == 0) && (P == 0));
if (P != 0) // I2C Stop bit seen.
{
LED_SystemOff ();
rxMsgDataSlave.numBytes = slaveIndexRX; // Update numBytes.
// Message is only received when expected bytes have come in.
runTimeVariables.msgStatus = RX_MESSAGE_VERIFY;
}
}
else
{
//Do nothing.
}
|
That tells me it got past while and shut the LED off, but something happens before the next message and the ISR seemingly is not called.
Can i2c_isr_state() ever block? _________________ Allen C. Huffman, Sub-Etha Software (est. 1990) http://www.subethasoftware.com
Embedded C, Arduino, MSP430, ESP8266/32, BASIC Stamp and PIC24 programmer.
http://www.whywouldyouwanttodothat.com ?
Using: 24FJ256GA106, 24EP256GP202 and 24FJ64GA002. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19587
|
|
Posted: Fri Dec 04, 2020 11:44 am |
|
|
Er. Level=7, is the highest priority. Not what is wanted.....
Level=1. |
|
|
|
|
You cannot post new topics in this forum You cannot reply to topics in this forum You cannot edit your posts in this forum You cannot delete your posts in this forum You cannot vote in polls in this forum
|
Powered by phpBB © 2001, 2005 phpBB Group
|