/* * libretroshare/src/services p3rtt.cc * * Round Trip Time Measurement for RetroShare. * * Copyright 2011-2013 by Robert Fernie. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License Version 2.1 as published by the Free Software Foundation. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 * USA. * * Please report all bugs and problems to "retroshare@lunamutt.com". * */ #include #include "util/rsdir.h" #include "retroshare/rsiface.h" #include "pqi/pqibin.h" #include "pqi/pqistore.h" #include "pqi/p3linkmgr.h" #include "rsserver/p3face.h" #include "services/p3rtt.h" #include "rsitems/rsrttitems.h" #include #include /**** * #define DEBUG_RTT 1 ****/ /* DEFINE INTERFACE POINTER! */ RsRtt *rsRtt = NULL; #define MAX_PONG_RESULTS 150 #define RTT_PING_PERIOD 10 /************ IMPLEMENTATION NOTES ********************************* * * Voice over Retroshare ;) * * This will be a simple test VoIP system aimed at testing out the possibilities. * * Important things to test: * 1) lag, and variability in data rate * - To do this we time tag every packet..., the destination can use this info to calculate the results. * - Like imixitup. Dt = clock_diff + lag. * we expect clock_diff to be relatively constant, but lag to vary. * lag cannot be negative, so minimal Dt is ~clock_diff, and delays on this are considered +lag. * * 2) we could directly measure lag. ping back and forth with Timestamps. * * 3) we also want to measure bandwidth... * - not sure the best method? * one way: send a ping, then a large amount of data (5 seconds worth), then another ping. * the delta in timestamps should be a decent indication of bandwidth. * say we have a 100kb/s connection... need 500kb. * actually the amount of data should be based on a reasonable maximum that we require. * what does decent video require? * Audio we can test for 64kb/s - which seems like a decent rate: e.g. mono, 16bit 22k = 1 x 2 x 22k = 44 kilobytes/sec * best to do this without a VoIP call going on ;) * * */ #ifdef WINDOWS_SYS #include #include #endif static double getCurrentTS() { #ifndef WINDOWS_SYS struct timeval cts_tmp; gettimeofday(&cts_tmp, NULL); double cts = (cts_tmp.tv_sec) + ((double) cts_tmp.tv_usec) / 1000000.0; #else struct _timeb timebuf; _ftime( &timebuf); double cts = (timebuf.time) + ((double) timebuf.millitm) / 1000.0; #endif return cts; } static uint64_t convertTsTo64bits(double ts) { uint32_t secs = (uint32_t) ts; uint32_t usecs = (uint32_t) ((ts - (double) secs) * 1000000); uint64_t bits = (((uint64_t) secs) << 32) + usecs; return bits; } static double convert64bitsToTs(uint64_t bits) { uint32_t usecs = (uint32_t) (bits & 0xffffffff); uint32_t secs = (uint32_t) ((bits >> 32) & 0xffffffff); double ts = (secs) + ((double) usecs) / 1000000.0; return ts; } p3rtt::p3rtt(p3ServiceControl *sc) :p3FastService(), mRttMtx("p3rtt"), mServiceCtrl(sc) { addSerialType(new RsRttSerialiser()); mSentPingTime = 0; mCounter = 0; } const std::string RTT_APP_NAME = "rtt"; const uint16_t RTT_APP_MAJOR_VERSION = 1; const uint16_t RTT_APP_MINOR_VERSION = 0; const uint16_t RTT_MIN_MAJOR_VERSION = 1; const uint16_t RTT_MIN_MINOR_VERSION = 0; RsServiceInfo p3rtt::getServiceInfo() { return RsServiceInfo(RS_SERVICE_TYPE_RTT, RTT_APP_NAME, RTT_APP_MAJOR_VERSION, RTT_APP_MINOR_VERSION, RTT_MIN_MAJOR_VERSION, RTT_MIN_MINOR_VERSION); } int p3rtt::tick() { sendPackets(); return 0; } int p3rtt::status() { return 1; } int p3rtt::sendPackets() { time_t now = time(NULL); time_t pt; { RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ pt = mSentPingTime; } if (now >= pt+RTT_PING_PERIOD) { sendPingMeasurements(); RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ mSentPingTime = now; } return true ; } void p3rtt::sendPingMeasurements() { /* we ping our peers */ /* who is online? */ std::set idList; mServiceCtrl->getPeersConnected(getServiceInfo().mServiceType, idList); /* prepare packets */ std::set::iterator it; for(it = idList.begin(); it != idList.end(); ++it) { double ts = getCurrentTS(); /* create the packet */ RsRttPingItem *pingPkt = new RsRttPingItem(); pingPkt->PeerId(*it); pingPkt->mSeqNo = mCounter; pingPkt->mPingTS = convertTsTo64bits(ts); storePingAttempt(*it, ts, mCounter); #ifdef DEBUG_RTT std::cerr << "p3rtt::sendPingMeasurements() Pinging: " << *it << " [" << pingPkt->mSeqNo << "," << std::hex << pingPkt->mPingTS << std::dec << "]" << std::endl;; #endif sendItem(pingPkt); } RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ mCounter++; } bool p3rtt::recvItem(RsItem *item) { switch(item->PacketSubType()) { default: break; case RS_PKT_SUBTYPE_RTT_PING: { handlePing(item); } break; case RS_PKT_SUBTYPE_RTT_PONG: { handlePong(item); } break; } /* clean up */ delete item; return true ; } int p3rtt::handlePing(RsItem *item) { /* cast to right type */ RsRttPingItem *ping = (RsRttPingItem *) item; double ts = getCurrentTS(); #ifdef DEBUG_RTT std::cerr << "p3rtt::handlePing() from: " << ping->PeerId() << " - [" << ping->mSeqNo << "," << std::hex << ping->mPingTS << std::dec << "] " << std::endl; std::cerr << "incoming ping travel time: " << ts - convert64bitsToTs(ping->mPingTS) << std::endl; #endif /* with a ping, we just respond as quickly as possible - they do all the analysis */ RsRttPongItem *pong = new RsRttPongItem(); pong->PeerId(ping->PeerId()); pong->mPingTS = ping->mPingTS; pong->mSeqNo = ping->mSeqNo; // add our timestamp. pong->mPongTS = convertTsTo64bits(ts); #ifdef DEBUG_RTT static double mLastResponseToPong = 0.0 ;// bad stuff std::cerr << "Delay since last response to PONG: " << ts - mLastResponseToPong << std::endl; mLastResponseToPong = ts ; #endif sendItem(pong); return true ; } int p3rtt::handlePong(RsItem *item) { /* cast to right type */ RsRttPongItem *pong = (RsRttPongItem *) item; #ifdef DEBUG_RTT std::cerr << "p3rtt::handlePong() from: " << pong->PeerId() << " - [" << pong->mSeqNo << "," << std::hex << pong->mPingTS << " -> " << pong->mPongTS << std::dec << "] "<< std::endl; #endif /* with a pong, we do the maths! */ double recvTS = getCurrentTS(); double pingTS = convert64bitsToTs(pong->mPingTS); double pongTS = convert64bitsToTs(pong->mPongTS); double rtt = recvTS - pingTS; double offset = pongTS - (recvTS - rtt / 2.0); // so to get to their time, we go ourTS + offset. #ifdef DEBUG_RTT std::cerr << "incoming pong travel time: " << recvTS - convert64bitsToTs(pong->mPongTS) << std::endl; std::cerr << " RTT analysis: pingTS: " << std::setprecision(16) << pingTS << ", pongTS: " << pongTS << ", recvTS: " << std::setprecision(16) << recvTS << " ==> rtt: " << rtt << ", offset: " << offset << std::endl; #endif storePongResult(pong->PeerId(), pong->mSeqNo, recvTS, rtt, offset); return true ; } int p3rtt::storePingAttempt(const RsPeerId& id, double ts, uint32_t seqno) { RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ /* find corresponding local data */ RttPeerInfo *peerInfo = locked_GetPeerInfo(id); #ifdef DEBUG_RTT std::cerr << "Delay since previous ping attempt: " << ts - peerInfo->mCurrentPingTS << std::endl; #endif peerInfo->mCurrentPingTS = ts; peerInfo->mCurrentPingCounter = seqno; peerInfo->mSentPings++; if (!peerInfo->mCurrentPongRecvd) { peerInfo->mLostPongs++; } peerInfo->mCurrentPongRecvd = true; return 1; } int p3rtt::storePongResult(const RsPeerId& id, uint32_t counter, double recv_ts, double rtt, double offset) { RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ /* find corresponding local data */ RttPeerInfo *peerInfo = locked_GetPeerInfo(id); if (peerInfo->mCurrentPingCounter != counter) { #ifdef DEBUG_RTT std::cerr << "p3rtt::storePongResult() ERROR Severly Delayed Measurements!" << std::endl; #endif } else { peerInfo->mCurrentPongRecvd = true; } #ifdef DEBUG_RTT if(!peerInfo->mPongResults.empty()) std::cerr << "Delay since last pong: " << recv_ts - peerInfo->mPongResults.back().mTS << std::endl; #endif peerInfo->mPongResults.push_back(RsRttPongResult(recv_ts, rtt, offset)); while(peerInfo->mPongResults.size() > MAX_PONG_RESULTS) { peerInfo->mPongResults.pop_front(); } //Wait at least 20 pongs before compute mean time offset if(peerInfo->mPongResults.size() > 20) { double mean = 0; for(std::list::const_iterator prIt = peerInfo->mPongResults.begin(), end = peerInfo->mPongResults.end(); prIt != end; ++ prIt) { mean += prIt->mOffset; } peerInfo->mCurrentMeanOffset = mean / peerInfo->mPongResults.size(); if(fabs(peerInfo->mCurrentMeanOffset) > 120) { p3Notify *notify = RsServer::notify(); if (notify) { //notify->AddPopupMessage(RS_POPUP_OFFSET, eerInfo->mId.toStdString(),"", "Time Offset: "); notify->AddFeedItem(RS_FEED_ITEM_PEER_OFFSET, peerInfo->mId.toStdString()); } std::cerr << "(WW) Peer:" << peerInfo->mId << " get time offset more than two minutes with you!!!" << std::endl; } } return 1; } uint32_t p3rtt::getPongResults(const RsPeerId& id, int n, std::list &results) { RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ RttPeerInfo *peer = locked_GetPeerInfo(id); std::list::reverse_iterator it; int i = 0; for(it = peer->mPongResults.rbegin(); (it != peer->mPongResults.rend()) && (i < n); ++it, i++) { /* reversing order - so its easy to trim later */ results.push_back(*it); } return i ; } double p3rtt::getMeanOffset(const RsPeerId &id) { RsStackMutex stack(mRttMtx); /****** LOCKED MUTEX *******/ RttPeerInfo *peer = locked_GetPeerInfo(id); if(peer) return peer->mCurrentMeanOffset; else return 0; } RttPeerInfo *p3rtt::locked_GetPeerInfo(const RsPeerId& id) { std::map::iterator it; it = mPeerInfo.find(id); if (it == mPeerInfo.end()) { /* add it in */ RttPeerInfo pinfo; /* initialise entry */ pinfo.initialisePeerInfo(id); mPeerInfo[id] = pinfo; it = mPeerInfo.find(id); } return &(it->second); } bool RttPeerInfo::initialisePeerInfo(const RsPeerId& id) { mId = id; /* reset variables */ mCurrentPingTS = 0; mCurrentPingCounter = 0; mCurrentPongRecvd = true; mSentPings = 0; mLostPongs = 0; mPongResults.clear(); return true; }