abclinuxu.cz AbcLinuxu.cz itbiz.cz ITBiz.cz HDmag.cz HDmag.cz abcprace.cz AbcPráce.cz
Inzerujte na AbcPráce.cz od 950 Kč
Rozšířené hledání
×
    dnes 01:55 | Nová verze

    Byla vydána verze 3.6 multiplatformního integrovaného vývojového prostředí (IDE) pro rychlý vývoj aplikaci (RAD) ve Free Pascalu Lazarus (Wikipedie). Přehled novinek v poznámkách k vydání. Využíván je Free Pascal Compiler (FPC) 3.2.2.

    Ladislav Hagara | Komentářů: 5
    dnes 00:33 | Komunita

    Na čem aktuálně pracují vývojáři GNOME a KDE? Pravidelný přehled novinek v Týden v GNOME a Týden v KDE.

    Ladislav Hagara | Komentářů: 1
    včera 15:33 | Nová verze

    Byla vydána nová verze 8.8 multiplatformní digitální pracovní stanice pro práci s audiem (DAW) Ardour. Přehled oprav, vylepšení a novinek v oficiálním oznámení.

    Ladislav Hagara | Komentářů: 1
    včera 14:44 | Nová verze

    Byla vydána nová major verze 11.0.0 nástroje mitmproxy určeného pro vytváření interaktivních MITM proxy pro HTTP a HTTPS komunikaci. Přehled novinek v příspěvku na blogu. Vypíchnuta je plná podpora HTTP/3 a vylepšená podpora DNS.

    Ladislav Hagara | Komentářů: 0
    včera 05:11 | Nová verze

    Richard Hughes na svém blogu představil nejnovější major verzi 2.0.0 nástroje fwupd umožňujícího aktualizovat firmware zařízení na počítačích s Linuxem. Podrobný přehled novinek v poznámkách k vydání. Přehled podporovaných zařízení, nejnovějších firmwarů a zapojených výrobců na stránkách LVFS (Linux Vendor Firmware Service).

    Ladislav Hagara | Komentářů: 0
    4.10. 15:44 | Zajímavý software

    Počítačová hra Kvark (Steam) od studia Perun Creative dospěla do verze 1.0 (𝕏). Běží také na Linuxu.

    Ladislav Hagara | Komentářů: 4
    4.10. 15:22 | Nová verze

    Byla vydána (𝕏) zářijová aktualizace aneb nová verze 1.94 editoru zdrojových kódů Visual Studio Code (Wikipedie). Přehled novinek i s náhledy a animovanými gify v poznámkách k vydání. Ve verzi 1.94 vyjde také VSCodium, tj. komunitní sestavení Visual Studia Code bez telemetrie a licenčních podmínek Microsoftu.

    Ladislav Hagara | Komentářů: 1
    4.10. 11:22 | Pozvánky

    O víkendu 5. a 6. října se koná ne-konference jOpenSpace. Pokud si chcete kouzlo živých přednášek vychutnat společně s námi, sledujte live streamy: sobota a neděle. Začínáme lehce po 9 hodině ranní. Zpracované záznamy jsou obvykle k dispozici do 14 dní na našem YouTube kanále.

    Zdenek H. | Komentářů: 0
    4.10. 11:11 | Humor

    Hodiny s unixovým časem dnes odbily 20 000 dnů. Unixový čas je počet sekund uplynulých od půlnoci 1. ledna 1970. Dnes ve 02:00 to bylo 1 728 000 000 sekund, tj. 20 000 dnů.

    Ladislav Hagara | Komentářů: 1
    4.10. 05:00 | IT novinky

    Notebook NitroPad V56 od společnosti Nitrokey byl oficiálně certifikován pro Qubes OS verze 4. Qubes OS (Wikipedie) je svobodný a otevřený operační systém zaměřený na bezpečnost desktopu.

    Ladislav Hagara | Komentářů: 8
    Rozcestník

    Dotaz: C/C++ presnost usleep

    9.7.2010 08:44 toxLinuch
    C/C++ presnost usleep
    Přečteno: 1797×
    Pekny den panove, programuju api pro cteni teploty z cidla DS18B20 a cidlo se mi neoziva. Postupuje dle datasheetu. Jediny problem, ktery me napada je casovani. Jelikoz jsou pro komunikaci po 1Wire potrebne presne casove useky v radech mikrosekund. Je tedy usleep dostatecne presne? Jedu na CentOS. Pro pripojeni uzivam COM port. Prisup k portu prez ioperm a pak outb inb.
    
    #define PORT 0x3f8
    #define OFFSET_RIZENI_M 4
    
    outb (0x00, PORT+OFFSET_RIZENI_M);    //nuluj registr rizeni modemu
    
    log1 (PORT+OFFSET_RIZENI_M);          // inicializace sbernice a poskytnuti napajeni po dobu 1ms
    usleep (1000);
    
    log0 (PORT+OFFSET_RIZENI_M);          //-|master reset
    usleep (500);                         //_|
                                                                                   
    log1 (PORT+OFFSET_RIZENI_M);          //-uvolnit sbernici
    usleep (70); 
    
    statusCTS=inb (PORT+OFFSET_STAV_M);
    if ((statusCTS & CTS_BIT)==0x00)      //-je nula na sbernici
       printf ("Ozval se\n");
    
    
    

    Řešení dotazu:


    Odpovědi

    9.7.2010 08:54 toxLinuch
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Příloha:
    jeste prikladam schema zapojeni
    Josef Kufner avatar 16.7.2010 10:07 Josef Kufner | skóre: 70
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    A co tam mezi sériový port a samotnou sběrnici vložit ještě nějaký levný málonožičkový jednočip, který by se postaral o řízení sběrnice a převod dat z/na standardní sériový přenos?
    Hello world ! Segmentation fault (core dumped)
    23.7.2010 11:57 Sfinx
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Taky si myslim, ze treba obyc atmega8 by tyhle problemy vyresila. Jednou jsem to resil a mam nekde i fungujici kod pro cteni z cidel a zapis na USART nebo na SD kartu. V pripade zajmu muzu poskytnout.
    9.7.2010 09:11 fraxinus | skóre: 20 | blog: fraxinus
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Pokial nepouzivas realtime kernel, myslim ze spravneho casovania nebude garantovane nikdy
    9.7.2010 10:02 toxLinuch
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Takze mam si stahnout vanila kernel a pak do nej naprat patch na RT prelozit a mam garantovanou presnost?
    9.7.2010 10:22 fraxinus | skóre: 20 | blog: fraxinus
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    No, napr. Ubuntu Studio (8.04, 9.10) ma realtime kernel ale neviem ci ti to pomoze, pisu tam 5-10ms ale neviem coho sa to tyka ci prepinania procesov alebo coho. Najlepsie by bolo zmerat to osciloskopom.
    9.7.2010 13:39 hajoucha | skóre: 22
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    radši si to přečti ještě jednou. Těch 5-10ms je to, co je schopen člověk zaznamenat (asi při poslechu). Na takové doby totiž nepotřebuješ real-time. 5-10ms dostaneš na běžném (tj. _ne_ RT) kernelu.
    9.7.2010 13:41 hajoucha | skóre: 22
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    no, RT-kernel pomůže, pak je ještě potřeba dát svému procesu tu správnou RT prioritu. Jestli to "pomůže dostatečně" tj. jestli Ti pak bude fungovat to monitorování - to netuším. Otázkou však není, jestli to bude nebo nebude fungovat. Otázkou je, jestli máš na výběr.
    21.7.2010 20:32 Peter Fodrek | skóre: 11
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    treba pouzit RTAI https://www.rtai.org/ a hlavne casovace nie sleep

    casovace RTAI maju rozlisenie bud podla HW obvodov (i8255 alebo APIC) alebo podla tikov CPU maximalna nepresnost casovaca RTAI je 15 mikrosekund

    Pripadne sa da priplatit(1999 USD/rok na vyvoj na jednej architekure alebo 4999 USD/rok za 5 architektur) si za RTLinuxPRO, ktory sa teraz vola WindRiver Linux, ktory ma maximlanu nepresnost 48 nanosekund, co je na hranici HW moznosti, ked si uvedomime, ze mozu nastat 3x citanie a jeden zapis do RAM..

    9.7.2010 10:42 M_P
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Mozna bude lepsi neproslapavat uz hotovou cestu a podivat se jak to delaji ti co jim to funguje... viz www.owfs.org

    M.
    10.7.2010 13:34 Radovan
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    10.7.2010 14:06 Tom K | skóre: 21
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Ahoj
    Planovac bezi vetsinou na frekvenci do 1000Hz a tezko se da pocitat s tim, ze se probudi presne za 70us. Na useky do 10000us bych (v pripade, ze se nepouzivaji nejak casto). Radeji pouzival aktivni cekani pres gettimeofday().
    echo -n "u48" | sha1sum | head -c3; echo
    15.7.2010 09:07 toxLinuch
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    chlape kurna moc ti dekuju ani nevys jak moc si mi pomohl.
    Shrnuti: usleep - nepresne (opravdu se to neblizi usekundam
    nanosleep - stejny bazmek jako usleep
    Reseni: gettimeofday v cyklu
    
    void __usleep_test02 (int usecond) {                                                                                                                 
      struct timeval ts,te;                    //ts-time start; te-time end
    
      gettimeofday(&ts, NULL);                 //read start time in usec
      do { 
          gettimeofday(&te, NULL);             //read end time in usec
      } while (te.tv_usec<ts.tv_usec+usecond); //when te(time end) < ts(time start) + usecond
    } 
    casovani je naprosto presne a cip se ozval jak ma.
    16.7.2010 13:18 pht | skóre: 48 | blog: pht
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Ten cyklus je ale špatně protože nepočítá s přetečením tv_usec.

    Jinak přesnost usleep (a i gettimeofday) záleží na tom, jak je nastavený časovač a jakou má prioritu váš proces. Mikrosekunda je z pohledu systému už docela dlouho, takže při správném nastavení by to mělo být v pohodě.
    In Ada the typical infinite loop would normally be terminated by detonation.
    16.7.2010 13:45 Tom K | skóre: 21
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Ten cyklus ma spravny v pripade, ze nebude mit cekani delsi nez 1s (v pripade, ze mu to nekdo preplanuje a ono to pretece, tak uz na tom vlastne nezalezi a casem se mu to vzbudi stejne).

    Ta pasaz s casovacem mne velmi zaujala a poprosil bych o rozvedeni. Mikrosekunda je z puhledu systemu cca 1000-2000 instrukci CPU a to neni az tak mnoho.
    echo -n "u48" | sha1sum | head -c3; echo
    16.7.2010 16:53 pht | skóre: 48 | blog: pht
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Příloha:
    Ten cyklus ma spravny v pripade, ze nebude mit cekani delsi nez 1s (v pripade, ze mu to nekdo preplanuje a ono to pretece, tak uz na tom vlastne nezalezi a casem se mu to vzbudi stejne).

    Ne, zatuhne to. Dejme tomu: ts.tv_usec == 999999 a usecond == 3: bude se čekat až te.tv_usec >= 1000002, což nemůže nikdy nastat. Doporučuji alespoň následující, s tím, že nemůžete zadat usecond > 999999.
    void __usleep_test02(int usecond) {
            struct timeval tv;
            time_t target_sec;
            suseconds_t target_usec;
    
            gettimeofday(&tv, NULL);
            target_sec = tv.tv_sec;
            target_usec = tv.tv_usec + usecond;
            if (target_usec >= 1000000) {
                    target_sec++;
                    target_usec -= 1000000;
            }
    
            while ((tv.tv_usec < target_usec) || (tv.tv_sec < target_sec)) {
                    gettimeofday(&tv, NULL);
            }
    }
    
    Ta pasaz s casovacem mne velmi zaujala a poprosil bych o rozvedeni. Mikrosekunda je z puhledu systemu cca 1000-2000 instrukci CPU a to neni az tak mnoho.
    No, základem všeho je si zajistit RT prioritu procesoru (SCHED_FIFO), jinak se můžete jít koulet - jádro do Vaší čekačky přepne jiný proces a místo 2 us čekáte třeba 20 ms. Pokud máte RT prioritu, tak se to nestane, na druhou stranu jelikož standardní jádro není hard-RT, tak stejně nemáte 100% jistotu, ale už máte aspoň něco, s čím se dá rámcově počítat.

    Samotné čekání pak můžete udělat metodou busy wait - tj. tak jak to je naznačeno výše, čekáním v úzké smyčce na ten správný okamžik. Zablokujete tím kompletně jedno jádro procesoru, proto je potřeba vážit kdy a jak se tento postup nasadí. V defaultním nastavení linuxu je proti kompletnímu zablokování pojistka:
    $ cat /proc/sys/kernel/sched_rt_period_us
    1000000
    $ cat /proc/sys/kernel/sched_rt_runtime_us
    950000
    
    ... což znamená, že i přes RT prioritu bude váš proces donucen ke spánku pokud sežere více než 950000 us z 1000000 us okna (hodnoty v proc lze samozřejmě upravovat). To znamená že na jednoprocesorovém stroji budete mít šanci odladit tu chybně napsanou funkci na čekání :)

    Otázkou může být jak je jádro schopné říct kolik je přesně gettimeofday(). V linuxu jsou 2 časovače - clock source a event source. První lze jen číst a druhý lze i programovat na odeslání interruptu za nějakou dobu. Event source je obvykle podstatně méně přesný. Klasický časovač má 1000 Hz, HPETy mají v řádu MHz. Clock source má v optimálním případě přesnost 1 cyklu procesoru. Funkce gettimeofday() funguje tak, že se podívá na globální proměnnou kde je informace o čase posledního eventu a přičte k ní aktuální offset z clock source. Samozřejmě než se poté vyšaší přepnutí z kernel režimu do userspace atd. tak ještě nějaký čas uteče ale obvykle je to o dost méně než ta 1 us která nás zajímá. (Pro ultra přesné časování nesmíte volat funkce kernelu, ale vystačit si třeba s čtením TSC přímo v programu, což je zdroj dalšího opruzu...)

    Na druhou stranu pokud zavoláte nanosleep() tak si jádro dá určitou práci s tím, aby nastavilo event source jak nejlépe to jde (tam je toho dost, je potřeba váš časovač zařadit do RB stromu a kdovíco ještě) a pak si dá CPU voraz, takže skutečně tu mikrosekundu spíte. Bohužel kvůli různým overheadům a granularitě event sourcu budete mít trochu horší výsledky než u busy wait, na druhou stranu šetříte lesy :)

    Udělal jsem nějaké rychlé testy (viz přiložený program) a zdá se, že se výše popsané potvrzuje. Busy wait sežere 100% cpu a do 2ms okna se netrefí třeba 200x ze 100 000 pokusů. Na druhou stranu nanosleep nežere skoro nic, ale má overhead, se kterým musíte dopředu počítat a díky tomu trochu větší rozptyl.

    In Ada the typical infinite loop would normally be terminated by detonation.
    18.7.2010 14:56 Tom K | skóre: 21
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    Prilozeny program mi rika, ze v pripade nanosleep se netrefi do timeoutu 2us priblizne v 20% pripadu: PASS/FAIL: 80454/19546, kdezto aktivni cekani se netrefi v cca 6% pripadu. Ono bude dost zalezet na tom, kde to vlastne pobezi, na jake verzi jadra, jestli bude k dispozici HPET a podobne veci, takze na takove kratke useky je aktivni cekani vhodnejsi.

    Cist TSC bych radeji nedoporucoval, protoze se chova na ruznych architekturach jinak (napriklad na amd64 se jeho frekvence snizuje se zmenou frekvence CPU, kdezto na Intel x86_64 je jeho frekvence konstantni nezavisle na frekvenci CPU).
    echo -n "u48" | sha1sum | head -c3; echo
    18.7.2010 20:25 pht | skóre: 48 | blog: pht
    Rozbalit Rozbalit vše Re: C/C++ presnost usleep
    kdezto aktivni cekani se netrefi v cca 6% pripadu
    ... aspoň je vidět, že ani tahle metoda neni samospásná.
    na jake verzi jadra
    Pokud se pamatuju dobře, tak precizní časování se předělávalo někde kolem verze 2.6.17.
    na takove kratke useky je aktivni cekani vhodnejsi
    Staré glibc uměly při volání nanosleep automaticky aktivně čekat do 2 mikrosekund pokud měl program RT prioritu, z novějších verzí to vyhodili.
    Cist TSC bych radeji nedoporucoval, protoze se chova na ruznych architekturach jinak
    Není to věc architektury ale konkrétního modelu CPU. Já mám jeden celkem nový amd64 procesor a používá se dynamická změna rychlosti a TSC funguje dobře.
    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   www.czech-server.cz
    © 1999-2015 Nitemedia s. r. o. Všechna práva vyhrazena.