/// <summary> /// Generates a new Message, containing a RREQ for the wanted destination. /// Implemented according to Section 6.3: Generating Route Requests (RREQ) in the RFC. /// /// Method also handles buffering of RREQ, to avoid sending a duplicate in next iteration. /// </summary> /// <param name="destination">String representation of the destination address.</param> /// <returns>A Message with the RREQ enclosed. Includes destination address set to broadcast, and TTL adjusted according to the RFC.</returns> public Message GenerateInitialRreqMessage(string destination) { //If RREQ has been sent for this destination by me, and has not yet timed out, dont send a new one if (RreqAttemptExist(destination, _localAddress) && _time.CurrentTime < GetBufferedRreqAttempt(destination, _localAddress).ExpirationTime) { _logger.WriteLine("RREQ has been sent for this destination, and has not yet timed out, so dont send a new one."); return(null); } //Else, no RREQ for this destination waiting for response, send a new one RoutingTable.IncrementSequenceNumber(); //Must be incremented before sending a RREQ, per section 6.1 and 6.3 var knownEntry = RoutingTable.GetEntry(destination); //Get stale entry if it exists var rreq = new Rreq { G = _conf.AODV_UseGratuitousRREPs, D = _conf.AODV_DestinationRespondOnlyFlag, U = knownEntry == null, //True if no current DSN is known, per section 6.3 HopCount = 0, RouteRequestId = RoutingTable.NextRouteRequestId, DestinationAddress = destination, DestinationSequenceNumber = knownEntry?.DestinationSequenceNumber ?? 0, //Known DSN and U=false, or unknown DSN=0 and U=true OriginatorAddress = _localAddress, OriginatorSequenceNumber = RoutingTable.SequenceNumber }; //Any stale routing table entry waiting for a RREP (because of the newly generated RREQ) should not be expunged before current_time + 2 * NET_TRAVERSAL_TIME if (knownEntry != null) { knownEntry.ExpirationTime = _time.GetFutureTime(2 * _aodvParameters.NetTraversalTime); } //Create Message to send RREQ in, and adjust RREQ var msg = new Message(rreq, SimulationConstants.BroadcastAddress, _localAddress, _conf.MessageTtlValue); msg.Ttl = knownEntry?.HopCount + _aodvParameters.TtlIncrement ?? _aodvParameters.TtlStart; //Per section 6.4, use last known hopcount as initial TTL + TTL_INCREMENT //Buffer RREQ ID and Originator address, pr. section 6.3 (taken from rreq) var rreqAttempt = new BufferedRreqAttempt(0, rreq, _conf.AodvConfiguration.PathDiscoveryTime); rreqAttempt.ExpirationTime = _time.GetFutureTime(_aodvParameters.PathDiscoveryTime); AddBufferedRreqAttempt(rreqAttempt); return(msg); }
/// <summary> /// Buffers a sent RREQ. /// </summary> /// <param name="rreqAttempt">The RREQ attempt to buffer.</param> public void AddBufferedRreqAttempt(BufferedRreqAttempt rreqAttempt) { _bufferedRreqAttempts.Add(rreqAttempt); }