Hi, ich habe da irgendwie ein Verständnisproblem bei der UART Initialisierung und zwar mit dem URSEL-bit. Mir ist zwar klar das durch dieses Bit eine Umschaltung zwischen UCSRC und UBRRH erfolgt, da beide die gleiche Adresse besitzen. Doch irgendwie gibt es bei der Initialisirung Probleme. Warum funktioniert z.B. UCSRC |= (1 << URSEL); UCSRC |= (1 << UCSZ1) | (1 << UCSZ0); nicht! und UCSRC |= (1 << URSEL)|(1 << UCSZ1) | (1 << UCSZ0) funktioniert. Folgende Anweisung funktioniert ebenfals nicht. //8-Datenbits UCSRC |= (1 << URSEL)|(1 << UCSZ1) | (1 << UCSZ0) //even-parity UCSRC|= (1 << URSEL) |(1 << UPM1) | (0 << UPM0); Gruß Jens
Hallo, dein Problem ist, das die verwendeten C-Operationen immer erst das Register lesen, dann die entspr. Bits setzten und anschl. schreiben. Laut Datenblatt erfolgt beim 1. lesen immer die Rückgabe von UBRRH und beim 2. lesen die Rückgabe von UCSRC. Die 2 Befehle müssen unbedingt direkt nacheinander ausgeführt werden! Da du ja die Schnitttelle im allgem. nur einmal einstellst, und die Register nur für den UART da sind schreibe lieber: UCSRC = (1 << URSEL)|(1 << UCSZ1) | (1 << UCSZ0) das umgeht das vorherige lesen des Registers. Sascha
Hallo, ich hatte das gleiche Problem! Am einfachsten ist das von Sascha erwähnte verfahren alles in einen Aufruf zu packen, so spart man sich das lesen. Oder man definiert sich ein Hilfsregister. z.B. uint8_t UCSRC_temp=0x0; UCSRC_temp|=(1 << URSEL); UCSRC_temp |= (1 << UCSZ1) | (1 << UCSZ0); UCSRC = (1 << URSEL)| UCSRC_temp;
Danke für die Hilfe. Das mit dem lesen der Register muss ich übersehen haben.
Hallo, auch wenn es jetzt überflüssig ist (und ich weiß, warum ich ASM bei µC lieber habe ;) Stell Dir einfach bei solchen Sachen vor, was Du programmierst und was der Compiler daraus machen kann: UCSRC |= (1 << UCSZ1) | (1 << UCSZ0); Du sagst ihm, er soll den Inhalt von UCSRC mit dem Ergebnis der rechten Seite oder verknüpfen. Dazu muß er den Wert von UCSRC ja erstmal haben. Wie bekommt er ihn? Durch einlesen... Es muß also was in der Art rauskommen: in r16,UCSRC ori r16,(1 << UCSZ1) | (1 << UCSZ0) out UCSRC,r16 Nur hier weiß ich, daß ich lesen muß, weil es einen ASM-Befehl, der logische Verknüfungen mit I/O-Ports macht, beim AVR nicht gibt. Gruß aus Berlin Michael
Hi @Michael Wenn ich irgendwelche Bits erhalten muss, ist ein Einlesen notwendig. Es gibt aber auch genügend Fälle, in dem man ein Register 'hart' überschreiben kann. Dann ist ein Einlesen unnötig. MfG Spess
Hallo, logisch. Mir ging es um die spezielle C-Problematik, daß ein OR mit einem Portregister den Compiler dazu zwingt, einzulesen. Egal, ob es nötig wäre oder ob es, wie im konkreten Beispiel, sogar Nebenwirkungen hat. Das ist meiner Meinung nach für einen C-Programmierer, der bei µC erst einsteigt oder jemanden. der beides beginnt, nicht immer ersichtlich. Wenn ich eine normale Variable und kein Portregister habe, steht ja das Problem auch nicht, darüber nachzudenken. Deren Wert hat ja der Compiler sozusagen immer vorrätig. Gruß aus Berlin Michael
Bitte melde dich an um einen Beitrag zu schreiben. Anmeldung ist kostenlos und dauert nur eine Minute.
Bestehender Account
Schon ein Account bei Google/GoogleMail? Keine Anmeldung erforderlich!
Mit Google-Account einloggen
Mit Google-Account einloggen
Noch kein Account? Hier anmelden.