stone2083

          SCP限速邏輯實現--帶中文注釋的bandwidth_limit函數

          SCP限速邏輯實現--帶中文注釋的bandwidth_limit函數

          SCP支持限速, 通過-l參數, 指定拷貝的速度
           -l limit
                       Limits the used bandwidth, specified in Kbit/s.

          具體實現, 是在misc.c中的bandwidth_limit函數
          下面附上帶中文注釋的bandwidth_limit函數代碼
           1 struct bwlimit {
           2     size_t buflen;                 // 每次read的buf長度
           3     u_int64_t rate, thresh, lamt;  // rate:  限速速率, 單位kpbs
           4                                    // thresh:統計周期,read長度到了指定閾值, 觸發限速統計
           5                                    // lamt:  一次統計周期內, read了多少長度
           6     struct timeval bwstart, bwend; // bwstart: 統計周期之開始時間
           7                                    // bwend:   統計周期之結束時間
           8 };
           9 
          10 void bandwidth_limit_init(struct bwlimit *bw, u_int64_t kbps, size_t buflen)
          11 {
          12     bw->buflen = buflen;          // 初始化read buf長度
          13     bw->rate = kbps;              // 初始化限速速率
          14     bw->thresh = bw->rate;        // 初始化統計周期
          15     bw->lamt = 0;                 // 初始化當前read長度
          16     timerclear(&bw->bwstart);     // 初始化統計開始時間
          17     timerclear(&bw->bwend);       // 初始化統計結束時間
          18 }
          19 
          20 void bandwidth_limit(struct bwlimit *bw, size_t read_len)
          21 {
          22     u_int64_t waitlen;
          23     struct timespec ts, rm;
          24 
          25     // 設置統計開始時間, 為當前時間
          26     if (!timerisset(&bw->bwstart)) {
          27         gettimeofday(&bw->bwstart, NULL);
          28         return;
          29     }
          30 
          31     // 設置當前read長度
          32     bw->lamt += read_len;
          33     // 判斷當前read長度是否到達統計周期的閾值
          34     if (bw->lamt < bw->thresh)
          35         return;
          36 
          37     // 設置統計結束時間,為當前時間
          38     gettimeofday(&bw->bwend, NULL);
          39     // bwend變量復用, 這個時候, bwend含義為, 本次統計周期實際開銷的時間: 既read thresh長度字節,花了多少時間.
          40     timersub(&bw->bwend, &bw->bwstart, &bw->bwend);
          41     if (!timerisset(&bw->bwend))
          42         return;
          43 
          44     // 將單位從Byte變成bit
          45     bw->lamt *= 8;
          46     // 根據限速速率, 計算理論應該花費多少時間
          47     waitlen = (double)1000000L * bw->lamt / bw->rate;
          48     // bwstart變量復用, 這個時候, bwstart含義為, 本次統計周期理論開銷的時間
          49     bw->bwstart.tv_sec = waitlen / 1000000L;
          50     bw->bwstart.tv_usec = waitlen % 1000000L;
          51 
          52     // 如果理論開銷時間 >  實際開銷時間, 則需要做限速
          53     if (timercmp(&bw->bwstart, &bw->bwend, >)) {
          54         // bwend變量復用, 這個時間, bwend含義為, 理論開銷時間 和 實際開銷時間的差值, 既需要sleep的時間, 確保達到限速到指定的rate值
          55         timersub(&bw->bwstart, &bw->bwend, &bw->bwend);
          56 
          57         // 如果差值達到了秒級, 則需要降低統計周期閾值, 確保統計相對精確
          58         // thresh變為原先的1/2, 但不能低于buflen的1/4
          59         if (bw->bwend.tv_sec) {
          60             bw->thresh /= 2;
          61             if (bw->thresh < bw->buflen / 4)
          62                 bw->thresh = bw->buflen / 4;
          63         } 
          64         // 如果差值小于10毫秒, 則需要加大統計周期閾值, 確保統計相對精確
          65         // thresh變為原先的2倍, 但不能高于buflen的8倍
          66         else if (bw->bwend.tv_usec < 10000) {
          67             bw->thresh *= 2;
          68             if (bw->thresh > bw->buflen * 8)
          69                 bw->thresh = bw->buflen * 8;
          70         }
          71 
          72         // 乖乖的睡一會吧, 以達到限速目的
          73         TIMEVAL_TO_TIMESPEC(&bw->bwend, &ts);
          74         while (nanosleep(&ts, &rm) == -1) {
          75             if (errno != EINTR)
          76                 break;
          77             ts = rm;
          78         }
          79     }
          80 
          81     // 新的統計周期開始, 初始化lamt, bwstart變量
          82     bw->lamt = 0;
          83     gettimeofday(&bw->bwstart, NULL);
          84 }

          posted on 2014-09-22 17:45 stone2083 閱讀(2488) 評論(4)  編輯  收藏 所屬分類: c

          Feedback

          # re: SCP限速邏輯實現--帶中文注釋的bandwidth_limit函數 2014-09-23 15:53 提升機

          學習了,多謝分享  回復  更多評論   

          # re: SCP限速邏輯實現--帶中文注釋的bandwidth_limit函數 2014-09-23 20:40 名皮網

          收藏了。多謝分享  回復  更多評論   

          # re: SCP限速邏輯實現--帶中文注釋的bandwidth_limit函數 2014-09-30 09:36 sjxifen

          怎樣用的呢?  回復  更多評論   


          只有注冊用戶登錄后才能發表評論。


          網站導航:
           
          主站蜘蛛池模板: 龙陵县| 古蔺县| 曲靖市| 芜湖县| 黄冈市| 岳西县| 秦皇岛市| 玉树县| 安庆市| 青铜峡市| 崇信县| 肇庆市| 大兴区| 攀枝花市| 二手房| 郧西县| 巴林右旗| 体育| 乌拉特前旗| 洮南市| 安多县| 霍州市| 福州市| 伊金霍洛旗| 北京市| 巴彦淖尔市| 青田县| 方正县| 县级市| 佛冈县| 泰顺县| 奇台县| 将乐县| 平南县| 赞皇县| 连城县| 全椒县| 株洲市| 大理市| 榆林市| 元江|