A Discrete-Event Network Simulator
API
Loading...
Searching...
No Matches
wifi-default-ack-manager.cc
Go to the documentation of this file.
1/*
2 * Copyright (c) 2020 Universita' degli Studi di Napoli Federico II
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation;
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
16 *
17 * Author: Stefano Avallone <stavallo@unina.it>
18 */
19
21
22#include "ap-wifi-mac.h"
23#include "ctrl-headers.h"
24#include "qos-utils.h"
25#include "wifi-mac-queue.h"
26#include "wifi-mpdu.h"
27#include "wifi-protection.h"
28#include "wifi-tx-parameters.h"
29
30#include "ns3/he-frame-exchange-manager.h"
31#include "ns3/log.h"
32
33namespace ns3
34{
35
36NS_LOG_COMPONENT_DEFINE("WifiDefaultAckManager");
37
38NS_OBJECT_ENSURE_REGISTERED(WifiDefaultAckManager);
39
40TypeId
42{
43 static TypeId tid =
44 TypeId("ns3::WifiDefaultAckManager")
46 .SetGroupName("Wifi")
47 .AddConstructor<WifiDefaultAckManager>()
48 .AddAttribute("UseExplicitBar",
49 "Specify whether to send Block Ack Requests (if true) or use"
50 " Implicit Block Ack Request ack policy (if false).",
51 BooleanValue(false),
54 .AddAttribute("BaThreshold",
55 "Immediate acknowledgment is requested upon transmission of a frame "
56 "whose sequence number is distant at least BaThreshold multiplied "
57 "by the transmit window size from the starting sequence number of "
58 "the transmit window. Set to zero to request a response for every "
59 "transmitted frame.",
60 DoubleValue(0.0),
62 MakeDoubleChecker<double>(0.0, 1.0))
63 .AddAttribute(
64 "DlMuAckSequenceType",
65 "Type of the acknowledgment sequence for DL MU PPDUs.",
67 MakeEnumAccessor<WifiAcknowledgment::Method>(&WifiDefaultAckManager::m_dlMuAckType),
69 "DL_MU_BAR_BA_SEQUENCE",
71 "DL_MU_TF_MU_BAR",
73 "DL_MU_AGGREGATE_TF"))
74 .AddAttribute("MaxBlockAckMcs",
75 "The MCS used to send a BlockAck in a TB PPDU is the minimum between "
76 "the MCS used for the PSDU sent in the preceding DL MU PPDU and the "
77 "value of this attribute.",
80 MakeUintegerChecker<uint8_t>(0, 11));
81 return tid;
82}
83
85{
86 NS_LOG_FUNCTION(this);
87}
88
90{
92}
93
94uint16_t
96 const WifiTxParameters& txParams) const
97{
98 NS_LOG_FUNCTION(this << *mpdu << &txParams);
99
100 auto receiver = mpdu->GetHeader().GetAddr1();
101 auto origReceiver = mpdu->GetOriginal()->GetHeader().GetAddr1();
102
103 uint8_t tid = mpdu->GetHeader().GetQosTid();
104 Ptr<QosTxop> edca = m_mac->GetQosTxop(tid);
106 "An established Block Ack agreement is required");
107
108 uint16_t startingSeq = edca->GetBaStartingSequence(origReceiver, tid);
109 uint16_t maxDistFromStartingSeq =
110 (mpdu->GetHeader().GetSequenceNumber() - startingSeq + SEQNO_SPACE_SIZE) % SEQNO_SPACE_SIZE;
111 NS_ABORT_MSG_IF(maxDistFromStartingSeq >= SEQNO_SPACE_HALF_SIZE,
112 "The given QoS data frame is too old");
113
114 const WifiTxParameters::PsduInfo* psduInfo = txParams.GetPsduInfo(receiver);
115
116 if (!psduInfo || !psduInfo->seqNumbers.contains(tid))
117 {
118 // there are no aggregated MPDUs (so far)
119 return maxDistFromStartingSeq;
120 }
121
122 for (const auto& seqNumber : psduInfo->seqNumbers.at(tid))
123 {
124 if (!QosUtilsIsOldPacket(startingSeq, seqNumber))
125 {
126 uint16_t currDistToStartingSeq =
127 (seqNumber - startingSeq + SEQNO_SPACE_SIZE) % SEQNO_SPACE_SIZE;
128
129 if (currDistToStartingSeq > maxDistFromStartingSeq)
130 {
131 maxDistFromStartingSeq = currDistToStartingSeq;
132 }
133 }
134 }
135
136 NS_LOG_DEBUG("Returning " << maxDistFromStartingSeq);
137 return maxDistFromStartingSeq;
138}
139
140bool
142 const WifiTxParameters& txParams) const
143{
144 NS_LOG_FUNCTION(this << *mpdu << &txParams);
145
146 uint8_t tid = mpdu->GetHeader().GetQosTid();
147 Mac48Address receiver = mpdu->GetOriginal()->GetHeader().GetAddr1();
148 Ptr<QosTxop> edca = m_mac->GetQosTxop(tid);
149
150 // An immediate response (Ack or Block Ack) is needed if any of the following holds:
151 // * the maximum distance between the sequence number of an MPDU to transmit
152 // and the starting sequence number of the transmit window is greater than
153 // or equal to the window size multiplied by the BaThreshold
154 // * no other frame belonging to this BA agreement is queued (because, in such
155 // a case, a Block Ack is not going to be requested anytime soon)
156 // * this is the initial frame of a transmission opportunity and it is not
157 // protected by RTS/CTS (see Annex G.3 of IEEE 802.11-2016)
158 return !(
159 m_baThreshold > 0 &&
160 GetMaxDistFromStartingSeq(mpdu, txParams) <
161 m_baThreshold * edca->GetBaBufferSize(receiver, tid) &&
162 (edca->GetWifiMacQueue()->GetNPackets({WIFI_QOSDATA_QUEUE, WIFI_UNICAST, receiver, tid}) -
163 edca->GetBaManager()->GetNBufferedPackets(receiver, tid) >
164 1) &&
165 !(edca->GetTxopLimit(m_linkId).IsStrictlyPositive() &&
166 edca->GetRemainingTxop(m_linkId) == edca->GetTxopLimit(m_linkId) &&
167 !(txParams.m_protection && txParams.m_protection->method == WifiProtection::RTS_CTS)));
168}
169
170bool
172{
173 NS_ASSERT(mpdu->GetHeader().IsQosData());
174 auto tid = mpdu->GetHeader().GetQosTid();
175 NS_ASSERT(mpdu->IsQueued());
176 auto queue = m_mac->GetTxopQueue(mpdu->GetQueueAc());
177 auto origReceiver = mpdu->GetOriginal()->GetHeader().GetAddr1();
178 auto agreement = m_mac->GetBaAgreementEstablishedAsOriginator(origReceiver, tid);
179 NS_ASSERT(agreement);
180 auto mpduDist = agreement->get().GetDistance(mpdu->GetHeader().GetSequenceNumber());
181
182 Ptr<WifiMpdu> item = queue->PeekByTidAndAddress(tid, origReceiver);
183
184 while (item)
185 {
186 auto itemDist = agreement->get().GetDistance(item->GetHeader().GetSequenceNumber());
187 if (itemDist == mpduDist)
188 {
189 NS_LOG_DEBUG("No previous MPDU in-flight on the same link");
190 return false;
191 }
192 NS_ABORT_MSG_IF(itemDist > mpduDist,
193 "While searching for given MPDU ("
194 << *mpdu << "), found first another one (" << *item
195 << ") with higher sequence number");
196 if (auto linkIds = item->GetInFlightLinkIds(); linkIds.contains(m_linkId))
197 {
198 NS_LOG_DEBUG("Found MPDU inflight on the same link");
199 return true;
200 }
201 item = queue->PeekByTidAndAddress(tid, origReceiver, item);
202 }
203 NS_ABORT_MSG("Should not get here");
204 return false;
205}
206
207std::unique_ptr<WifiAcknowledgment>
209{
210 NS_LOG_FUNCTION(this << *mpdu << &txParams);
211
212 // If the TXVECTOR indicates a DL MU PPDU, call a separate method
213 if (txParams.m_txVector.IsDlMu())
214 {
215 switch (m_dlMuAckType)
216 {
218 return GetAckInfoIfBarBaSequence(mpdu, txParams);
220 return GetAckInfoIfTfMuBar(mpdu, txParams);
222 return GetAckInfoIfAggregatedMuBar(mpdu, txParams);
223 default:
224 NS_ABORT_MSG("Unknown DL acknowledgment method");
225 return nullptr;
226 }
227 }
228
229 const WifiMacHeader& hdr = mpdu->GetHeader();
230 Mac48Address receiver = hdr.GetAddr1();
231
232 // Acknowledgment for TB PPDUs
233 if (txParams.m_txVector.IsUlMu())
234 {
235 if (hdr.IsQosData() && !hdr.HasData())
236 {
237 // QoS Null frame
238 std::unique_ptr<WifiAcknowledgment> acknowledgment;
239
240 if (txParams.m_acknowledgment)
241 {
243 acknowledgment = txParams.m_acknowledgment->Copy();
244 }
245 else
246 {
247 acknowledgment = std::make_unique<WifiNoAck>();
248 }
249 acknowledgment->SetQosAckPolicy(receiver, hdr.GetQosTid(), WifiMacHeader::NO_ACK);
250 return acknowledgment;
251 }
252
253 if (txParams.m_acknowledgment)
254 {
256 return nullptr;
257 }
258
259 auto acknowledgment = std::make_unique<WifiAckAfterTbPpdu>();
260 if (hdr.IsQosData())
261 {
262 acknowledgment->SetQosAckPolicy(receiver, hdr.GetQosTid(), WifiMacHeader::NORMAL_ACK);
263 }
264 return acknowledgment;
265 }
266
267 // if this is a Trigger Frame, call a separate method
268 if (hdr.IsTrigger())
269 {
270 return TryUlMuTransmission(mpdu, txParams);
271 }
272
273 // if the current acknowledgment method (if any) is already BLOCK_ACK, it will not
274 // change by adding an MPDU
275 if (txParams.m_acknowledgment &&
277 {
278 return nullptr;
279 }
280
281 if (receiver.IsGroup())
282 {
283 NS_ABORT_MSG_IF(txParams.GetSize(receiver) > 0, "Unicast frames only can be aggregated");
284 auto acknowledgment = std::make_unique<WifiNoAck>();
285 if (hdr.IsQosData())
286 {
287 acknowledgment->SetQosAckPolicy(receiver, hdr.GetQosTid(), WifiMacHeader::NO_ACK);
288 }
289 return acknowledgment;
290 }
291
292 if ((!hdr.IsQosData() ||
294 !hdr.IsBlockAckReq())
295 {
297 "Non-QoS data frame or Block Ack agreement not established, request Normal Ack");
298 auto acknowledgment = std::make_unique<WifiNormalAck>();
299 acknowledgment->ackTxVector =
300 GetWifiRemoteStationManager()->GetAckTxVector(receiver, txParams.m_txVector);
301 if (hdr.IsQosData())
302 {
303 acknowledgment->SetQosAckPolicy(receiver, hdr.GetQosTid(), WifiMacHeader::NORMAL_ACK);
304 }
305 return acknowledgment;
306 }
307
308 // we get here if mpdu is a QoS data frame related to an established Block Ack agreement
309 // or mpdu is a BlockAckReq frame
310 if (!hdr.IsBlockAckReq() && !IsResponseNeeded(mpdu, txParams))
311 {
312 NS_LOG_DEBUG("A response is not needed: no ack for now, use Block Ack policy");
313 if (txParams.m_acknowledgment &&
314 txParams.m_acknowledgment->method == WifiAcknowledgment::NONE)
315 {
316 // no change if the ack method is already NONE
317 return nullptr;
318 }
319
320 auto acknowledgment = std::make_unique<WifiNoAck>();
321 if (hdr.IsQosData())
322 {
323 acknowledgment->SetQosAckPolicy(receiver, hdr.GetQosTid(), WifiMacHeader::BLOCK_ACK);
324 }
325 return acknowledgment;
326 }
327
328 // we get here if a response is needed
329 uint8_t tid = GetTid(mpdu->GetPacket(), hdr);
330 if (!hdr.IsBlockAckReq() && txParams.GetSize(receiver) == 0 && !ExistInflightOnSameLink(mpdu))
331 {
332 NS_LOG_DEBUG("Sending a single MPDU, no previous frame to ack: request Normal Ack");
333 auto acknowledgment = std::make_unique<WifiNormalAck>();
334 acknowledgment->ackTxVector =
335 GetWifiRemoteStationManager()->GetAckTxVector(receiver, txParams.m_txVector);
336 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::NORMAL_ACK);
337 return acknowledgment;
338 }
339
340 // we get here if multiple MPDUs are being/have been sent
341 if (!hdr.IsBlockAckReq() && (txParams.GetSize(receiver) == 0 || m_useExplicitBar))
342 {
343 // in case of single MPDU, there are previous unacknowledged frames, thus
344 // we cannot use Implicit Block Ack Request policy, otherwise we get a
345 // normal ack as response
346 NS_LOG_DEBUG("Request to schedule a Block Ack Request");
347
348 auto acknowledgment = std::make_unique<WifiBarBlockAck>();
349 acknowledgment->blockAckReqTxVector =
350 GetWifiRemoteStationManager()->GetBlockAckTxVector(receiver, txParams.m_txVector);
351 acknowledgment->blockAckTxVector = acknowledgment->blockAckReqTxVector;
352 acknowledgment->barType = m_mac->GetBarTypeAsOriginator(receiver, tid);
353 acknowledgment->baType = m_mac->GetBaTypeAsOriginator(receiver, tid);
354 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::BLOCK_ACK);
355 return acknowledgment;
356 }
357
359 "A-MPDU using Implicit Block Ack Request policy or BlockAckReq, request Block Ack");
360 auto acknowledgment = std::make_unique<WifiBlockAck>();
361 acknowledgment->blockAckTxVector =
362 GetWifiRemoteStationManager()->GetBlockAckTxVector(receiver, txParams.m_txVector);
363 acknowledgment->baType = m_mac->GetBaTypeAsOriginator(receiver, tid);
364 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::NORMAL_ACK);
365 return acknowledgment;
366}
367
368std::unique_ptr<WifiAcknowledgment>
370{
371 NS_LOG_FUNCTION(this << *msdu << &txParams);
372
373 // Aggregating an MSDU does not change the acknowledgment method
374 return nullptr;
375}
376
377std::unique_ptr<WifiAcknowledgment>
379 const WifiTxParameters& txParams)
380{
381 NS_LOG_FUNCTION(this << *mpdu << &txParams);
382 NS_ASSERT(txParams.m_txVector.IsDlMu());
384
385 const WifiMacHeader& hdr = mpdu->GetHeader();
386 Mac48Address receiver = hdr.GetAddr1();
387
388 const WifiTxParameters::PsduInfo* psduInfo = txParams.GetPsduInfo(receiver);
389
391 "QoS data frames only can be aggregated when transmitting a "
392 "DL MU PPDU acknowledged via a sequence of BAR and BA frames");
393 uint8_t tid = hdr.GetQosTid();
395
396 NS_ASSERT(!txParams.m_acknowledgment ||
398
399 WifiDlMuBarBaSequence* acknowledgment = nullptr;
400 if (txParams.m_acknowledgment)
401 {
402 acknowledgment = static_cast<WifiDlMuBarBaSequence*>(txParams.m_acknowledgment.get());
403 }
404
405 if (psduInfo)
406 {
407 // an MPDU addressed to the same receiver has been already added
408 NS_ASSERT(acknowledgment);
409
410 if (acknowledgment->stationsSendBlockAckReqTo.contains(receiver) ||
411 acknowledgment->stationsReplyingWithBlockAck.contains(receiver))
412 {
413 // the receiver either is already listed among the stations that will
414 // receive a BlockAckReq frame or is the station that will immediately
415 // respond with a BlockAck frame, hence no change is needed
416 return nullptr;
417 }
418
419 // the receiver was scheduled for responding immediately with a Normal Ack.
420 // Given that we are adding an MPDU, the receiver must be scheduled for
421 // responding immediately with a Block Ack
422 NS_ASSERT(acknowledgment->stationsReplyingWithNormalAck.size() == 1 &&
423 acknowledgment->stationsReplyingWithNormalAck.begin()->first == receiver);
424
425 // acknowledgment points to the m_acknowledgment member of txParams, which is
426 // passed as const reference because it must not be modified. Therefore, we
427 // make a copy of the object pointed to by acknowledgment and make changes to
428 // the copy
429 acknowledgment = new WifiDlMuBarBaSequence(*acknowledgment);
430 acknowledgment->stationsReplyingWithNormalAck.clear();
431
432 acknowledgment->stationsReplyingWithBlockAck.emplace(
433 receiver,
435 GetWifiRemoteStationManager()->GetBlockAckTxVector(receiver, txParams.m_txVector),
436 m_mac->GetBaTypeAsOriginator(receiver, tid)});
437 return std::unique_ptr<WifiDlMuBarBaSequence>(acknowledgment);
438 }
439
440 // we get here if this is the first MPDU for this receiver
441 auto htFem = DynamicCast<HtFrameExchangeManager>(m_mac->GetFrameExchangeManager(m_linkId));
442 NS_ASSERT(htFem);
443 if (auto bar = htFem->GetBar(QosUtilsMapTidToAc(tid), tid, receiver);
444 bar || (acknowledgment && (!acknowledgment->stationsReplyingWithNormalAck.empty() ||
445 !acknowledgment->stationsReplyingWithBlockAck.empty())))
446 {
447 // there is a pending BlockAckReq for this receiver or another receiver
448 // was selected for immediate response.
449 // Add this receiver to the list of stations receiving a BlockAckReq.
450 if (acknowledgment)
451 {
452 // txParams.m_acknowledgment points to an existing WifiDlMuBarBaSequence object.
453 // We have to return a copy of this object including the needed changes
454 acknowledgment = new WifiDlMuBarBaSequence(*acknowledgment);
455 }
456 else
457 {
458 // we have to create a new WifiDlMuBarBaSequence object
459 acknowledgment = new WifiDlMuBarBaSequence;
460 }
461
462 NS_LOG_DEBUG("Adding STA " << receiver
463 << " to the list of stations receiving a BlockAckReq");
464 acknowledgment->stationsSendBlockAckReqTo.emplace(
465 receiver,
467 GetWifiRemoteStationManager()->GetBlockAckTxVector(receiver, txParams.m_txVector),
468 m_mac->GetBarTypeAsOriginator(receiver, tid),
469 GetWifiRemoteStationManager()->GetBlockAckTxVector(receiver, txParams.m_txVector),
470 m_mac->GetBaTypeAsOriginator(receiver, tid)});
471
472 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::BLOCK_ACK);
473 return std::unique_ptr<WifiDlMuBarBaSequence>(acknowledgment);
474 }
475
476 // Add the receiver as the station that will immediately reply with a Normal Ack
477 if (acknowledgment)
478 {
479 // txParams.m_acknowledgment points to an existing WifiDlMuBarBaSequence object.
480 // We have to return a copy of this object including the needed changes
481 acknowledgment = new WifiDlMuBarBaSequence(*acknowledgment);
482 }
483 else
484 {
485 // we have to create a new WifiDlMuBarBaSequence object
486 acknowledgment = new WifiDlMuBarBaSequence;
487 }
488
489 NS_LOG_DEBUG("Adding STA " << receiver
490 << " as the station that will immediately reply with a Normal Ack");
491 acknowledgment->stationsReplyingWithNormalAck.emplace(
492 receiver,
494 GetWifiRemoteStationManager()->GetAckTxVector(receiver, txParams.m_txVector)});
495
496 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::NORMAL_ACK);
497 return std::unique_ptr<WifiDlMuBarBaSequence>(acknowledgment);
498}
499
500std::unique_ptr<WifiAcknowledgment>
502 const WifiTxParameters& txParams)
503{
504 NS_LOG_FUNCTION(this << *mpdu << &txParams);
505 NS_ASSERT(txParams.m_txVector.IsDlMu());
507
508 const WifiMacHeader& hdr = mpdu->GetHeader();
509 Mac48Address receiver = hdr.GetAddr1();
510
511 const WifiTxParameters::PsduInfo* psduInfo = txParams.GetPsduInfo(receiver);
512
513 NS_ASSERT(!txParams.m_acknowledgment ||
515
516 WifiDlMuTfMuBar* acknowledgment = nullptr;
517 if (txParams.m_acknowledgment)
518 {
519 acknowledgment = static_cast<WifiDlMuTfMuBar*>(txParams.m_acknowledgment.get());
520 }
521
522 if (!psduInfo)
523 {
524 // we get here if this is the first MPDU for this receiver.
525 Ptr<ApWifiMac> apMac = DynamicCast<ApWifiMac>(m_mac);
526 NS_ABORT_MSG_IF(!apMac, "HE APs only can send DL MU PPDUs");
527 uint16_t staId = apMac->GetAssociationId(receiver, m_linkId);
528
530 "QoS data frames only can be aggregated when transmitting a "
531 "DL MU PPDU acknowledged via a MU-BAR sent as SU frame");
532 uint8_t tid = hdr.GetQosTid();
533
534 // Add the receiver to the list of stations that will reply with a Block Ack
535 if (acknowledgment)
536 {
537 // txParams.m_acknowledgment points to an existing WifiDlMuTfMuBar object.
538 // We have to return a copy of this object including the needed changes
539 acknowledgment = new WifiDlMuTfMuBar(*acknowledgment);
540 }
541 else
542 {
543 // we have to create a new WifiDlMuTfMuBar object
544 acknowledgment = new WifiDlMuTfMuBar;
545 }
546
547 // determine the TX vector used to send the BlockAck frame
548 WifiTxVector blockAckTxVector;
549 auto preamble = IsEht(txParams.m_txVector.GetPreambleType()) ? WIFI_PREAMBLE_EHT_TB
551 blockAckTxVector.SetPreambleType(preamble);
552 blockAckTxVector.SetChannelWidth(txParams.m_txVector.GetChannelWidth());
553 // 800ns GI is not allowed for HE TB
554 blockAckTxVector.SetGuardInterval(
555 std::max<uint16_t>(txParams.m_txVector.GetGuardInterval(), 1600));
556 const auto& userInfo = txParams.m_txVector.GetHeMuUserInfo(staId);
557 blockAckTxVector.SetHeMuUserInfo(
558 staId,
559 {userInfo.ru, std::min(userInfo.mcs, m_maxMcsForBlockAckInTbPpdu), userInfo.nss});
560
561 NS_LOG_DEBUG("Adding STA "
562 << receiver
563 << " to the list of stations that will be solicited by the MU-BAR");
565 acknowledgment->stationsReplyingWithBlockAck.emplace(
566 receiver,
567 WifiDlMuTfMuBar::BlockAckInfo{edca->GetBaManager()->GetBlockAckReqHeader(
568 mpdu->GetOriginal()->GetHeader().GetAddr1(),
569 tid),
570 blockAckTxVector,
571 m_mac->GetBaTypeAsOriginator(receiver, tid)});
572
573 acknowledgment->barTypes.push_back(m_mac->GetBarTypeAsOriginator(receiver, tid));
574 acknowledgment->muBarTxVector =
575 GetWifiRemoteStationManager()->GetRtsTxVector(receiver,
576 txParams.m_txVector.GetChannelWidth());
577 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::BLOCK_ACK);
578 return std::unique_ptr<WifiDlMuTfMuBar>(acknowledgment);
579 }
580
581 // an MPDU addressed to the same receiver has been already added
582 NS_ASSERT(acknowledgment);
584 "QoS data frames only can be aggregated when transmitting a DL MU PPDU");
585
586 // no change is needed
587 return nullptr;
588}
589
590std::unique_ptr<WifiAcknowledgment>
592 const WifiTxParameters& txParams)
593{
594 NS_LOG_FUNCTION(this << *mpdu << &txParams);
595 NS_ASSERT(txParams.m_txVector.IsDlMu());
597
598 const WifiMacHeader& hdr = mpdu->GetHeader();
599 Mac48Address receiver = hdr.GetAddr1();
600
601 const WifiTxParameters::PsduInfo* psduInfo = txParams.GetPsduInfo(receiver);
602
603 NS_ASSERT(!txParams.m_acknowledgment ||
605
606 WifiDlMuAggregateTf* acknowledgment = nullptr;
607 if (txParams.m_acknowledgment)
608 {
609 acknowledgment = static_cast<WifiDlMuAggregateTf*>(txParams.m_acknowledgment.get());
610 }
611
612 if (!psduInfo)
613 {
614 // we get here if this is the first MPDU for this receiver.
615 Ptr<ApWifiMac> apMac = DynamicCast<ApWifiMac>(m_mac);
616 NS_ABORT_MSG_IF(!apMac, "HE APs only can send DL MU PPDUs");
617 uint16_t staId = apMac->GetAssociationId(receiver, m_linkId);
618
620 "QoS data frames only can be aggregated when transmitting a "
621 "DL MU PPDU acknowledged via a sequence of BAR and BA frames");
622 uint8_t tid = hdr.GetQosTid();
623
624 // Add the receiver to the list of stations that will reply with a Block Ack
625 if (acknowledgment)
626 {
627 // txParams.m_acknowledgment points to an existing WifiDlMuAggregateTf object.
628 // We have to return a copy of this object including the needed changes
629 acknowledgment = new WifiDlMuAggregateTf(*acknowledgment);
630 }
631 else
632 {
633 // we have to create a new WifiDlMuAggregateTf object
634 acknowledgment = new WifiDlMuAggregateTf;
635 }
636
637 // determine the TX vector used to send the BlockAck frame
638 WifiTxVector blockAckTxVector;
639 auto preamble = IsEht(txParams.m_txVector.GetPreambleType()) ? WIFI_PREAMBLE_EHT_TB
641 blockAckTxVector.SetPreambleType(preamble);
642 blockAckTxVector.SetChannelWidth(txParams.m_txVector.GetChannelWidth());
643 // 800ns GI is not allowed for HE TB
644 blockAckTxVector.SetGuardInterval(
645 std::max<uint16_t>(txParams.m_txVector.GetGuardInterval(), 1600));
646 const auto& userInfo = txParams.m_txVector.GetHeMuUserInfo(staId);
647 blockAckTxVector.SetHeMuUserInfo(
648 staId,
649 {userInfo.ru, std::min(userInfo.mcs, m_maxMcsForBlockAckInTbPpdu), userInfo.nss});
650
651 NS_LOG_DEBUG("Adding STA " << receiver
652 << " to the list of stations that will reply with a Block Ack");
654 acknowledgment->stationsReplyingWithBlockAck.emplace(
655 receiver,
657 GetMuBarSize({m_mac->GetBarTypeAsOriginator(receiver, tid)}),
658 edca->GetBaManager()->GetBlockAckReqHeader(
659 mpdu->GetOriginal()->GetHeader().GetAddr1(),
660 tid),
661 blockAckTxVector,
662 m_mac->GetBaTypeAsOriginator(receiver, tid)});
663
664 acknowledgment->SetQosAckPolicy(receiver, tid, WifiMacHeader::NO_EXPLICIT_ACK);
665 return std::unique_ptr<WifiDlMuAggregateTf>(acknowledgment);
666 }
667
668 // an MPDU addressed to the same receiver has been already added
669 NS_ASSERT(acknowledgment);
671 !hdr.IsQosData(),
672 "QoS data and MU-BAR Trigger frames only can be aggregated when transmitting a DL MU PPDU");
673
674 // no change is needed
675 return nullptr;
676}
677
678std::unique_ptr<WifiAcknowledgment>
680 const WifiTxParameters& txParams)
681{
682 NS_LOG_FUNCTION(this << *mpdu << &txParams);
683 NS_ASSERT(mpdu->GetHeader().IsTrigger());
684
685 Ptr<ApWifiMac> apMac = DynamicCast<ApWifiMac>(m_mac);
686 NS_ABORT_MSG_IF(!apMac, "HE APs only can send Trigger Frames");
687
688 auto heFem = DynamicCast<HeFrameExchangeManager>(m_mac->GetFrameExchangeManager(m_linkId));
689 NS_ABORT_MSG_IF(!heFem, "HE APs only can send Trigger Frames");
690
691 CtrlTriggerHeader trigger;
692 mpdu->GetPacket()->PeekHeader(trigger);
693
694 if (trigger.IsBasic())
695 {
696 // the only supported ack method for now is through a multi-STA BlockAck frame
697 auto acknowledgment = std::make_unique<WifiUlMuMultiStaBa>();
698
699 for (const auto& userInfo : trigger)
700 {
701 uint16_t aid12 = userInfo.GetAid12();
702
703 if (aid12 == NO_USER_STA_ID)
704 {
705 NS_LOG_INFO("Unallocated RU");
706 continue;
707 }
708 NS_ABORT_MSG_IF(aid12 == 0 || aid12 > 2007, "Allocation of RA-RUs is not supported");
709
710 const auto it = apMac->GetStaList(m_linkId).find(aid12);
711 NS_ASSERT(it != apMac->GetStaList(m_linkId).end());
712 const auto staAddress = it->second;
713
714 // find a TID for which a BA agreement exists with the given originator
715 uint8_t tid = 0;
716 while (tid < 8 && !m_mac->GetBaAgreementEstablishedAsRecipient(staAddress, tid))
717 {
718 tid++;
719 }
720 NS_ASSERT_MSG(tid < 8,
721 "No Block Ack agreement established with originator " << staAddress);
722
723 std::size_t index = acknowledgment->baType.m_bitmapLen.size();
724 acknowledgment->stationsReceivingMultiStaBa.emplace(std::make_pair(staAddress, tid),
725 index);
726
727 // we assume the Block Acknowledgment context is used for the multi-STA BlockAck frame
728 // (since it requires the longest TX time due to the presence of a bitmap)
729 acknowledgment->baType.m_bitmapLen.push_back(
730 m_mac->GetBaTypeAsRecipient(staAddress, tid).m_bitmapLen.at(0));
731 }
732
733 uint16_t staId = trigger.begin()->GetAid12();
734 acknowledgment->tbPpduTxVector = trigger.GetHeTbTxVector(staId);
735 acknowledgment->multiStaBaTxVector = GetWifiRemoteStationManager()->GetBlockAckTxVector(
736 apMac->GetStaList(m_linkId).find(staId)->second,
737 acknowledgment->tbPpduTxVector);
738 return acknowledgment;
739 }
740 else if (trigger.IsBsrp())
741 {
742 return std::make_unique<WifiNoAck>();
743 }
744
745 return nullptr;
746}
747
748} // namespace ns3
AttributeValue implementation for Boolean.
Definition: boolean.h:37
Headers for Trigger frames.
Definition: ctrl-headers.h:942
bool IsBasic() const
Check if this is a Basic Trigger frame.
bool IsBsrp() const
Check if this is a Buffer Status Report Poll Trigger frame.
This class can be used to hold variables of floating point type such as 'double' or 'float'.
Definition: double.h:42
Hold variables of type enum.
Definition: enum.h:62
an EUI-48 address
Definition: mac48-address.h:46
bool IsGroup() const
Smart pointer class similar to boost::intrusive_ptr.
Definition: ptr.h:77
a unique identifier for an interface.
Definition: type-id.h:59
TypeId SetParent(TypeId tid)
Set the parent TypeId.
Definition: type-id.cc:932
Hold an unsigned integer type.
Definition: uinteger.h:45
WifiAckManager is an abstract base class.
uint8_t m_linkId
ID of the link this Acknowledgment Manager is operating on.
Ptr< WifiMac > m_mac
MAC which is using this Acknowledgment Manager.
Ptr< WifiRemoteStationManager > GetWifiRemoteStationManager() const
WifiDefaultAckManager is the default ack manager.
bool IsResponseNeeded(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams) const
Determine whether the (A-)MPDU containing the given MPDU and the MPDUs (if any) included in the given...
static TypeId GetTypeId()
Get the type ID.
virtual std::unique_ptr< WifiAcknowledgment > GetAckInfoIfBarBaSequence(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams)
Compute the information about the acknowledgment of the current multi-user frame (as described by the...
bool m_useExplicitBar
true for sending BARs, false for using Implicit BAR policy
uint8_t m_maxMcsForBlockAckInTbPpdu
Max MCS used to send a BlockAck in a TB PPDU.
double m_baThreshold
Threshold to determine when a BlockAck must be requested.
virtual std::unique_ptr< WifiAcknowledgment > GetAckInfoIfTfMuBar(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams)
Compute the information about the acknowledgment of the current multi-user frame (as described by the...
WifiAcknowledgment::Method m_dlMuAckType
Type of the ack sequence for DL MU PPDUs.
std::unique_ptr< WifiAcknowledgment > TryAggregateMsdu(Ptr< const WifiMpdu > msdu, const WifiTxParameters &txParams) override
Determine the acknowledgment method to use if the given MSDU is aggregated to the current frame.
uint16_t GetMaxDistFromStartingSeq(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams) const
Get the maximum distance between the starting sequence number of the Block Ack agreement which the gi...
bool ExistInflightOnSameLink(Ptr< const WifiMpdu > mpdu) const
std::unique_ptr< WifiAcknowledgment > TryAddMpdu(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams) override
Determine the acknowledgment method to use if the given MPDU is added to the current frame.
virtual std::unique_ptr< WifiAcknowledgment > GetAckInfoIfAggregatedMuBar(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams)
Compute the information about the acknowledgment of the current multi-user frame (as described by the...
virtual std::unique_ptr< WifiAcknowledgment > TryUlMuTransmission(Ptr< const WifiMpdu > mpdu, const WifiTxParameters &txParams)
Calculate the acknowledgment method for the TB PPDUs solicited by the given Trigger Frame.
Implements the IEEE 802.11 MAC header.
uint8_t GetQosTid() const
Return the Traffic ID of a QoS header.
bool IsBlockAckReq() const
Return true if the header is a BlockAckRequest header.
Mac48Address GetAddr1() const
Return the address in the Address 1 field.
bool IsTrigger() const
Return true if the header is a Trigger header.
bool HasData() const
Return true if the header type is DATA and is not DATA_NULL.
bool IsQosData() const
Return true if the Type is DATA and Subtype is one of the possible values for QoS Data.
Ptr< FrameExchangeManager > GetFrameExchangeManager(uint8_t linkId=SINGLE_LINK_OP_ID) const
Get the Frame Exchange Manager associated with the given link.
Definition: wifi-mac.cc:864
BlockAckType GetBaTypeAsRecipient(Mac48Address originator, uint8_t tid) const
Definition: wifi-mac.cc:1719
BlockAckType GetBaTypeAsOriginator(const Mac48Address &recipient, uint8_t tid) const
Definition: wifi-mac.cc:1701
virtual Ptr< WifiMacQueue > GetTxopQueue(AcIndex ac) const
Get the wifi MAC queue of the (Qos)Txop associated with the given AC, if such (Qos)Txop is installed,...
Definition: wifi-mac.cc:545
BlockAckReqType GetBarTypeAsOriginator(const Mac48Address &recipient, uint8_t tid) const
Definition: wifi-mac.cc:1710
OriginatorAgreementOptConstRef GetBaAgreementEstablishedAsOriginator(Mac48Address recipient, uint8_t tid) const
Definition: wifi-mac.cc:1679
Ptr< QosTxop > GetQosTxop(AcIndex ac) const
Accessor for a specified EDCA object.
Definition: wifi-mac.cc:499
This class stores the TX parameters (TX vector, protection mechanism, acknowledgment mechanism,...
std::unique_ptr< WifiProtection > m_protection
protection method
uint32_t GetSize(Mac48Address receiver) const
Get the size in bytes of the (A-)MPDU addressed to the given receiver.
std::unique_ptr< WifiAcknowledgment > m_acknowledgment
acknowledgment method
const PsduInfo * GetPsduInfo(Mac48Address receiver) const
Get a pointer to the information about the PSDU addressed to the given receiver, if present,...
WifiTxVector m_txVector
TXVECTOR of the frame being prepared.
This class mimics the TXVECTOR which is to be passed to the PHY in order to define the parameters whi...
uint16_t GetGuardInterval() const
void SetChannelWidth(uint16_t channelWidth)
Sets the selected channelWidth (in MHz)
void SetGuardInterval(uint16_t guardInterval)
Sets the guard interval duration (in nanoseconds)
void SetHeMuUserInfo(uint16_t staId, HeMuUserInfo userInfo)
Set the HE MU user-specific transmission information for the given STA-ID.
WifiPreamble GetPreambleType() const
HeMuUserInfo GetHeMuUserInfo(uint16_t staId) const
Get the HE MU user-specific transmission information for the given STA-ID.
bool IsDlMu() const
bool IsUlMu() const
uint16_t GetChannelWidth() const
void SetPreambleType(WifiPreamble preamble)
Sets the preamble type.
#define NS_ASSERT(condition)
At runtime, in debugging builds, if this condition is not true, the program prints the source file,...
Definition: assert.h:66
#define NS_ASSERT_MSG(condition, message)
At runtime, in debugging builds, if this condition is not true, the program prints the message to out...
Definition: assert.h:86
Ptr< const AttributeAccessor > MakeBooleanAccessor(T1 a1)
Definition: boolean.h:81
Ptr< const AttributeChecker > MakeBooleanChecker()
Definition: boolean.cc:124
Ptr< const AttributeAccessor > MakeDoubleAccessor(T1 a1)
Definition: double.h:43
Ptr< const AttributeAccessor > MakeUintegerAccessor(T1 a1)
Definition: uinteger.h:46
#define NS_ABORT_MSG(msg)
Unconditional abnormal program termination with a message.
Definition: abort.h:49
#define NS_ABORT_MSG_IF(cond, msg)
Abnormal program termination if a condition is true, with a message.
Definition: abort.h:108
#define NS_LOG_COMPONENT_DEFINE(name)
Define a Log component with a specific name.
Definition: log.h:202
#define NS_LOG_DEBUG(msg)
Use NS_LOG to output a message of level LOG_DEBUG.
Definition: log.h:268
#define NS_LOG_FUNCTION_NOARGS()
Output the name of the function.
#define NS_LOG_FUNCTION(parameters)
If log level LOG_FUNCTION is enabled, this macro will output all input parameters separated by ",...
#define NS_LOG_INFO(msg)
Use NS_LOG to output a message of level LOG_INFO.
Definition: log.h:275
#define NS_OBJECT_ENSURE_REGISTERED(type)
Register an Object subclass with the TypeId system.
Definition: object-base.h:46
AcIndex QosUtilsMapTidToAc(uint8_t tid)
Maps TID (Traffic ID) to Access classes.
Definition: qos-utils.cc:134
bool QosUtilsIsOldPacket(uint16_t startingSeq, uint16_t seqNumber)
This function checks if packet with sequence number seqNumber is an "old" packet.
Definition: qos-utils.cc:182
uint8_t GetTid(Ptr< const Packet > packet, const WifiMacHeader hdr)
This function is useful to get traffic id of different packet types.
Definition: qos-utils.cc:191
@ WIFI_PREAMBLE_EHT_TB
@ WIFI_PREAMBLE_HE_TB
Every class exported by the ns3 library is enclosed in the ns3 namespace.
static constexpr uint16_t SEQNO_SPACE_HALF_SIZE
Size of the half the space of sequence numbers (used to determine old packets)
Definition: wifi-utils.h:188
bool IsEht(WifiPreamble preamble)
Return true if a preamble corresponds to an EHT transmission.
Ptr< const AttributeChecker > MakeEnumChecker(T v, std::string n, Ts... args)
Make an EnumChecker pre-configured with a set of allowed values by name.
Definition: enum.h:189
static constexpr uint16_t NO_USER_STA_ID
STA_ID for a RU that is intended for no user (Section 26.11.1 802.11ax-2021)
uint32_t GetMuBarSize(std::list< BlockAckReqType > types)
Return the total MU-BAR size (including FCS trailer).
Definition: wifi-utils.cc:86
static constexpr uint16_t SEQNO_SPACE_SIZE
Size of the space of sequence numbers.
Definition: wifi-utils.h:185
std::vector< uint8_t > m_bitmapLen
Length (bytes) of included bitmaps.
void SetQosAckPolicy(Mac48Address receiver, uint8_t tid, WifiMacHeader::QosAckPolicy ackPolicy)
Set the QoS Ack policy to use for the MPDUs addressed to the given receiver and belonging to the give...
information related to a BlockAck frame sent by a station
WifiDlMuAggregateTf specifies that a DL MU PPDU made of PSDUs including each a MU-BAR Trigger Frame i...
std::map< Mac48Address, BlockAckInfo > stationsReplyingWithBlockAck
Set of stations replying with a BlockAck frame.
information related to an Ack frame sent by a station
information related to a BlockAck frame sent by a station
information related to a BlockAckReq frame sent to a station
WifiDlMuBarBaSequence specifies that a DL MU PPDU is acknowledged through a sequence of BlockAckReq a...
std::map< Mac48Address, BlockAckReqInfo > stationsSendBlockAckReqTo
Set of stations receiving a BlockAckReq frame and replying with a BlockAck frame.
std::map< Mac48Address, BlockAckInfo > stationsReplyingWithBlockAck
Set of stations replying with a BlockAck frame (no more than one)
std::map< Mac48Address, AckInfo > stationsReplyingWithNormalAck
Set of stations replying with an Ack frame (no more than one)
information related to a BlockAck frame sent by a station
WifiDlMuTfMuBar specifies that a DL MU PPDU is followed after a SIFS duration by a MU-BAR Trigger Fra...
std::list< BlockAckReqType > barTypes
BAR types.
std::map< Mac48Address, BlockAckInfo > stationsReplyingWithBlockAck
Set of stations replying with a BlockAck frame.
WifiTxVector muBarTxVector
TXVECTOR used to transmit the MU-BAR Trigger Frame.
information about the frame being prepared for a specific receiver
std::map< uint8_t, std::set< uint16_t > > seqNumbers
set of the sequence numbers of the MPDUs added for each TID