Atmega8 - použití reset pinu jako výstupu

Diskuze a poradna o programátorech a programování různých obvodů

Moderátor: Moderátoři

Odpovědět
Zpráva
Autor
Uživatelský avatar
Dan555
Příspěvky: 36
Registrován: 03 bře 2016, 01:00

Atmega8 - použití reset pinu jako výstupu

#1 Příspěvek od Dan555 »

Zdravím. Poradili byste mi prosím, jak v prostředí arduino IDE ovládat pin atmegy8 č.1 (reset) jako výstup? Potřebuji v setup() nadefinovat pin jako výstupní a pak jej v loop() ovládat. Jde mi jen o fragmenty kódu v jazyku C, které by šly použít v programovacím prostředí arduino IDE. Pojistky přepínat umím a vysokonapěťově mazat obvod umím též. Ale jazyk C se teprve učím...

Uživatelský avatar
FHonza
Příspěvky: 1443
Registrován: 20 lis 2012, 01:00
Bydliště: Praha

#2 Příspěvek od FHonza »

Úplně jednoduše to nejde, Arduino tento port "nemapuje", aby šel ovládat pomocí pinMode(), digitalWrite(), ...
Musíš buď pomocí PINC, DDRC a PORTC nebo upravit zdrojové texty Arduina. Konkrétně v pins_arduino.h dvě konstanty PROGMEM digital_pin_to_port_PGM[] a PROGMEM digital_pin_to_bit_mask_PGM[]

Kód: Vybrat vše

const uint8_t PROGMEM digital_pin_to_port_PGM[] = {
	PD, /* 0 */
	PD,
	PD,
	PD,
	PD,
	PD,
	PD,
	PD,
	PB, /* 8 */
	PB,
	PB,
	PB,
	PB,
	PB,
	PC, /* 14 */
	PC,
	PC,
	PC,
	PC,
	PC,
   PB, /* 20  - PB6 */
   PB, /* 21  - PB7 */
   PC, /* 22  - PC6 */
};

const uint8_t PROGMEM digital_pin_to_bit_mask_PGM[] = {
	_BV(0), /* 0, port D */
	_BV(1),
	_BV(2),
	_BV(3),
	_BV(4),
	_BV(5),
	_BV(6),
	_BV(7),
	_BV(0), /* 8, port B */
	_BV(1),
	_BV(2),
	_BV(3),
	_BV(4),
	_BV(5),
	_BV(0), /* 14, port C */
	_BV(1),
	_BV(2),
	_BV(3),
	_BV(4),
	_BV(5),
	_BV(6), /* 20 - PB6 */
	_BV(7), /* 21 - PB7 */
   _BV(6), /* 22 - PC6 */
};
V této úpravě je navíc ještě pin 20 a 21 namapován na PB6 a PB7 (XTAL1, XTAL2) pro desky, které používají interní RC oscilátor. PC6 (RESET) je pak pin 22.

Nezapomeň, že jakmile pomocí fuses "odstavíš" reset, tak nenahraješ nic přes bootloader ani přes ISP.

Uživatelský avatar
ross4435897
Příspěvky: 426
Registrován: 10 lis 2009, 01:00
Bydliště: Brno

#3 Příspěvek od ross4435897 »

... Nezapomeň, že jakmile pomocí fuses "odstavíš" reset, tak nenahraješ nic přes bootloader ani přes ISP.
To je myslim to nejpodstatnejsi ...

Uživatelský avatar
Dan555
Příspěvky: 36
Registrován: 03 bře 2016, 01:00

#4 Příspěvek od Dan555 »

Díky za radu i ohledně pinů krystalu. To mi také občas chybělo. S tím DDRC a PORTC, šlo by to takto? Chybí mi tam ale PINC...

setup() {
DDRC |= (1<<6); // nastaveni pc6-reset jako vystupni
...
}

loop() {
PORTC |= _BV(PC6); // reset do H
...

PORTC &= ~_BV(PC6); // reset do L
...
}

Uživatelský avatar
FHonza
Příspěvky: 1443
Registrován: 20 lis 2012, 01:00
Bydliště: Praha

#5 Příspěvek od FHonza »

Takto by to šlo, jenom bych čistě pro přehlednost zapsal:
DDRC |= (1<<PC6);

PINC použiješ jenom v případě kdyby příslušný pin byl vstupní (odpovídá funkci digitalRead() ).

To fakt už nemáš žádnou další volnou nožičku, že se chceš vzdát všech výhod ISP ? Budeš muset pokaždé vyndavat procesor ... Co spíš zvolit variantu s interním RC oscilátorem a použít PB6 a PB7 ? Osobně jsem se vždy "ubránil" odstavení resetu.

Uživatelský avatar
Dan555
Příspěvky: 36
Registrován: 03 bře 2016, 01:00

#6 Příspěvek od Dan555 »

FHonza píše:Takto by to šlo, jenom bych čistě pro přehlednost zapsal:
DDRC |= (1<<PC6);

PINC použiješ jenom v případě kdyby příslušný pin byl vstupní (odpovídá funkci digitalRead() ).

To fakt už nemáš žádnou další volnou nožičku, že se chceš vzdát všech výhod ISP ? Budeš muset pokaždé vyndavat procesor ... Co spíš zvolit variantu s interním RC oscilátorem a použít PB6 a PB7 ? Osobně jsem se vždy "ubránil" odstavení resetu.
Atmegou řídím přímo dva dvojité displeje (viz. http://www.elweb.cz/clanky.php?clanek=105 , zde je to řízené PICem) a v tomto řešení moc místa na optimálnější propojení nebylo - na rozdíl od článku jsem na jednotlivé segmenty ještě i dával smd odpory, ale i tak se mi to málem na danou velikost desky nevešlo. Naštěstí mám na resetu desetinnou tečku, takže to nebrání ladění i bez ní, ovládat jí budu až na závěr. Vím, že je třeba pro nové naprogramování čip nejdříve vysokonapěťově smazat, eraser na to mám a používám ho.
Při ladění jsem narazil ještě na jeden zajímavý problém - prostředí IDE mi brání použít přerušení od timeru0, protože ho používá a definuje soubor wiring.c. Při pokusu o vytvoření obsluhy přerušení ISR(TIMER0_OVF_vect) mi prostředí IDE píše problém s dvojí deklarací. Nevíte, co s tím? Jak jinak udělat programově přerušení od přetečení čítače v prostředí IDE pro atmega8, pokud nechci prostředí IDE modifikovat? U atmega328p to jde, ale to má interruptů od přetečení čítačů víc...

Uživatelský avatar
FHonza
Příspěvky: 1443
Registrován: 20 lis 2012, 01:00
Bydliště: Praha

#7 Příspěvek od FHonza »

Co použít Timer2 ? Ten je použit pouze funkcí tone().

Osobně bych místo "obsazení" resetu použil MAX7219 a řídil ty segmentovky po třech drátech. Nebo zapoj všechny čtyři segmenty na společný multiplex.

MAX7219 ti můžu poslat (jak v DIL, tak SMD)

Uživatelský avatar
Dan555
Příspěvky: 36
Registrován: 03 bře 2016, 01:00

#8 Příspěvek od Dan555 »

FHonza píše:Co použít Timer2 ? Ten je použit pouze funkcí tone().

Osobně bych místo "obsazení" resetu použil MAX7219 a řídil ty segmentovky po třech drátech. Nebo zapoj všechny čtyři segmenty na společný multiplex.

MAX7219 ti můžu poslat (jak v DIL, tak SMD)
MAX7219 používám, je to skvělá součástka pro řízení displeje. Mne ale problém s přerušením zaujal hlavně proto, že bych rád uměl na atmega8 s přerušením pracovat. Jak prosím nadefinovat ten timer2 v arduino IDE, aby přerušoval přetečením? Mi se to bohužel nepodařilo...

Uživatelský avatar
FHonza
Příspěvky: 1443
Registrován: 20 lis 2012, 01:00
Bydliště: Praha

#9 Příspěvek od FHonza »

Stejně: ISR(TIMER2_OVF_vect)

nebo se dá použít knihovna MsTimer2. Definuje jednoduchý objekt, do kterého se dá "podstrčit" callback funkce. Viz tady

Uživatelský avatar
Dan555
Příspěvky: 36
Registrován: 03 bře 2016, 01:00

#10 Příspěvek od Dan555 »

FHonza píše:Stejně: ISR(TIMER2_OVF_vect)

nebo se dá použít knihovna MsTimer2. Definuje jednoduchý objekt, do kterého se dá "podstrčit" callback funkce. Viz tady
Wow, skvělé. Už to funguje:

TCCR2 = _BV(CS21) | _BV(CS20);
TIMSK = _BV(TOIE2);
sei();

...

ISR(TIMER2_OVF_vect)

...

Odpovědět

Zpět na „Programování PIC, ATMEL, EEPROM a dalších obvodů“