Blame src/Speedometer.cc

Packit 8f70b4
/*
Packit 8f70b4
 * lftp - file transfer program
Packit 8f70b4
 *
Packit 8f70b4
 * Copyright (c) 1996-2012 by Alexander V. Lukyanov (lav@yars.free.net)
Packit 8f70b4
 *
Packit 8f70b4
 * This program is free software; you can redistribute it and/or modify
Packit 8f70b4
 * it under the terms of the GNU General Public License as published by
Packit 8f70b4
 * the Free Software Foundation; either version 3 of the License, or
Packit 8f70b4
 * (at your option) any later version.
Packit 8f70b4
 *
Packit 8f70b4
 * This program is distributed in the hope that it will be useful,
Packit 8f70b4
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f70b4
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 8f70b4
 * GNU General Public License for more details.
Packit 8f70b4
 *
Packit 8f70b4
 * You should have received a copy of the GNU General Public License
Packit 8f70b4
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 8f70b4
 */
Packit 8f70b4
Packit 8f70b4
#include <config.h>
Packit 8f70b4
#include <math.h>
Packit 8f70b4
#include <stdlib.h>
Packit 8f70b4
#include "Speedometer.h"
Packit 8f70b4
#include "misc.h"
Packit 8f70b4
#include "xstring.h"
Packit 8f70b4
#include "ResMgr.h"
Packit 8f70b4
Packit 8f70b4
#define now SMTask::now
Packit 8f70b4
Packit 8f70b4
Speedometer::Speedometer(const char *p)
Packit 8f70b4
{
Packit 8f70b4
   period=15;
Packit 8f70b4
   rate=0;
Packit 8f70b4
   last_second=now;
Packit 8f70b4
   start=now;
Packit 8f70b4
   last_bytes=0;
Packit 8f70b4
   terse=true;
Packit 8f70b4
   period_resource=p;
Packit 8f70b4
   Reconfig(0);
Packit 8f70b4
}
Packit 8f70b4
bool Speedometer::Valid()
Packit 8f70b4
{
Packit 8f70b4
   return now>=start+TimeDiff(1,0) && now
Packit 8f70b4
}
Packit 8f70b4
float Speedometer::Get()
Packit 8f70b4
{
Packit 8f70b4
   Add(0);
Packit 8f70b4
   SMTask::current->Timeout(500);
Packit 8f70b4
   return rate;
Packit 8f70b4
}
Packit 8f70b4
void Speedometer::Add(int b)
Packit 8f70b4
{
Packit 8f70b4
   if(b==0 && (now==last_second || TimeDiff(now,last_second).MilliSeconds()<100))
Packit 8f70b4
      return;
Packit 8f70b4
Packit 8f70b4
   // This makes Speedometer start only when first data come.
Packit 8f70b4
   if(rate==0)
Packit 8f70b4
      Reset();
Packit 8f70b4
Packit 8f70b4
   double div=period;
Packit 8f70b4
Packit 8f70b4
   if(start>now)
Packit 8f70b4
      start=now;  // time was adjusted?
Packit 8f70b4
   if(now
Packit 8f70b4
      last_second=now;
Packit 8f70b4
Packit 8f70b4
   double time_passed_since_start=TimeDiff(now,start);
Packit 8f70b4
   double time_passed=TimeDiff(now,last_second);
Packit 8f70b4
   if(time_passed_since_start
Packit 8f70b4
      div=time_passed_since_start;
Packit 8f70b4
   if(div<1)
Packit 8f70b4
      div=1;
Packit 8f70b4
Packit 8f70b4
   rate*=1-time_passed/div;
Packit 8f70b4
   rate+=b/div;
Packit 8f70b4
Packit 8f70b4
   last_second=now;
Packit 8f70b4
   if(b>0)
Packit 8f70b4
      last_bytes=now;
Packit 8f70b4
   if(rate<0)
Packit 8f70b4
      rate=0;
Packit 8f70b4
}
Packit 8f70b4
// These `incorrect' units are used to fit on status line
Packit 8f70b4
xstring& Speedometer::GetStr(float r)
Packit 8f70b4
{
Packit 8f70b4
   if(r<1)
Packit 8f70b4
      return xstring::get_tmp("");
Packit 8f70b4
   if(r<1024)
Packit 8f70b4
      // for translator: those are the units. This is 'byte per second'
Packit 8f70b4
      return xstring::format(_("%.0fb/s"),r);
Packit 8f70b4
   else if(r<1024*1024)
Packit 8f70b4
      // for translator: This is 'Kibibyte per second'
Packit 8f70b4
      return xstring::format(_("%.1fK/s"),r/1024.);
Packit 8f70b4
   else
Packit 8f70b4
      // for translator: This is 'Mebibyte per second'
Packit 8f70b4
      return xstring::format(_("%.2fM/s"),r/(1024.*1024));
Packit 8f70b4
}
Packit 8f70b4
xstring& Speedometer::GetStrProper(float r)
Packit 8f70b4
{
Packit 8f70b4
   if(r<1)
Packit 8f70b4
      return xstring::get_tmp("");
Packit 8f70b4
   if(r<1024)
Packit 8f70b4
      return xstring::format(_("%.0f B/s"),r);
Packit 8f70b4
   else if(r<1024*1024)
Packit 8f70b4
      return xstring::format(_("%.1f KiB/s"),r/1024.);
Packit 8f70b4
   else
Packit 8f70b4
      return xstring::format(_("%.2f MiB/s"),r/(1024.*1024));
Packit 8f70b4
}
Packit 8f70b4
xstring& Speedometer::GetStr()
Packit 8f70b4
{
Packit 8f70b4
   return Valid() ? GetStr(Get()) : xstring::get_tmp("");
Packit 8f70b4
}
Packit 8f70b4
xstring& Speedometer::GetETAStrFromSize(off_t size)
Packit 8f70b4
{
Packit 8f70b4
   if(!Valid() || Get()<1)
Packit 8f70b4
      return xstring::get_tmp("");
Packit 8f70b4
   return GetETAStrFromTime(long(size/rate+.5));
Packit 8f70b4
}
Packit 8f70b4
xstring& Speedometer::GetETAStrFromTime(long eta)
Packit 8f70b4
{
Packit 8f70b4
   if(eta<0)
Packit 8f70b4
      return xstring::get_tmp("");
Packit 8f70b4
Packit 8f70b4
   unsigned flags=TimeInterval::TO_STR_TRANSLATE;
Packit 8f70b4
   if(terse)
Packit 8f70b4
      flags+=TimeInterval::TO_STR_TERSE;
Packit 8f70b4
Packit 8f70b4
   // for translator: Estimated Time of Arrival.
Packit 8f70b4
   return xstring::cat(_("eta:"),TimeInterval(eta,0).toString(flags),NULL);
Packit 8f70b4
}
Packit 8f70b4
const char *Speedometer::GetStrS(float r)
Packit 8f70b4
{
Packit 8f70b4
   xstring &rate=GetStr(r);
Packit 8f70b4
   if(rate.length())
Packit 8f70b4
      rate.append(' ');
Packit 8f70b4
   return rate;
Packit 8f70b4
}
Packit 8f70b4
const char *Speedometer::GetStrS()
Packit 8f70b4
{
Packit 8f70b4
   return Valid() ? GetStrS(Get()) : "";
Packit 8f70b4
}
Packit 8f70b4
const char *Speedometer::GetETAStrSFromSize(off_t s)
Packit 8f70b4
{
Packit 8f70b4
   xstring &eta=GetETAStrFromSize(s);
Packit 8f70b4
   if(eta.length())
Packit 8f70b4
      eta.append(' ');
Packit 8f70b4
   return eta;
Packit 8f70b4
}
Packit 8f70b4
const char *Speedometer::GetETAStrSFromTime(long s)
Packit 8f70b4
{
Packit 8f70b4
   xstring &eta=GetETAStrFromTime(s);
Packit 8f70b4
   if(eta.length())
Packit 8f70b4
      eta.append(' ');
Packit 8f70b4
   return eta;
Packit 8f70b4
}
Packit 8f70b4
void Speedometer::Reset()
Packit 8f70b4
{
Packit 8f70b4
   start=now;
Packit 8f70b4
   last_second=now;
Packit 8f70b4
   rate=0;
Packit 8f70b4
   last_bytes=0;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
ResDecl res_eta_terse("xfer:eta-terse",  "yes",ResMgr::BoolValidate,ResMgr::NoClosure);
Packit 8f70b4
void Speedometer::Reconfig(const char *n)
Packit 8f70b4
{
Packit 8f70b4
   terse=res_eta_terse.QueryBool(0);
Packit 8f70b4
   SetPeriod(ResMgr::Query(period_resource,0));
Packit 8f70b4
}