实现数据包超时c ++

时间:2011-04-20 01:52:20

标签: c++ timeout packets

对于计算机网络项目,我们使用不同的窗口协议在c ++中编写ftp服务器。我们在实现工作超时功能时遇到问题。它的要点是设置发送数据包的时间戳,如果“ack”在一定时间内(例如2.5毫秒)没有恢复,则重新发送数据包。我们目前使用clock_gettime()函数来获取数据包的当前时间和时间戳。

以下是我们遇到问题的while循环:

while(packet_sync_array[prev].recieved ==0 && t==0)
{
    clock_gettime(CLOCK_PROCESS_CPUTIME_ID,&timeout2);
    currtime = timeout2.tv_nsec;
    cout << "Is currtime: "<<currtime<< " - ptimeout " << ptimeout 
         << " = " << (currtime - ptimeout) << " > " 
         << (connection_parameters->timeoutInterval)*1000 << "?" << endl;

    if((currtime - ptimeout)>((connection_parameters->timeoutInterval)*1000))
    {
        t = 1;
        cout << "Prev PACKET TIMEDOUT" << endl;
        cout << "Is currtime: "<<currtime<< " - ptimeout " << ptimeout 
             << " = " << (currtime - ptimeout) << " > " 
             << (connection_parameters->timeoutInterval)*1000 << "?" << endl;
    }
}  

其中ptimeout是发送数据包的时间,connection_parameters->timeoutInterval是超时间隔。问题是因为ptimeout是时间的长整数表示,有时它是一个非常大的值(例如999715992)。这意味着它永远无法判断超时是否已经发生,因为当值变得足够大时,以纳秒为单位的当前时间将休息为0。

有没有其他人在c ++中处理这些时序问题并有可能的解决方案?

谢谢!

编辑:

感谢您的快速回复!我能够搞清楚一些事情。修改while循环以检查并查看超时+时间戳是否大于允许的长整数大小让我看看在比较之前clock_gettime是否会回调为零。知道这一点,我检查了当前时间&gt; (超时间隔 - (最大长整数 - 时间戳))。这允许最多1秒的超时,这对于该问题的目的应该是足够的。如果有人认为他们有更好的解决方案,请告诉我!谢谢! 以下是感兴趣的代码:

if(((999999998-ptimeout)< (connection_parameters->timeoutInterval)*1000)&&(currtime - ptimeout)<0){
          if(currtime > (((connection_parameters->timeoutInterval)*1000)-(999999998-ptimeout))){
              t = 1;
              cout << "this wrapped since " << currtime << " + " << (connection_parameters->timeoutInterval)*1000 << "is bigger than 999999999 and then timed out" << endl;
              cout << "curr time " << currtime << "is bigger than" << endl;
              cout << (connection_parameters->timeoutInterval)*1000 << endl;
              cout << "-" << endl;
              cout << (999999998-ptimeout) << endl;
              cout << "---------------------------" << endl;
              cout << (((connection_parameters->timeoutInterval)*1000)-(999999998-ptimeout)) << endl;
              cout << "Prev PACKET TIMEDOUT" << endl;
              cout << "Is currtime: "<<currtime<< " - ptimeout " << ptimeout << " = " << (currtime - ptimeout) << " > " << (connection_parameters->timeoutInterval)*1000 << "?" << endl;
          }
      }
      else if((currtime - ptimeout)>((connection_parameters->timeoutInterval)*1000)){
       t = 1;
       cout << "Prev PACKET TIMEDOUT" << endl;
       cout << "Is currtime: "<<currtime<< " - ptimeout " << ptimeout << " = " << (currtime - ptimeout) << " > " << (connection_parameters->timeoutInterval)*1000 << "?" << endl;
      }

2 个答案:

答案 0 :(得分:3)

您可能希望将代码更改为:

// time in microseconds
currtime = timeout2.tv_sec * 1000000 + timeout2.tv_nsec / 1000;

确保没有任何整数溢出!例如,明确地转换为64位整数可能是个好主意。

此外,您最有可能希望超时至少大于2.5毫秒的一个数量级,可能更多。例如,操作系统中的时间片很可能大约为17毫秒,更不用说网络延迟了。

答案 1 :(得分:1)

您需要同时查看tv_sec填充的tv_nsec对象的timespecclock_gettime组件。您的ptimeout字段应为{ {1}}而不是整数。你必须编写函数来对timespec个对象进行数学运算,我认为没有任何库存。