激情五月天婷婷,亚洲愉拍一区二区三区,日韩视频一区,a√天堂中文官网8

<ul id="buwfs"><strike id="buwfs"><strong id="buwfs"></strong></strike></ul>
    <output id="buwfs"></output>
  • <dfn id="buwfs"><source id="buwfs"></source></dfn>
      <dfn id="buwfs"><td id="buwfs"></td></dfn>
      <div id="buwfs"><small id="buwfs"></small></div>
      <dfn id="buwfs"><source id="buwfs"></source></dfn>
      1. <dfn id="buwfs"><td id="buwfs"></td></dfn>
        始創(chuàng)于2000年 股票代碼:831685
        咨詢熱線:0371-60135900 注冊有禮 登錄
        • 掛牌上市企業(yè)
        • 60秒人工響應
        • 99.99%連通率
        • 7*24h人工
        • 故障100倍補償
        全部產品
        您的位置: 網站首頁 > 幫助中心>文章內容

        深入研究Linux高精確時序函數

        發(fā)布時間:  2012/8/10 15:09:44
         首先, 我會說不保證你在使用者模式 (user-mode) 中執(zhí)行的行程 (process) 能夠精確地控制時序因為 Linux 是個多工的作業(yè)環(huán)境. 你在執(zhí)行中的行程 (process) 隨時會因為各種原因被暫停大約 10 毫秒到數秒 (在系統(tǒng)負荷非常高的時候). 然而, 對於大多數使用 I/O 埠的應用而言, 這個延遲時間實際上算不了什麼. 要縮短延遲時間, 你得使用函式 nice 將你在執(zhí)行中的行程 (process ) 設定成高優(yōu)先權(請參考 nice(2) 使用說明文件) 或使用即時排程法 (real-time scheduling) (請看下面).

          如果你想獲得比在一般使用者模式 (user-mode) 中執(zhí)行的行程 (process) 還要精確的時序, 有一些方法可以讓你在使用者模式 (user-mode) 中做到 `即時' 排程的支援. Linux 2.x 版本的核心中有軟體方式的即時排程支援; 詳細的說明請參考 sched_setscheduler(2) 使用說明文件. 有一個特殊的核心支援硬體的即時排程;

          (Sleeping) : sleep() 與 usleep()

          現在, 讓我們開始較簡單的時序函式呼叫. 想要延遲數秒的時間, 最佳的方法大概 是使用函式 sleep() . 想要延遲至少數十毫秒的時間 (10 ms 似乎已是最短的 延遲時間了), 函式 usleep() 應該可以使用. 這些函式是讓出 CPU 的使用權 給其他想要執(zhí)行的行程 (processes) (``自己休息去了''), 所以沒有浪費掉 CPU 的時間. 細節(jié)請參考 sleep(3) 與 usleep(3) 的說明文件.

          如果讓出 CPU 的使用權因而使得時間延遲了大約 50 毫秒 (這取決於處理器與機器的速度, 以及系統(tǒng)的負荷), 就浪費掉 CPU 太多的時間, 因為 Linux 的排程器 (scheduler) (單就 x86 架構而言) 在將控制權發(fā)還給你的行程 (process) 之前通常至少要花費 10-30 毫秒的時間. 因此, 短時間的延遲, 使用函式 usleep(3) 所得到的延遲結果通常會大於你在參數所指定的值, 大約至少有 10 ms.

          nanosleep()

          在 Linux 2.0.x 一系列的核心發(fā)行版本中, 有一個新的系統(tǒng)呼叫 (system call), nanosleep() (請參考 nanosleep(2) 的說明文件), 他讓你能夠 休息或延遲一個短的時間 (數微秒或更多).

          如果延遲的時間 <= 2 ms, 若(且唯若)你執(zhí)行中的行程 (process) 設定了軟體的即時 排程 (就是使用函式 tt/sched_setscheduler()/), 呼叫函式 nanosleep() 時 不是使用一個忙碌回圈來延遲時間; 就是會像函式 usleep() 一樣讓出 CPU 的使用權休息去了.

          這個忙碌回圈使用函式 udelay() (一個驅動程式常會用到的核心內部的函式) 來達成, 并且使用 BogoMips 值 (BogoMips 可以準確量測這類忙碌回圈的速度) 來計算回圈延遲的時間長度. 其如何動作的細節(jié)請參考 /usr/include/asm/delay.h).

          使用 I/O 埠來延遲時間

          另一個延遲數微秒的方法是使用 I/O 埠. 就是從埠位址 0x80 輸入或輸出任何 byte 的資料 (請參考前面) 等待的時間應該幾乎只要 1 微秒這要看你的處理器的型別與速度. 如果要延遲數微秒的時間你可以將這個動作多做幾次. 在任何標準的機器上輸出資料到該 埠位址應該不會有不良的後果□對 (而且有些核心的設備驅動程式也在使用他). {in|out}[bw]_p() 等函式就是使用這個方法來產生時間延遲的 (請參考檔案 asm/io.h).

          實際上, 一個使用到埠位址□圍為 0-0x3ff 的 I/O 埠指令幾乎只要 1 微秒的時間, 所以如果你要如此做, 例如, 直接使用并列埠, 只要加上幾個 inb() 函式從該 埠位址□圍讀入 byte 的資料即可.

          使用組合語言來延遲時間

          如果你知道執(zhí)行程式所在機器的處理器型別與時鐘速度, 你可以執(zhí)行某些組合語言指令以便獲得較短的延遲時間 (但是記住, 你在執(zhí)行中的行程 (process) 隨時會被暫停, 所以有時延遲的時間會比實際長). 如下面的表格所示, 內部處理器的速度決定了所要使用的時鐘周期數; 如, 一個 50 MHz 的處理器 (486DX-50 或 486DX2-50), 一個時鐘周期要花費 1/50000000 秒 (=200 奈秒).

          指令 i386 時鐘周期數 i486 時鐘周期數nop 3 1xchg %ax,%ax 3 3or %ax,%ax 2 1mov %ax,%ax 2 1add %ax,0 2 1

          (對不起, 我不知道 Pentiums 的資料, 或許與 i486 接近吧. 我無法在 i386 的資料上找到只花費一個時鐘周期的指令. 如果能夠就請使用花費一個時鐘周期的指令, 要不然就使用管線技術的新式處理器也是可以縮短時間的.)

          上面的表格中指令 nop 與 xchg 應該不會有不良的後果. 指令最後可能會 改變旗號暫存器的內容, 但是這沒關系因為 gcc 會處理. 指令 nop 是個好的選擇.

          想要在你的程式中使用到這些指令, 你得使用 asm("instruction"). 指令的語法就如同上面表格的用法; 如果你想要在單一的 asm() 敘述中使用多個指令, 可以使用分號將他們隔開. 例如, asm("nop ; nop ; nop ; nop") 會執(zhí)行四個 nop 指令, 在 i486 或 Pentium 處理器中會延遲四個時鐘周期 (或是 i386 會延遲 12 個時鐘周期).

          gcc 會將 asm() 翻譯成單行組合語言程式碼, 所以不會有呼叫函式的負荷.

          在 Intel x86 架構中不可能有比一個時鐘周期還短的時間延遲.

          在 Pentiums 處理器上使用函式 rdtsc

          對於 Pentiums 處理器而言, 你可以使用下面的 C 語言程式碼來取得自從上次重新開機 到現在經過了多少個時鐘周期:

          --------------------------------------------------------------------------------

          extern __inline__ unsigned long long int rdtsc() { unsigned long long int x; __asm__ volatile (".byte 0x0f, 0x31" : "=A" (x)); return x; }

          --------------------------------------------------------------------------------

          你可以詢問參考此值以便延遲你想要的時鐘周期數.

          想要時間精確到一秒鐘, 使用函式 time() 或許是最簡單的方法. 想要時間更精確, 函式 gettimeofday() 大約可以精確到微秒 (但是如前所述會受到 CPU 排程的影響). 至於 Pentiums 處理器, 使用上面的程式碼片斷就可以精確到一個時鐘周期.

          如果你要你執(zhí)行中的行程 (process) 在一段時間到了之後能夠被通知 (get a signal), 你得使用函式 setitimer() 或 alarm() . 細節(jié)請參考函式的使用說明文件


        本文出自:億恩科技【mszdt.com】

        服務器租用/服務器托管中國五強!虛擬主機域名注冊頂級提供商!15年品質保障!--億恩科技[ENKJ.COM]

      2. 您可能在找
      3. 億恩北京公司:
      4. 經營性ICP/ISP證:京B2-20150015
      5. 億恩鄭州公司:
      6. 經營性ICP/ISP/IDC證:豫B1.B2-20060070
      7. 億恩南昌公司:
      8. 經營性ICP/ISP證:贛B2-20080012
      9. 服務器/云主機 24小時售后服務電話:0371-60135900
      10. 虛擬主機/智能建站 24小時售后服務電話:0371-60135900
      11. 專注服務器托管17年
        掃掃關注-微信公眾號
        0371-60135900
        Copyright© 1999-2019 ENKJ All Rights Reserved 億恩科技 版權所有  地址:鄭州市高新區(qū)翠竹街1號總部企業(yè)基地億恩大廈  法律顧問:河南亞太人律師事務所郝建鋒、杜慧月律師   京公網安備41019702002023號
          1
         
         
         
         

        0371-60135900
        7*24小時客服服務熱線