- 3D-Druck Einstieg und Tipps         
Seite 1 von 2 12 LetzteLetzte
Ergebnis 1 bis 10 von 11

Thema: I2C Kommunikation zwischen Nucleo STM32F103RBT6 und Sensor Honeywell HIH9120

  1. #1
    Neuer Benutzer Öfters hier
    Registriert seit
    30.04.2012
    Beiträge
    13

    I2C Kommunikation zwischen Nucleo STM32F103RBT6 und Sensor Honeywell HIH9120

    Anzeige

    Powerstation Test
    Hallo zusammen,

    ich versuche ein Temperatur- und Feuchtigkeitssensor mit dem Entwicklungsboard Nucleo STM32 zu verbinden. Dieser Sensor lässt sich mit I2C ansprechen und die Adresse ist laut Datenblatt 0x27. Nun arbeite ich mit dem mbed Compiler und habe folgendes Programm zur Kommunikation.

    Mein Programmcode sieht wie folgt aus:
    #include "mbed.h"
    I2C i2c(PB_11, PB_10);

    const int addr = 0x27;

    int main() {
    char cmd[2];
    while (1) {
    cmd[0] = 0x01;
    cmd[1] = 0x00;
    i2c.write(addr, cmd, 1);

    wait_ms(10);
    }
    }


    Mit einem Logic Analyzer nehme ich die Kommunikation auf um mein Problem zu beheben und stelle folgendes fest:
    - Bei einer eingegebenen Adresse addr=0x26 spricht der Mikrocontroller eine Adresse 0x26 an
    - Bei einer eingegebenen Adresse addr=0x27 spricht der Mikrocontroller eine Adresse 0x28 an
    - Bei einer eingegebenen Adresse addr=0x28 spricht der Mikrocontroller eine Adresse 0x28 an

    Das ist in meinen Augen extrem merkwürdig und ich kann mir da nicht weiterhelfen. Hat jemand Erfahrung mit dem Entwicklungsboard und kann mir bei dem Fehler auf die Sprünge helfen?

    Danke für Hilfestellungen im voraus!

    Angehangen das Screenshot des Logic Analyzers bei addr=0x27

    Klicke auf die Grafik für eine größere Ansicht

Name:	screenshot.jpg
Hits:	10
Größe:	55,9 KB
ID:	30223
    Geändert von Klacknack (02.06.2015 um 05:37 Uhr)

  2. #2
    Erfahrener Benutzer Roboter Experte
    Registriert seit
    04.09.2011
    Ort
    Hessen
    Beiträge
    707
    Hallo,

    eigentlich hat eine I2C-Adresse nur 7 Bit, das 8. zeigt an, ob gelesen oder geschrieben wird. Die mbed Implementierungen von read und write mit mehreren Parametern erwarten eine 8-Bit Adresse. Da steht ja auch in der Doku
    The mbed API uses 8 bit addresses, so make sure to take that 7 bit address and left shift it by 1 before passing it.
    bei read
    address 8-bit I2C slave address [ addr | 1 ]
    und write
    address 8-bit I2C slave address [ addr | 0 ]
    Wenn du rohe 8-Bit Werte schreiben willst, kannst du die write Funktion mit einem Parameter verwenden und die Übertragung Byte für Byte in einer for-Schleife machen.
    Geändert von Mxt (02.06.2015 um 07:23 Uhr) Grund: Verschrieben

  3. #3
    Neuer Benutzer Öfters hier
    Registriert seit
    30.04.2012
    Beiträge
    13
    Hallo,

    danke erstmal für die Antwort und entschuldigung dass ich eine Woche darauf nicht reagiert habe, war aber durch andere Baustellen leider völlig eingenommen. Also deine Antwort erklärt das Problem sehr gut. Nun habe ich mir gedacht ich mache es ganz simpel, indem erstmal beginne den Sensor lediglich einen write befehl zu geben. Der Code ist recht simpel und wie folgt:

    #include"mbed.h"

    I2C i2c(PB_11, PB_10); //sda, scl
    const int addr = 0x4E; //Adresse 0x27 um ein Bit nach links geschoben
    double result=0;

    int main() {
    while(1){
    i2c.start();
    i2c.write(addr);
    i2c.stop();
    }
    }

    Das Ergebnis mit dem Logic Analyzer gemessen ist angehangen. Meiner Meinung nach sieht die I2C Kommunikation des Mikrocontrollers richtig aus, jedoch bleibt das Acknowledge des Sensors aus. Ich habe die Beschaltung gefühlt 100 mal kontrolliert und ich kann mir nicht vorstellen dass der Sensor kaputt ist. Kann mir jemand mit Erfahrung in der Kommunikation weiterhelfen und erkennt da einen Fehler?

    Danke
    Miniaturansichten angehängter Grafiken Miniaturansichten angehängter Grafiken screenshot.jpg  

  4. #4
    Erfahrener Benutzer Begeisterter Techniker
    Registriert seit
    14.11.2013
    Ort
    Home
    Beiträge
    213
    Hallo Klacknack,

    kenne Entwicklungsboard Nucleo STM32 nicht. Somit nur eine allgemeine Antwort.

    Bin ein begeisterter Fan von I²C Bus. Die häufigsten Fehler liegen beim Bustakt und der H-Pegel der Leidungen.
    Mit einem Takt bis 100kHz sollten ziemlich alle Sensoren funktionieren. Wichtig sind die Abschlusswiderstände von SDA und SCL zu den Sensoren. Meine Erfahrungen 4,7 k auf Vcc sind bis ca. 20 Meter Leitungslänge OK.
    Leider programmiere ich „nur“ noch mit BASCOM. Hatte schon mal ein Programm eingestellt wie viele unterschiedliche I²C –Teilnehmer am Bus angeschossen und geprüft werden können.
    Wie gesagt, sehr oft liegt es am Aufbau der Hardware, wenn solche Fehler auftreten.
    Softwaremäßig ist es wie bei allen Bussysteme nötig, hat sich der Teilnehmer gemeldet und die Informationen geschickt den Bus wieder frei zu geben[Start ansprechen, lesen oder schreiben Stop].


    Mit freundlichen Grüßen
    fredred

  5. #5
    Erfahrener Benutzer Roboter-Spezialist
    Registriert seit
    08.08.2008
    Ort
    DE
    Beiträge
    523
    Bau dir doch mal einen I2C Scanner (Arduino Beispiel):

    Code:
    1. void loop()
    2. {
    3. byte error, address;
    4. int nDevices;
    5. Serial.println("Scanning...");
    6. nDevices = 0;
    7. for(address = 1; address < 127; address++ )
    8. {
    9. // The i2c_scanner uses the return value of
    10. // the Write.endTransmisstion to see if
    11. // a device did acknowledge to the address.
    12. Wire.beginTransmission(address);
    13. error = Wire.endTransmission();
    14. if (error == 0)
    15. {
    16. Serial.print("I2C device found at address 0x");
    17. if (address<16)
    18. Serial.print("0");
    19. Serial.print(address,HEX);
    20. Serial.println(" !");
    21. nDevices++;
    22. }
    23. else if (error==4)
    24. {
    25. Serial.print("Unknow error at address 0x");
    26. if (address<16)
    27. Serial.print("0");
    28. Serial.println(address,HEX);
    29. }
    30. }
    31. if (nDevices == 0)
    32. Serial.println("No I2C devices found\n");
    33. else
    34. Serial.println("done\n");
    35. delay(5000); // wait 5 seconds for next scan
    36. }

    mfg

  6. #6
    Erfahrener Benutzer Begeisterter Techniker
    Registriert seit
    14.11.2013
    Ort
    Home
    Beiträge
    213
    Hallo Wsk8,

    Ein Scanner wie du schreibst ist okay. Mein Hinweis sollte nur Vermitteln, ist die Hardware „Dumm“, kann eine Software, nicht den Fehler eindeutig erkennen. Die vielen guten Softwarelösungen gehen ja immer davon aus, es ist ein Fehler im Programm. Oft sind es aber nur kleine Fehler, wenn ein neues Modul nicht korrekt angeschlossen wird.

    Gruß fredred

  7. #7
    Neuer Benutzer Öfters hier
    Registriert seit
    30.04.2012
    Beiträge
    13
    Danke für beide Vorschläge.

    Zur Hardware: Ich habe auch neben 100kHz andere Frequenzen ausgetestet, jedoch stelle ich da kein Unterschied fest. Datenblatt des Sensors sagt aus, dass alles zwischen 100kHz und 400kHz in Ordnung ist. Die Pull-Up Widerstände betragen 1,8kOhm, nach Datenblatt sollten es 2,2kOhm sein. Da hab ich auch schon variiert. Jedoch wiederum kein Unterschied. Die Entstör-Kondensatoren betragen den verlangten Wert und ich kann mir wirklich nicht vorstellen, dass es daran liegt. Der Aufbau ist auf einem Breadboard und ich überlege momentan ob ich den Testaufbau auf eine Lochrasterplatine löten soll. Mir fällt kein präziser Grund ein, denn beim Kontrollieren mit Multimeter oder Logic Analyzer waren alle Pegel auf dem Breadboard richtig, nur aus Erfahrung weiß ich, dass die Dinger nicht immer 100% zuverlässig sind. Als letztes fällt mir noch ein, einen anderen Sensor auszutesten. Vllt habe ich diesen beim Testen in irgendeiner Weise zerstört.

    Dann werde ich mich als nächstes mal mit der Software befassen und mir den I2C Scanner ansehen. Zurzeit programmiere ich das Nucleo Board online mit dem mbed Compiler, weil ich eine recht kleine Aufgabe zu implementieren habe und ich keine Lust hatte mir eine IDE auf dem PC einzurichten. Vielleicht sollte ich dann doch dazu übergehen.

    Ich halte euch auf dem laufenden was die Ergebnisse der nächsten Tage ergeben.

    Danke, Gruß!

  8. #8
    Erfahrener Benutzer Roboter Experte
    Registriert seit
    04.09.2011
    Ort
    Hessen
    Beiträge
    707
    An sich funktioniert I2C mit mbed ziemlich gut. Ich habe zwar nicht genau den Nucleo aus dem Titel, sondern einen F401, aber da sollte kein großer Unterschied sein.

    Mit der normalen 400 kHz Einstellung habe ich verschiedene Sensoren und auch EEPROMs an verschiedenen mbed im Einsatz.

  9. #9
    Erfahrener Benutzer Begeisterter Techniker
    Registriert seit
    14.11.2013
    Ort
    Home
    Beiträge
    213
    Hallo Klacknack,

    wie geschrieben, ist die Hardware (Busleitungen okay), kannst Du deine Sensoren kaum zerschießen.
    Mein Vorschlag ist.
    Teste den I²C erst mal mit einem Digital- IC z B. PCF 8574. Der muss ja nicht erst Messdaten aufbereiten um Status „ich bin fertig“ zu generieren damit es weiter geht. Wenn sich IC meldet weist du erst mal, der Bus funktioniert. Rest ist die Softwareanpassung für die unterschiedlichen Teilnehmer.
    Viele Sensoren sind meist für ein bestimmtes System entwickelt wurden. Somit sind Laufzeiten und Flanken zu beachten.
    Das I²C Prodokoll ist zwar kaum noch in der Industrie im Einsatz, aber für Amateure wie ich es bin, leicht zu handhaben.
    Beispiel: Meine Testplatine hat 40 Sockel für Test der I²C IC’s.
    Bei voller Bestückung..

    1. 16x Digital IC 64 Ports zum Schreiben und 64 zum Lesen ein IC ist Master für LCD-Anzeige 20x4 eingestellt.(Hat keinen I²C Anschluss). Also Modus abbilden
    2. 8x Analog IC gleich 32 Ports
    3. 8x Digital Poti 16 mal analog Out
    4. 4x EEprom (256 k) für Datenlogger.


    Werden alle Ereignisse mit ATmega 644P (Prozessortakt 16 MHz) generiert, ist die Laufzeit < 1 Sekunde.
    Muss sagen für viele Anwendungen ist dies ausreichend.
    Natürlich sind in der Software die Auswertungen, Anzeigen und Berechnungen für Laufzeiten zu beachten.

    Gruß
    fredred

  10. #10
    Erfahrener Benutzer Roboter Experte
    Registriert seit
    04.09.2011
    Ort
    Hessen
    Beiträge
    707
    Hallo,

    ich habe in meiner Sammlung ein kleines Beispiel für einen 24LC256 EEPROM, ist allerdings für den LPC1768 konfiguriert. Man muss die Pin Bezeichnungen für einen Nucleo anpassen, auch hat der nur eine LED. Aber es ist das schöne an mbed, dass die Programme mit so wenigen Änderungen auf Cortex-M verschiedener Hersteller laufen.

    Headerdatei
    Code:
    #ifndef I2C_EEPROM_H
    #define I2C_EEPROM_H
    
    #include <vector>
    
    //
    // Klasse für 24LC256 usw.
    //
    class I2C_Memory {
    private:
        int m_Address;
        I2C m_i2c;
        bool m_Ok;    
        
        //
        // Übertragungsfehler merken
        //
        void DoWrite(int value) {
            if (!m_i2c.write(value))
                m_Ok = false;
        }
        
    public:
    
        //
        // Konstruktor
        //
        I2C_Memory(int address, PinName sda, PinName scl) : m_Address(address), m_i2c(sda, scl), m_Ok(true) 
        {}   
        
        //
        // Schreiben
        //
        bool Write(unsigned short start_address, const std::vector<uint8_t>& data) {
            m_Ok = true;
            
            char hb = (start_address & 0xff00) >> 8;
            char lb = start_address & 0xff;
            
            m_i2c.start();
            
            DoWrite(m_Address);
            DoWrite(hb);
            DoWrite(lb);
            
            for(unsigned int i = 0; i < data.size(); i++) {
                DoWrite(data[i]);
            }
            
            m_i2c.stop();
            
            return m_Ok;
        }
        
        //
        // Lesen
        //
        bool Read(unsigned int start_address, unsigned int count, std::vector<uint8_t>& data) {
            m_Ok = true;
            data.clear();
            
            char hb = (start_address & 0xff00) >> 8;
            char lb = start_address & 0xff;
            
            m_i2c.start();
            
            DoWrite(m_Address);
            DoWrite(hb);
            DoWrite(lb); 
            
            m_i2c.start();
            
            DoWrite(m_Address + 1);   
            
            for(unsigned int i = 0; i < count - 1; i++) {
                data.push_back(m_i2c.read(1));
            }
            if ( count > 0 ) {
                data.push_back(m_i2c.read(0));   // Letztes Lesen mit NACK quittieren
            }      
            
            m_i2c.stop();                      
            
            return m_Ok;
        } 
    };
        
    #endif
    Beispielprogramm
    Code:
    #include "mbed.h"
    #include "I2C_EEPROM.h"
    
    I2C i2c(p9, p10);
    
    DigitalOut led1(LED1);
    DigitalOut led2(LED2);
    DigitalOut led3(LED3);
    
    int main() {
        I2C_Memory memory(0xa0, p9, p10);
        
        while(1) {
            led1 = 0;
            led2 = 0;
            led3 = 0;
            
            wait(1.0);
            
            std::vector<uint8_t> data;
            
            if ( memory.Read(0, 256, data) ) {
                led1 = 1;
                
                for(int i = 0; i < data.size(); i++) {
                    printf("%0.2x ", data[i]);
                }
                printf("\r\n\r\n");
            }
            else {
                led2 = 1;
            }
            
            wait(5.0);
        }
    }

Seite 1 von 2 12 LetzteLetzte

Ähnliche Themen

  1. Antworten: 0
    Letzter Beitrag: 26.02.2014, 07:09
  2. i2c Kommunikation zwischen ATmega2561 und MD25
    Von loewe.xy im Forum C - Programmierung (GCC u.a.)
    Antworten: 1
    Letzter Beitrag: 23.05.2013, 22:37
  3. Kommunikation zwischen AVR's I2c- Bascom Code?
    Von toter_fisch im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 4
    Letzter Beitrag: 01.04.2012, 20:53
  4. Kommunikation zwischen uP und Pc
    Von XC866 im Forum C - Programmierung (GCC u.a.)
    Antworten: 3
    Letzter Beitrag: 14.02.2007, 14:48
  5. [ERLEDIGT] Kommunikation zwischen AVR und PC
    Von im Forum AVR Hardwarethemen
    Antworten: 8
    Letzter Beitrag: 10.02.2004, 12:26

Berechtigungen

  • Neue Themen erstellen: Nein
  • Themen beantworten: Nein
  • Anhänge hochladen: Nein
  • Beiträge bearbeiten: Nein
  •  

MultiPlus Wechselrichter Insel und Nulleinspeisung Conrad