Portál AbcLinuxu, 13. května 2025 20:05

Dotaz: funkce poll() a timeout při nečinnosti sériového portu - jazyk C

21.3.2010 19:37 J@rys | skóre: 11 | blog: Jarduv_blog
funkce poll() a timeout při nečinnosti sériového portu - jazyk C
Přečteno: 315×
Odpovědět | Admin
Ahojte. Potřeboval bych poradit. Vytvořil jsem fungujici aplikaci v C pro posilani a prijem dat ze serioveho portu. Aplikace jako celek funguje bezvadne. Jedine co nefunguje je timeout u funkce poll(). Potrebuji totiz, aby pokud nastane necinnost na seriovem portu po dobu 30s aplikace provedla resend dat odzacatku. Problem je jaksi v tom, ze se mi timeout ze zahadneho duvodu nedari naplnit. Misto toho zustane aplikace stat na volani poll() a je tedy jakoby zamrzla. U funkce select(), kterou jsem zkousel ve vedlejsi verzi tento problem nemam. Ale select() je pro me nedostacujici, nebot nedokaze rozlisit, zda jde o udalost "odeslana" nebo "prijata" data. Zkusil jsem to u ni trochu obejit, ale nedarilo se mi. Zde jsou fragmenty kodu s funkci poll():

//Otevreni portu pro cteni a zapis se zamezenim kontroly tarminalu: fd = open(argv[2], O_RDWR | O_NOCTTY );

//Ulozeni aktualnijho nastaveni terminalu pro otevreny com port: tcgetattr(fd,&oldtio); //Naplneni struktury s novym nastavenim nulami: bzero(&newtio, sizeof(newtio)); //Nastaveni konfigurace: prenosove rychlosti, hardveroveho rizeni RTSCTS, 8bitu, ignorovat modem control lines, povoleni prijmu newtio.c_cflag = BAUDRATE | CS8 | CLOCAL | CREAD;

//Vstupni zpracovani - ignoruji se chyby parity a vzorkovani, je zapnuto interpretace carriege return na novy radek: newtio.c_iflag = IGNPAR | IGNCR; //ICRNL, ne IGNCR

//Vystupni zpracovani zadne pro odchozi data: newtio.c_oflag = 0; //Kanonicky vstup - konce radku \n: newtio.c_lflag = ICANON;

//Zrusi vsechna zapsana, ale neodeslana data a prijata, ale neprectena: tcflush(fd, TCIFLUSH);

//Provedeni zmeny nastaveni: tcsetattr(fd,TCSANOW,&newtio);

//Specifikace monitorovani deskriptoru: struct pollfd fds[1];

//Timeout, ktery vyprsi, pokud nenastane zadna udalost na fd [ms]: int timeout = 20000;

//Navratova hodnota funkce poll: int ret; //Nastaveni monitorovaneho fd: fds[0].fd = fd;

//Nastaveni priznaku, ktere budeme monitorovat (cteni zapis): fds[0].events = POLLIN | POLLOUT;

//Maska nastalych udalosti na 0 (zadna udalost dosud nenastala): fds[0].revents = 0; //Monitorovani filedescriptoru: ret = poll(fds, 1, timeout);

Podotykam jeste, ze preklad je bez Warningu.

Řešení dotazu:


Nástroje: Začni sledovat (1) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

Řešení 1× (J@rys (tazatel))
22.3.2010 00:47 Jirka P
Rozbalit Rozbalit vše Re: funkce poll() a timeout při nečinnosti sériového portu - jazyk C
Odpovědět | | Sbalit | Link | Blokovat | Admin
1) Nafomátujte ten příspěvek (<pre>...), takhle se to nedá číst.

2) select()em se dá rozlišit, jestli data přišla nebo odešla (viz parametry readfds, writefds)

3) Možná je to nějaké chyba v jádře, mohl byste napsat konfiguraci? Zkuste, jestli ten timeout funguje bez filedeskriptorů.

4) Zkuste volání ppoll().

5) Zkuste použít timerfd místo timeoutu.

6) (poslední zoufalý krok) Můžete se nechat probudit signálem, viz alarm(). Bohužel, tohle většinou znamená race (s obyč. select()em).
22.3.2010 18:42 pht | skóre: 48 | blog: pht
Rozbalit Rozbalit vše Re: funkce poll() a timeout při nečinnosti sériového portu - jazyk C
Správně, takhle se to nedá přečíst.
In Ada the typical infinite loop would normally be terminated by detonation.

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.