00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037 #include <stdio.h>
00038 #include "ip.h"
00039 #include "dsred.h"
00040 #include "random.h"
00041 #include "dsredq.h"
00042
00043
00044
00045
00046 redQueue::redQueue() {
00047 numPrec = MAX_PREC;
00048 mredMode = rio_c;
00049
00050 q_ = new PacketQueue();
00051 }
00052
00053
00054
00055
00056
00057 void redQueue::config(int prec, int argc, const char*const* argv) {
00058 qParam_[prec].qlen = 0;
00059 qParam_[prec].edp_.th_min = atoi(argv[4]);
00060 qParam_[prec].edp_.th_max = atoi(argv[5]);
00061 qParam_[prec].edp_.max_p_inv = 1.0 / atof(argv[6]);
00062
00063
00064
00065 if (argc > 7)
00066 qParam_[prec].edp_.q_w = atof(argv[7]);
00067 else
00068 qParam_[prec].edp_.q_w = 0.002;
00069
00070 qParam_[prec].edv_.v_ave = 0.0;
00071 qParam_[prec].idle_ = 1;
00072 if (&Scheduler::instance() != NULL)
00073 qParam_[prec].idletime_ = Scheduler::instance().clock();
00074 else
00075 qParam_[prec].idletime_ = 0.0;
00076 }
00077
00078
00079
00080 void redQueue::initREDStateVar(void) {
00081 for (int i = 0; i < numPrec; i++) {
00082 qParam_[i].idle_ = 1;
00083
00084 if (&Scheduler::instance() != NULL)
00085 qParam_[i].idletime_ = Scheduler::instance().clock();
00086 else
00087 qParam_[i].idletime_ = 0.0;
00088 }
00089 }
00090
00091
00092
00093 void redQueue::updateVREDLen(int prec) {
00094
00095 qParam_[prec].qlen--;
00096 }
00097
00098
00099
00100
00101 void redQueue::updateIdleFlag(int prec) {
00102 if (mredMode == rio_c) {
00103 for (int i = prec; i < numPrec; i++)
00104 qParam_[i].idle_ = 0;
00105 } else if (mredMode == rio_d) {
00106 qParam_[prec].idle_ = 0;
00107 } else {
00108 qParam_[0].idle_ = 0;
00109 }
00110 }
00111
00112
00113
00114 void redQueue::updateREDStateVar(int prec) {
00115 int idle = 1;
00116 int i;
00117
00118 double now = Scheduler::instance().clock();
00119
00120
00121
00122 if (qParam_[prec].qlen == 0) {
00123 if (mredMode == rio_c) {
00124 for(i=0; i<prec; i++)
00125 if(qParam_[i].qlen != 0) idle = 0;
00126 if (idle) {
00127 for (i=prec;i<numPrec;i++) {
00128 if (qParam_[i].qlen == 0) {
00129 qParam_[i].idle_ = 1;
00130 qParam_[i].idletime_ = now;
00131 } else break;
00132 }
00133 }
00134 } else if (mredMode == rio_d) {
00135 qParam_[prec].idle_ = 1;
00136 qParam_[prec].idletime_ = now;
00137 } else if (mredMode == wred) {
00138 qParam_[0].idle_ = 1;
00139 qParam_[0].idletime_ = now;
00140 }
00141 }
00142 }
00143
00144
00145
00146
00147
00148
00149
00150 int redQueue::enque(Packet *pkt, int prec, int ecn) {
00151 int m = 0;
00152 double now, u;
00153 double pa,pb;
00154
00155 if (q_->length() > (qlim-1))
00156 return PKT_DROPPED;
00157
00158 now = Scheduler::instance().clock();
00159
00160
00161
00162
00163 if (mredMode == dropTail) {
00164 if (qParam_[prec].qlen >= qParam_[prec].edp_.th_min) {
00165 return PKT_DROPPED;
00166 }
00167 } else if (mredMode == rio_c) {
00168
00169
00170 for (int i = prec; i < numPrec; i++) {
00171 m = 0;
00172 if (qParam_[i].idle_) {
00173
00174 m = int(qParam_[i].edp_.ptc * (now - qParam_[i].idletime_));
00175 }
00176 calcAvg(i, m+1);
00177 }
00178 } else if (mredMode == rio_d) {
00179 if (qParam_[prec].idle_) {
00180
00181 m = int(qParam_[prec].edp_.ptc * (now - qParam_[prec].idletime_));
00182 }
00183 calcAvg(prec, m+1);
00184 } else {
00185 if (qParam_[0].idle_) {
00186
00187 m = int(qParam_[0].edp_.ptc * (now - qParam_[0].idletime_));
00188 }
00189 calcAvg(0, m+1);
00190 }
00191
00192
00193 if (ecn) {
00194 q_->enque(pkt);
00195
00196
00197
00198 qParam_[prec].qlen++;
00199 }
00200
00201
00202
00203 if (qParam_[prec].edv_.v_ave > qParam_[prec].edp_.th_min) {
00204
00205 if (qParam_[prec].edv_.v_ave <= qParam_[prec].edp_.th_max) {
00206
00207
00208 qParam_[prec].edv_.count++;
00209 qParam_[prec].edv_.v_prob = (1/qParam_[prec].edp_.max_p_inv) *
00210 (qParam_[prec].edv_.v_ave-qParam_[prec].edp_.th_min) /
00211 (qParam_[prec].edp_.th_max-qParam_[prec].edp_.th_min);
00212
00213 pb = qParam_[prec].edv_.v_prob;
00214 pa = pb/(1.0 - qParam_[prec].edv_.count*pb);
00215
00216
00217 u = Random::uniform(0.0, 1.0);
00218
00219
00220 if (u <= pa) {
00221
00222
00223
00224 qParam_[prec].edv_.count = 0;
00225 if (ecn) {
00226
00227 updateIdleFlag(prec);
00228 return PKT_MARKED;
00229 }
00230 return PKT_EDROPPED;
00231 }
00232 } else {
00233 qParam_[prec].edv_.count = 0;
00234 if (ecn) {
00235
00236 updateIdleFlag(prec);
00237 return PKT_MARKED;
00238 }
00239 return PKT_DROPPED;
00240 }
00241 } else {
00242
00243
00244
00245
00246 qParam_[prec].edv_.count = -1;
00247 }
00248
00249
00250 if(ecn) {
00251
00252 updateIdleFlag(prec);
00253 return PKT_ENQUEUED;
00254 }
00255
00256
00257
00258 q_->enque(pkt);
00259
00260
00261
00262 qParam_[prec].qlen++;
00263
00264
00265 updateIdleFlag(prec);
00266
00267 return PKT_ENQUEUED;
00268 }
00269
00270
00271
00272 Packet* redQueue::deque() {
00273 return(q_->deque());
00274 }
00275
00276
00277
00278
00279
00280
00281
00282 void redQueue::calcAvg(int prec, int m) {
00283 float f;
00284 int i;
00285
00286 f = qParam_[prec].edv_.v_ave;
00287
00288 while (--m >= 1) {
00289 f *= 1.0 - qParam_[prec].edp_.q_w;
00290 }
00291 f *= 1.0 - qParam_[prec].edp_.q_w;
00292
00293 if (mredMode == rio_c)
00294 for (i = 0; i <= prec; i ++)
00295 f += qParam_[i].edp_.q_w * qParam_[i].qlen;
00296 else if (mredMode == rio_d)
00297 f += qParam_[prec].edp_.q_w * qParam_[prec].qlen;
00298 else
00299 f += qParam_[prec].edp_.q_w * q_->length();
00300
00301 if (mredMode == wred)
00302 for (i = 0; i < numPrec; i ++)
00303 qParam_[i].edv_.v_ave = f;
00304 else
00305 qParam_[prec].edv_.v_ave = f;
00306 }
00307
00308
00309
00310
00311 double redQueue::getWeightedLength() {
00312 double sum = 0.0;
00313
00314 if (mredMode == rio_c)
00315 return qParam_[numPrec-1].edv_.v_ave;
00316 else {
00317 for (int prec = 0; prec < numPrec; prec++)
00318 sum += qParam_[prec].edv_.v_ave;
00319 return(sum);
00320 }
00321 }
00322
00323
00324
00325 int redQueue::getRealLength(void) {
00326 return(q_->length());
00327 }
00328
00329
00330
00331 double redQueue::getWeightedLength_v(int prec) {
00332 return qParam_[prec].edv_.v_ave;
00333 }
00334
00335
00336
00337 int redQueue::getRealLength_v(int prec) {
00338 return qParam_[prec].qlen;
00339 }
00340
00341
00342
00343
00344 void redQueue::setPTC(double outLinkBW) {
00345 for (int i = 0; i < numPrec; i++)
00346 qParam_[i].edp_.ptc = outLinkBW/(8.0*qParam_[i].edp_.mean_pktsize);
00347 }
00348
00349
00350
00351 void redQueue::setMPS(int mps) {
00352 for (int i = 0; i < numPrec; i++)
00353 qParam_[i].edp_.mean_pktsize = mps;
00354 }