source: trunk/examples/network/torrent/ratecontroller.cpp

Last change on this file was 846, checked in by Dmitry A. Kuminov, 14 years ago

trunk: Merged in qt 4.7.2 sources from branches/vendor/nokia/qt.

File size: 5.5 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
4** All rights reserved.
5** Contact: Nokia Corporation (qt-info@nokia.com)
6**
7** This file is part of the examples of the Qt Toolkit.
8**
9** $QT_BEGIN_LICENSE:BSD$
10** You may use this file under the terms of the BSD license as follows:
11**
12** "Redistribution and use in source and binary forms, with or without
13** modification, are permitted provided that the following conditions are
14** met:
15** * Redistributions of source code must retain the above copyright
16** notice, this list of conditions and the following disclaimer.
17** * Redistributions in binary form must reproduce the above copyright
18** notice, this list of conditions and the following disclaimer in
19** the documentation and/or other materials provided with the
20** distribution.
21** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor
22** the names of its contributors may be used to endorse or promote
23** products derived from this software without specific prior written
24** permission.
25**
26** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
27** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
28** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
29** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
30** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
31** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
32** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
33** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
34** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
35** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
36** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
37** $QT_END_LICENSE$
38**
39****************************************************************************/
40
41#include "peerwireclient.h"
42#include "ratecontroller.h"
43
44#include <QtCore>
45
46Q_GLOBAL_STATIC(RateController, rateController)
47
48RateController *RateController::instance()
49{
50 return rateController();
51}
52
53void RateController::addSocket(PeerWireClient *socket)
54{
55 connect(socket, SIGNAL(readyToTransfer()), this, SLOT(scheduleTransfer()));
56 socket->setReadBufferSize(downLimit * 4);
57 sockets << socket;
58 scheduleTransfer();
59}
60
61void RateController::removeSocket(PeerWireClient *socket)
62{
63 disconnect(socket, SIGNAL(readyToTransfer()), this, SLOT(scheduleTransfer()));
64 socket->setReadBufferSize(0);
65 sockets.remove(socket);
66}
67
68void RateController::setDownloadLimit(int bytesPerSecond)
69{
70 downLimit = bytesPerSecond;
71 foreach (PeerWireClient *socket, sockets)
72 socket->setReadBufferSize(downLimit * 4);
73}
74
75void RateController::scheduleTransfer()
76{
77 if (transferScheduled)
78 return;
79 transferScheduled = true;
80 QTimer::singleShot(50, this, SLOT(transfer()));
81}
82
83void RateController::transfer()
84{
85 transferScheduled = false;
86 if (sockets.isEmpty())
87 return;
88
89 int msecs = 1000;
90 if (!stopWatch.isNull())
91 msecs = qMin(msecs, stopWatch.elapsed());
92
93 qint64 bytesToWrite = (upLimit * msecs) / 1000;
94 qint64 bytesToRead = (downLimit * msecs) / 1000;
95 if (bytesToWrite == 0 && bytesToRead == 0) {
96 scheduleTransfer();
97 return;
98 }
99
100 QSet<PeerWireClient *> pendingSockets;
101 foreach (PeerWireClient *client, sockets) {
102 if (client->canTransferMore())
103 pendingSockets << client;
104 }
105 if (pendingSockets.isEmpty())
106 return;
107
108 stopWatch.start();
109
110 bool canTransferMore;
111 do {
112 canTransferMore = false;
113 qint64 writeChunk = qMax<qint64>(1, bytesToWrite / pendingSockets.size());
114 qint64 readChunk = qMax<qint64>(1, bytesToRead / pendingSockets.size());
115
116 QSetIterator<PeerWireClient *> it(pendingSockets);
117 while (it.hasNext() && (bytesToWrite > 0 || bytesToRead > 0)) {
118 PeerWireClient *socket = it.next();
119 if (socket->state() != QAbstractSocket::ConnectedState) {
120 pendingSockets.remove(socket);
121 continue;
122 }
123
124 bool dataTransferred = false;
125 qint64 available = qMin<qint64>(socket->socketBytesAvailable(), readChunk);
126 if (available > 0) {
127 qint64 readBytes = socket->readFromSocket(qMin<qint64>(available, bytesToRead));
128 if (readBytes > 0) {
129 bytesToRead -= readBytes;
130 dataTransferred = true;
131 }
132 }
133
134 if (upLimit * 2 > socket->bytesToWrite()) {
135 qint64 chunkSize = qMin<qint64>(writeChunk, bytesToWrite);
136 qint64 toWrite = qMin(upLimit * 2 - socket->bytesToWrite(), chunkSize);
137 if (toWrite > 0) {
138 qint64 writtenBytes = socket->writeToSocket(toWrite);
139 if (writtenBytes > 0) {
140 bytesToWrite -= writtenBytes;
141 dataTransferred = true;
142 }
143 }
144 }
145
146 if (dataTransferred && socket->canTransferMore())
147 canTransferMore = true;
148 else
149 pendingSockets.remove(socket);
150 }
151 } while (canTransferMore && (bytesToWrite > 0 || bytesToRead > 0) && !pendingSockets.isEmpty());
152
153 if (canTransferMore || bytesToWrite == 0 || bytesToRead == 0)
154 scheduleTransfer();
155}
Note: See TracBrowser for help on using the repository browser.