-
Notifications
You must be signed in to change notification settings - Fork 60
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
First write on I2C bus never NACK's #132
Comments
I can confirm that this problem does not occur when using the bitbang-hal. Note: bitbang hal has currently a bug where all empty transactions are ACK'd. This is fixed here: sajattack/bitbang-hal#22 |
Not to spoil your fun but a bitbang implementation doesn't really have any relevance here. Any reason why you're not using the internal pull-ups? |
Good point. I am using external ones on my PCB, so I thought the internal ones are not required. |
I've just pulled up a F072 Nucleo and some I2C hardware... Let's see whether I can find something. Which version did you test? Master? |
As per my Cargo.toml: 0.17.1 |
I can confirm the behaviour. On an empty write a NACK to address 0 is sometimes not detected/reported by the hardware. Checking with the RM I don't see any way to improve that. What's preventing you from actually sending some data? |
Can you check whether you can reproduce with other addresses, too? 0 is general call address so might be special. Sending no data is actually necessary for smbus quick command. And if I do send some data in the scan, some device on the bus might actually interpret it. Sending address with no data is also sometimes required for waking up devices from sleep mode (I think). And, every i2c scanner works like this :) |
I haven't tried. I have tried a better version of my I2C scanner though and that one works by sending a byte of data which did work great.
SMBus is a different mode and a bit special in a lot of regards (like defining a timeout).
No idea. Sounds like undefined behaviour to me.
Nothing really. It doesn't say anything special about NBYTES == 0. It does read a bit as if that case was not really considered though. I also checked the errata sheet and there're half a ton (scary) I2C related errata but nothing sounding like this particular behaviour. |
Referring to this: https://github.com/barafael/smbus-graph/blob/main/smbus-graph.pdf i2cdetect man page does not recommend scanning either with just write-byte or with just quick-write: https://linux.die.net/man/8/i2cdetect Regardless of this issue - do you confirm this behaviour does only occur if writing 0 bytes? |
It does suggest not to use SMBus methods to scan the bus. I don't see anything saying it will not attempt to write to the bus.
I can confirm I've only seen it when writing 0 bytes and writing more than 0 bytes fixes it. I also haven't seen anything in the manual that would allow me to change anything about that; I can only check for a NACK after sending a START condition which I can only do after setting the address and configuring NBYTES. If there's indeed an (hardware) implementation error causing only the first 0-bytes write to auto-NACK then the only "fix" I can think of would be to do just that during initialisation. But then again, if we did that we'd be writing random values to the bus which according to you is dangerous so it's definitely not something I'd consider. |
Writing one byte to the bus is an smbus method btw :) I'll check some C implementations (mbed and ST HAL) and possibly check with my ST contact. Thank you for taking the time and effort! |
Why would that be an SMBus method? 🤔 And what is the point of using I2C if you're not going to transmit anything at all? Also I have not see scanning documented anywhere in the specification, IIRC someone came up with this clever (ab-)use of I2C and everyone else copied the idea because it can be handy at times. I think the
|
If this is a hack, should the |
What isn't a hack? 😅 Maybe we can improve the documentation to set the expectations right? |
Is there a solution to the problem now? |
Problem:
When performing the first write transaction of the i2c scanner example, there is never a NACK, even if no device is attached on the bus.
Hardware required:
1x STM32F072 Nucleo
2x appropriate (for example 4.7KOhm) pullup resistor (alternatively, use any known OK I2C breakout board with a known I2C address which already includes pullups)
1x https://github.com/stm32-rs/stm32f0xx-hal/blob/master/examples/i2c_find_address.rs (for easier pin access, you may want to change the SDA pin to be PB9 instead of PB7)
Steps to reproduce:
Wire up the pull up resistors from SCL/SDA to 3v3.
Run the i2c scanner example: https://github.com/stm32-rs/stm32f0xx-hal/blob/master/examples/i2c_find_address.rs
_devices
== 0_devices
== 1, ACK at address 0stm32f0xx-hal/examples/i2c_find_address.rs
Line 32 in fba9834
_devices
== 0_devices
== 1, ACK at address e.g. 6let _ = i2c.write(0x4, &[]);
and re-run the scan._devices
== 0_devices
== 0 (yay!)_devices
== 1_devices
== 2The text was updated successfully, but these errors were encountered: