| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045 | /* <!-- copyright *//* * aria2 - The high speed download utility * * Copyright (C) 2006 Tatsuhiro Tsujikawa * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA * * In addition, as a special exception, the copyright holders give * permission to link the code of portions of this program with the * OpenSSL library under certain conditions as described in each * individual source file, and distribute linked combinations * including the two. * You must obey the GNU General Public License in all respects * for all of the code used other than OpenSSL.  If you modify * file(s) with this exception, you may extend this exception to your * version of the file(s), but you are not obligated to do so.  If you * do not wish to do so, delete this exception statement from your * version.  If you delete this exception statement from all source * files in the program, then also delete it here. *//* copyright --> */#include "DownloadEngine.h"#ifdef ENABLE_ASYNC_DNS#include "AsyncNameResolver.h"#endif // ENABLE_ASYNC_DNS#include "StatCalc.h"#include "RequestGroup.h"#include "RequestGroupMan.h"#include "FileAllocationMan.h"#include "CheckIntegrityMan.h"#include "DownloadResult.h"#include "StatCalc.h"#include "LogFactory.h"#include "Logger.h"#include "TimeA2.h"#include "a2time.h"#include "Socket.h"#include "Util.h"#include "a2functional.h"#include "DlAbortEx.h"#include "ServerStatMan.h"#include "CookieStorage.h"#include "A2STR.h"#include <signal.h>#include <cstring>#include <algorithm>#include <numeric>#include <cerrno>namespace aria2 {// 0 ... running// 1 ... stop signal detected// 2 ... stop signal processed by DownloadEngine// 3 ... 2nd stop signal(force shutdown) detected// 4 ... 2nd stop signal processed by DownloadEnginevolatile sig_atomic_t globalHaltRequested = 0;CommandEvent::CommandEvent(Command* command, int events):  _command(command), _events(events) {}bool CommandEvent::operator==(const CommandEvent& commandEvent) const{  return _command == commandEvent._command;}int CommandEvent::getEvents() const{  return _events;}void CommandEvent::addEvents(int events){  _events |= events;}void CommandEvent::removeEvents(int events){  _events &= (~events); }bool CommandEvent::eventsEmpty() const{  return _events == 0;}void CommandEvent::processEvents(int events){  if((_events&events) ||     ((SocketEntry::EVENT_ERROR|SocketEntry::EVENT_HUP)&events)) {    _command->setStatusActive();  }  if(SocketEntry::EVENT_READ&events) {    _command->readEventReceived();  }  if(SocketEntry::EVENT_WRITE&events) {    _command->writeEventReceived();  }  if(SocketEntry::EVENT_ERROR&events) {    _command->errorEventReceived();  }  if(SocketEntry::EVENT_HUP&events) {    _command->hupEventReceived();  }}#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNSADNSEvent::ADNSEvent(const SharedHandle<AsyncNameResolver>& resolver,		     Command* command,		     sock_t socket, int events):  _resolver(resolver), _command(command), _socket(socket), _events(events) {}bool ADNSEvent::operator==(const ADNSEvent& event) const{  return _resolver == event._resolver;}int ADNSEvent::getEvents() const{  return _events;}void ADNSEvent::processEvents(int events){  ares_socket_t readfd;  ares_socket_t writefd;  if(events&(SocketEntry::EVENT_READ|SocketEntry::EVENT_ERROR|SocketEntry::EVENT_HUP)) {    readfd = _socket;  } else {    readfd = ARES_SOCKET_BAD;  }  if(events&(SocketEntry::EVENT_WRITE|SocketEntry::EVENT_ERROR|SocketEntry::EVENT_HUP)) {    writefd = _socket;  } else {    writefd = ARES_SOCKET_BAD;  }  _resolver->process(readfd, writefd);  _command->setStatusActive();}#endif // HAVE_EPOLL && ENABLE_ASYNC_DNSSocketEntry::SocketEntry(sock_t socket):_socket(socket){#ifdef HAVE_EPOLL  memset(&_epEvent, 0, sizeof(struct epoll_event));#endif // HAVE_EPOLL}bool SocketEntry::operator==(const SocketEntry& entry) const{  return _socket == entry._socket;}bool SocketEntry::operator<(const SocketEntry& entry) const{  return _socket < entry._socket;}void SocketEntry::addCommandEvent(Command* command, int events){  CommandEvent cev(command, events);  std::deque<CommandEvent>::iterator i = std::find(_commandEvents.begin(),						   _commandEvents.end(),						   cev);  if(i == _commandEvents.end()) {    _commandEvents.push_back(cev);  } else {    (*i).addEvents(events);  }}void SocketEntry::removeCommandEvent(Command* command, int events){  CommandEvent cev(command, events);  std::deque<CommandEvent>::iterator i = std::find(_commandEvents.begin(),						   _commandEvents.end(),						   cev);  if(i == _commandEvents.end()) {    // not found  } else {    (*i).removeEvents(events);    if((*i).eventsEmpty()) {      _commandEvents.erase(i);    }  }}#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNSvoid SocketEntry::addADNSEvent(const SharedHandle<AsyncNameResolver>& resolver,			       Command* command, int events){  ADNSEvent aev(resolver, command, _socket, events);  std::deque<ADNSEvent>::iterator i = std::find(_adnsEvents.begin(),						_adnsEvents.end(),						aev);  if(i == _adnsEvents.end()) {    _adnsEvents.push_back(aev);  }}void SocketEntry::removeADNSEvent(const SharedHandle<AsyncNameResolver>& resolver,				  Command* command){  ADNSEvent aev(resolver, command, _socket, 0);  std::deque<ADNSEvent>::iterator i = std::find(_adnsEvents.begin(),						_adnsEvents.end(),						aev);  if(i == _adnsEvents.end()) {    // not found  } else {    _adnsEvents.erase(i);  }}#endif // HAVE_EPOLL && ENABLE_ASYNC_DNSvoid SocketEntry::processEvents(int events){  std::for_each(_commandEvents.begin(), _commandEvents.end(),		std::bind2nd(std::mem_fun_ref(&CommandEvent::processEvents),			     events));#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNS  std::for_each(_adnsEvents.begin(), _adnsEvents.end(),		std::bind2nd(std::mem_fun_ref(&ADNSEvent::processEvents),			     events));#endif // HAVE_EPOLL && ENABLE_ASYNC_DNS}sock_t SocketEntry::getSocket() const{  return _socket;}bool SocketEntry::eventEmpty() const{#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNS  return _commandEvents.empty() && _adnsEvents.empty();#else // !(HAVE_EPOLL && ENABLE_ASYNC_DNS)  return _commandEvents.empty();#endif // !(HAVE_EPOLL && ENABLE_ASYNC_DNS)}class AccEvent {public:  int operator()(int events, const CommandEvent& commandEvent) const  {    return events|commandEvent.getEvents();  }#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNS  int operator()(int events, const ADNSEvent& adnsEvent) const  {    return events|adnsEvent.getEvents();  }#endif // HAVE_EPOLL && ENABLE_ASYNC_DNS};#ifdef  HAVE_EPOLLstruct epoll_event& SocketEntry::getEpEvent(){  _epEvent.data.ptr = this;#ifdef ENABLE_ASYNC_DNS  _epEvent.events =    std::accumulate(_adnsEvents.begin(),		    _adnsEvents.end(),		    std::accumulate(_commandEvents.begin(),				    _commandEvents.end(), 0, AccEvent()),		    AccEvent());#else // !ENABLE_ASYNC_DNS  _epEvent.events =    std::accumulate(_commandEvents.begin(), _commandEvents.end(), 0, AccEvent());#endif // !ENABLE_ASYNC_DNS  return _epEvent;}#else // !HAVE_EPOLLint SocketEntry::getEvents(){  return    std::accumulate(_commandEvents.begin(), _commandEvents.end(), 0, AccEvent());}#endif // !HAVE_EPOLL#ifdef ENABLE_ASYNC_DNSAsyncNameResolverEntry::AsyncNameResolverEntry(const SharedHandle<AsyncNameResolver>& nameResolver, Command* command):  _nameResolver(nameResolver), _command(command)#ifdef HAVE_EPOLL  , _socketsSize(0)#endif // HAVE_EPOLL{}bool AsyncNameResolverEntry::operator==(const AsyncNameResolverEntry& entry){  return _nameResolver == entry._nameResolver &&    _command == entry._command;}#ifdef HAVE_EPOLLvoid AsyncNameResolverEntry::addSocketEvents(DownloadEngine* e){  _socketsSize = 0;  int mask = _nameResolver->getsock(_sockets);  if(mask == 0) {    return;  }  size_t i;  for(i = 0; i < ARES_GETSOCK_MAXNUM; ++i) {    //epoll_event_t* epEventPtr = &_epEvents[_socketsSize];    int events = 0;    if(ARES_GETSOCK_READABLE(mask, i)) {      events |= EPOLLIN;    }    if(ARES_GETSOCK_WRITABLE(mask, i)) {      events |= EPOLLOUT;    }    if(events == 0) {      // assume no further sockets are returned.      break;    }    e->addSocketEvents(_sockets[i], _command, events, _nameResolver);  }  _socketsSize = i;}  void AsyncNameResolverEntry::removeSocketEvents(DownloadEngine* e){  for(size_t i = 0; i < _socketsSize; ++i) {    e->deleteSocketEvents(_sockets[i], _command, 0, _nameResolver);  }}#else // !HAVE_EPOLLint AsyncNameResolverEntry::getFds(fd_set* rfdsPtr, fd_set* wfdsPtr){  return _nameResolver->getFds(rfdsPtr, wfdsPtr);}void AsyncNameResolverEntry::process(fd_set* rfdsPtr, fd_set* wfdsPtr){  _nameResolver->process(rfdsPtr, wfdsPtr);  switch(_nameResolver->getStatus()) {  case AsyncNameResolver::STATUS_SUCCESS:  case AsyncNameResolver::STATUS_ERROR:    _command->setStatusActive();    break;  default:    break;  }}#endif // !HAVE_EPOLL#endif // ENABLE_ASYNC_DNSDownloadEngine::DownloadEngine():logger(LogFactory::getInstance()),				 _haltRequested(false),				 _noWait(false),				 _cookieStorage(new CookieStorage()){#ifdef HAVE_EPOLL  _epfd = epoll_create(EPOLL_EVENTS_MAX);  _epEvents = new struct epoll_event[EPOLL_EVENTS_MAX];#else // !HAVE_EPOLL  updateFdSet();#endif // !HAVE_EPOLL}DownloadEngine::~DownloadEngine() {  cleanQueue();#ifdef HAVE_EPOLL  if(_epfd != -1) {    int r;    while((r = close(_epfd)) == -1 && errno == EINTR);    if(r == -1) {      logger->error("Error occurred while closing epoll file descriptor %d: %s",		    _epfd, strerror(errno));    }  }  delete [] _epEvents;#endif // HAVE_EPOLL}void DownloadEngine::cleanQueue() {  std::for_each(commands.begin(), commands.end(), Deleter());  commands.clear();}static void executeCommand(std::deque<Command*>& commands,			   Command::STATUS statusFilter){  size_t max = commands.size();  for(size_t i = 0; i < max; i++) {    Command* com = commands.front();    commands.pop_front();    if(com->statusMatch(statusFilter)) {      com->transitStatus();      if(com->execute()) {	delete com;	com = 0;      }    } else {      commands.push_back(com);    }    if(com) {      com->clearIOEvents();    }  }}void DownloadEngine::run() {#ifdef HAVE_EPOLL  if(_epfd == -1) {    throw DlAbortEx("epoll_init() failed.");  }#endif // HAVE_EPOLL  Time cp;  cp.setTimeInSec(0);  while(!commands.empty() || !_routineCommands.empty()) {    if(cp.elapsed(1)) {      cp.reset();      executeCommand(commands, Command::STATUS_ALL);    } else {      executeCommand(commands, Command::STATUS_ACTIVE);    }    executeCommand(_routineCommands, Command::STATUS_ALL);    afterEachIteration();    if(!commands.empty()) {      waitData();    }    _noWait = false;    calculateStatistics();  }  onEndOfRun();}void DownloadEngine::shortSleep() const {  struct timeval tv;  tv.tv_sec = 0;  tv.tv_usec = 1000;  fd_set rfds;  FD_ZERO(&rfds);  select(0, &rfds, NULL, NULL, &tv);}void DownloadEngine::waitData(){#ifdef HAVE_EPOLL  // timeout is millisec  int timeout = _noWait ? 0 : 1000;  int res;  while((res = epoll_wait(_epfd, _epEvents, EPOLL_EVENTS_MAX, timeout)) == -1 &&	errno == EINTR);  if(res > 0) {    for(int i = 0; i < res; ++i) {      SocketEntry* p = (SocketEntry*)_epEvents[i].data.ptr;      p->processEvents(_epEvents[i].events);    }  }  // TODO timeout of name resolver is determined in Command(AbstractCommand,  // DHTEntryPoint...Command)#else // !HAVE_EPOLL  fd_set rfds;  fd_set wfds;  struct timeval tv;    memcpy(&rfds, &rfdset, sizeof(fd_set));  memcpy(&wfds, &wfdset, sizeof(fd_set));  #ifdef ENABLE_ASYNC_DNS  for(std::deque<SharedHandle<AsyncNameResolverEntry> >::iterator itr =	nameResolverEntries.begin(); itr != nameResolverEntries.end(); ++itr) {    SharedHandle<AsyncNameResolverEntry>& entry = *itr;    int fd = entry->getFds(&rfds, &wfds);    // TODO force error if fd == 0    if(fdmax < fd) {      fdmax = fd;    }  }#endif // ENABLE_ASYNC_DNS  tv.tv_sec = _noWait ? 0 : 1;  tv.tv_usec = 0;  int retval = select(fdmax+1, &rfds, &wfds, NULL, &tv);  if(retval > 0) {    for(std::deque<SharedHandle<SocketEntry> >::iterator i =	  socketEntries.begin(); i != socketEntries.end(); ++i) {      int events = 0;      if(FD_ISSET((*i)->getSocket(), &rfds)) {	events |= SocketEntry::EVENT_READ;      }      if(FD_ISSET((*i)->getSocket(), &wfds)) {	events |= SocketEntry::EVENT_WRITE;      }      (*i)->processEvents(events);    }  }#ifdef ENABLE_ASYNC_DNS  for(std::deque<SharedHandle<AsyncNameResolverEntry> >::iterator i =	nameResolverEntries.begin(); i != nameResolverEntries.end(); ++i) {    (*i)->process(&rfds, &wfds);  }#endif // ENABLE_ASYNC_DNS#endif // !HAVE_EPOLL}#ifndef HAVE_EPOLLvoid DownloadEngine::updateFdSet() {  fdmax = 0;  FD_ZERO(&rfdset);  FD_ZERO(&wfdset);  for(std::deque<SharedHandle<SocketEntry> >::iterator i =	socketEntries.begin(); i != socketEntries.end(); ++i) {    sock_t fd = (*i)->getSocket();    int events = (*i)->getEvents();    if(events&SocketEntry::EVENT_READ) {      FD_SET(fd, &rfdset);    }    if(events&SocketEntry::EVENT_WRITE) {      FD_SET(fd, &wfdset);    }    if(fdmax < fd) {      fdmax = fd;    }  }}#endif // !HAVE_EPOLLbool DownloadEngine::addSocketEvents(sock_t socket, Command* command, int events#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNS				     ,const SharedHandle<AsyncNameResolver>& rs#endif // HAVE_EPOLL && ENABLE_ASYNC_DNS				     ){  SharedHandle<SocketEntry> socketEntry(new SocketEntry(socket));  std::deque<SharedHandle<SocketEntry> >::iterator i =    std::lower_bound(socketEntries.begin(), socketEntries.end(), socketEntry);  int r = 0;  if(i != socketEntries.end() && (*i) == socketEntry) {#ifdef HAVE_EPOLL#ifdef ENABLE_ASYNC_DNS    if(rs.isNull()) {      (*i)->addCommandEvent(command, events);    } else {      (*i)->addADNSEvent(rs, command, events);    }#else // !ENABLE_ASYNC_DNS    (*i)->addCommandEvent(command, events);#endif // !ENABLE_ASYNC_DNS    r = epoll_ctl(_epfd, EPOLL_CTL_MOD, (*i)->getSocket(), &(*i)->getEpEvent());    if(r == -1) {      // try EPOLL_CTL_ADD: There is a chance that previously socket X is      // added to epoll, but it is closed and is not yet removed from      // SocketEntries. In this case, EPOLL_CTL_MOD is failed with ENOENT.      r = epoll_ctl(_epfd, EPOLL_CTL_ADD, (*i)->getSocket(), &(*i)->getEpEvent());    }#else // !HAVE_EPOLL    (*i)->addCommandEvent(command, events);#endif // !HAVE_EPOLL  } else {    socketEntries.insert(i, socketEntry);#ifdef HAVE_EPOLL#ifdef ENABLE_ASYNC_DNS    if(rs.isNull()) {      socketEntry->addCommandEvent(command, events);    } else {      socketEntry->addADNSEvent(rs, command, events);    }#else // !ENABLE_ASYNC_DNS    socketEntry->addCommandEvent(command, events);    #endif // !ENABLE_ASYNC_DNS    r = epoll_ctl(_epfd, EPOLL_CTL_ADD, socketEntry->getSocket(), &socketEntry->getEpEvent());#else // !HAVE_EPOLL    socketEntry->addCommandEvent(command, events);#endif // !HAVE_EPOLL  }#ifndef HAVE_EPOLL  updateFdSet();#endif // !HAVE_EPOLL  if(r == -1) {    logger->debug("Failed to add socket event %d:%s", socket, strerror(errno));    return false;  } else {    return true;  }}bool DownloadEngine::deleteSocketEvents(sock_t socket, Command* command, int events#if defined HAVE_EPOLL && defined ENABLE_ASYNC_DNS					,const SharedHandle<AsyncNameResolver>& rs#endif // HAVE_EPOLL && ENABLE_ASYNC_DNS					){  SharedHandle<SocketEntry> socketEntry(new SocketEntry(socket));  std::deque<SharedHandle<SocketEntry> >::iterator i =    std::lower_bound(socketEntries.begin(), socketEntries.end(), socketEntry);  if(i != socketEntries.end() && (*i) == socketEntry) {#ifdef HAVE_EPOLL#ifdef ENABLE_ASYNC_DNS    if(rs.isNull()) {      (*i)->removeCommandEvent(command, events);    } else {      (*i)->removeADNSEvent(rs, command);    }#else // !ENABLE_ASYNC_DNS    (*i)->removeCommandEvent(command, events);    #endif // !ENABLE_ASYNC_DNS#else // !HAVE_EPOLL    (*i)->removeCommandEvent(command, events);#endif // !HAVE_EPOLL    int r = 0;    if((*i)->eventEmpty()) {#ifdef HAVE_EPOLL      r = epoll_ctl(_epfd, EPOLL_CTL_DEL, (*i)->getSocket(), 0);#endif // HAVE_EPOLL      socketEntries.erase(i);    } else {#ifdef HAVE_EPOLL      // If socket is closed, then it seems it is automatically removed from      // epoll, so following EPOLL_CTL_MOD may fail.      r = epoll_ctl(_epfd, EPOLL_CTL_MOD, (*i)->getSocket(), &(*i)->getEpEvent());      if(r == -1) {	logger->debug("Failed to delete socket event, but may be ignored:%s", strerror(errno));      }#endif // HAVE_EPOLL    }#ifndef HAVE_EPOLL    updateFdSet();#endif // !HAVE_EPOLL    if(r == -1) {      logger->debug("Failed to delete socket event:%s", strerror(errno));      return false;    } else {      return true;    }  } else {    logger->debug("Socket %d is not found in SocketEntries.", socket);    return false;  }}bool DownloadEngine::addSocketForReadCheck(const SocketHandle& socket,					   Command* command){  return addSocketEvents(socket->getSockfd(), command, SocketEntry::EVENT_READ);}bool DownloadEngine::deleteSocketForReadCheck(const SocketHandle& socket,					      Command* command){  return deleteSocketEvents(socket->getSockfd(), command, SocketEntry::EVENT_READ);}bool DownloadEngine::addSocketForWriteCheck(const SocketHandle& socket,					    Command* command){  return addSocketEvents(socket->getSockfd(), command, SocketEntry::EVENT_WRITE);}bool DownloadEngine::deleteSocketForWriteCheck(const SocketHandle& socket,					       Command* command){  return deleteSocketEvents(socket->getSockfd(), command, SocketEntry::EVENT_WRITE);}void DownloadEngine::calculateStatistics(){  if(!_statCalc.isNull()) {    _statCalc->calculateStat(_requestGroupMan, _fileAllocationMan, _checkIntegrityMan);  }}void DownloadEngine::onEndOfRun(){  _requestGroupMan->updateServerStat();  _requestGroupMan->closeFile();  _requestGroupMan->save();}void DownloadEngine::afterEachIteration(){  if(globalHaltRequested == 1) {    logger->notice(_("Shutdown sequence commencing... Press Ctrl-C again for emergency shutdown."));    requestHalt();    globalHaltRequested = 2;  } else if(globalHaltRequested == 3) {    logger->notice(_("Emergency shutdown sequence commencing..."));    _requestGroupMan->forceHalt();    globalHaltRequested = 4;  }}void DownloadEngine::requestHalt(){  _haltRequested = true;  _requestGroupMan->halt();}void DownloadEngine::fillCommand(){  std::deque<Command*> commands;  _requestGroupMan->getInitialCommands(commands, this);  addCommand(commands);}void DownloadEngine::setStatCalc(const StatCalcHandle& statCalc){  _statCalc = statCalc;}void DownloadEngine::addCommand(const Commands& commands){  this->commands.insert(this->commands.end(), commands.begin(), commands.end());}#ifdef ENABLE_ASYNC_DNSbool DownloadEngine::addNameResolverCheck(const SharedHandle<AsyncNameResolver>& resolver, Command* command){  SharedHandle<AsyncNameResolverEntry> entry    (new AsyncNameResolverEntry(resolver, command));  std::deque<SharedHandle<AsyncNameResolverEntry> >::iterator itr =    std::find(nameResolverEntries.begin(), nameResolverEntries.end(), entry);  if(itr == nameResolverEntries.end()) {    nameResolverEntries.push_back(entry);#ifdef HAVE_EPOLL    entry->addSocketEvents(this);#endif // HAVE_EPOLL    return true;  } else {    return false;  }}bool DownloadEngine::deleteNameResolverCheck(const SharedHandle<AsyncNameResolver>& resolver, Command* command){  SharedHandle<AsyncNameResolverEntry> entry    (new AsyncNameResolverEntry(resolver, command));  std::deque<SharedHandle<AsyncNameResolverEntry> >::iterator itr =    std::find(nameResolverEntries.begin(), nameResolverEntries.end(), entry);  if(itr == nameResolverEntries.end()) {    return false;  } else {#ifdef HAVE_EPOLL    (*itr)->removeSocketEvents(this);#endif // HAVE_EPOLL    nameResolverEntries.erase(itr);    return true;  }}#endif // ENABLE_ASYNC_DNSvoid DownloadEngine::setNoWait(bool b){  _noWait = b;}void DownloadEngine::addRoutineCommand(Command* command){  _routineCommands.push_back(command);}SharedHandle<CookieStorage> DownloadEngine::getCookieStorage() const{  return _cookieStorage;}void DownloadEngine::poolSocket(const std::string& ipaddr,				uint16_t port,				const SocketPoolEntry& entry){  std::string addr = ipaddr+":"+Util::uitos(port);  logger->info("Pool socket for %s", addr.c_str());  std::multimap<std::string, SocketPoolEntry>::value_type p(addr, entry);  _socketPool.insert(p);  if(_lastSocketPoolScan.elapsed(60)) {    std::multimap<std::string, SocketPoolEntry> newPool;    logger->debug("Scaning SocketPool and erasing timed out entry.");    _lastSocketPoolScan.reset();    for(std::multimap<std::string, SocketPoolEntry>::iterator i =	  _socketPool.begin(); i != _socketPool.end(); ++i) {      if(!(*i).second.isTimeout()) {	newPool.insert(*i);      }    }    logger->debug      ("%lu entries removed.",       static_cast<unsigned long>(_socketPool.size()-newPool.size()));    _socketPool = newPool;  }}void DownloadEngine::poolSocket(const std::string& ipaddr, uint16_t port, const SharedHandle<SocketCore>& sock, const std::map<std::string, std::string>& options, time_t timeout){  SocketPoolEntry e(sock, options, timeout);  poolSocket(ipaddr, port, e);}void DownloadEngine::poolSocket(const std::string& ipaddr, uint16_t port, const SharedHandle<SocketCore>& sock, time_t timeout){  SocketPoolEntry e(sock, std::map<std::string, std::string>(), timeout);  poolSocket(ipaddr, port, e);}std::multimap<std::string, DownloadEngine::SocketPoolEntry>::iteratorDownloadEngine::findSocketPoolEntry(const std::string& ipaddr, uint16_t port){  std::string addr = ipaddr+":"+Util::uitos(port);  std::pair<std::multimap<std::string, SocketPoolEntry>::iterator,    std::multimap<std::string, SocketPoolEntry>::iterator> range =    _socketPool.equal_range(addr);  for(std::multimap<std::string, SocketPoolEntry>::iterator i = range.first;      i != range.second; ++i) {    const SocketPoolEntry& e = (*i).second;    if(!e.isTimeout()) {      logger->info("Found socket for %s", addr.c_str());      return i;    }  }  return _socketPool.end();}SharedHandle<SocketCore>DownloadEngine::popPooledSocket(const std::string& ipaddr, uint16_t port){  SharedHandle<SocketCore> s;  std::multimap<std::string, SocketPoolEntry>::iterator i =    findSocketPoolEntry(ipaddr, port);  if(i != _socketPool.end()) {    s = (*i).second.getSocket();    _socketPool.erase(i);  }  return s;}SharedHandle<SocketCore>DownloadEngine::popPooledSocket(std::map<std::string, std::string>& options,				const std::string& ipaddr, uint16_t port){  SharedHandle<SocketCore> s;  std::multimap<std::string, SocketPoolEntry>::iterator i =    findSocketPoolEntry(ipaddr, port);  if(i != _socketPool.end()) {    s = (*i).second.getSocket();    options = (*i).second.getOptions();    _socketPool.erase(i);  }  return s;  }SharedHandle<SocketCore>DownloadEngine::popPooledSocket(const std::deque<std::string>& ipaddrs, uint16_t port){  SharedHandle<SocketCore> s;  for(std::deque<std::string>::const_iterator i = ipaddrs.begin();      i != ipaddrs.end(); ++i) {    s = popPooledSocket(*i, port);    if(!s.isNull()) {      break;    }  }  return s;}SharedHandle<SocketCore>DownloadEngine::popPooledSocket(std::map<std::string, std::string>& options, const std::deque<std::string>& ipaddrs, uint16_t port){  SharedHandle<SocketCore> s;  for(std::deque<std::string>::const_iterator i = ipaddrs.begin();      i != ipaddrs.end(); ++i) {    s = popPooledSocket(options, *i, port);    if(!s.isNull()) {      break;    }  }  return s;}DownloadEngine::SocketPoolEntry::SocketPoolEntry(const SharedHandle<SocketCore>& socket, const std::map<std::string, std::string>& options, time_t timeout):  _socket(socket),  _options(options),  _timeout(timeout) {}DownloadEngine::SocketPoolEntry::~SocketPoolEntry() {}bool DownloadEngine::SocketPoolEntry::isTimeout() const{  return _registeredTime.elapsed(_timeout);}SharedHandle<SocketCore> DownloadEngine::SocketPoolEntry::getSocket() const{  return _socket;}const std::map<std::string, std::string>&DownloadEngine::SocketPoolEntry::getOptions() const{  return _options;}} // namespace aria2
 |