]> git.proxmox.com Git - mirror_edk2.git/blame - NetworkPkg/Ip6Dxe/Ip6Mld.c
UefiCpuPkg: Move AsmRelocateApLoopStart from Mpfuncs.nasm to AmdSev.nasm
[mirror_edk2.git] / NetworkPkg / Ip6Dxe / Ip6Mld.c
CommitLineData
a3bcde70
HT
1/** @file\r
2 Multicast Listener Discovery support routines.\r
3\r
4 Copyright (c) 2009 - 2010, Intel Corporation. All rights reserved.<BR>\r
5\r
ecf98fbc 6 SPDX-License-Identifier: BSD-2-Clause-Patent\r
a3bcde70
HT
7\r
8**/\r
9\r
10#include "Ip6Impl.h"\r
11\r
12/**\r
13 Create a IP6_MLD_GROUP list entry node and record to IP6 service binding data.\r
14\r
15 @param[in, out] IpSb Points to IP6 service binding instance.\r
16 @param[in] MulticastAddr The IPv6 multicast address to be recorded.\r
17 @param[in] DelayTimer The maximum allowed delay before sending a responding\r
18 report, in units of milliseconds.\r
19 @return The created IP6_ML_GROUP list entry or NULL.\r
20\r
21**/\r
22IP6_MLD_GROUP *\r
23Ip6CreateMldEntry (\r
d1050b9d
MK
24 IN OUT IP6_SERVICE *IpSb,\r
25 IN EFI_IPv6_ADDRESS *MulticastAddr,\r
26 IN UINT32 DelayTimer\r
a3bcde70
HT
27 )\r
28{\r
d1050b9d 29 IP6_MLD_GROUP *Entry;\r
a3bcde70
HT
30\r
31 NET_CHECK_SIGNATURE (IpSb, IP6_SERVICE_SIGNATURE);\r
32 ASSERT (MulticastAddr != NULL && IP6_IS_MULTICAST (MulticastAddr));\r
33\r
34 Entry = AllocatePool (sizeof (IP6_MLD_GROUP));\r
35 if (Entry != NULL) {\r
36 Entry->RefCnt = 1;\r
37 Entry->DelayTimer = DelayTimer;\r
38 Entry->SendByUs = FALSE;\r
39 IP6_COPY_ADDRESS (&Entry->Address, MulticastAddr);\r
40 InsertTailList (&IpSb->MldCtrl.Groups, &Entry->Link);\r
41 }\r
42\r
43 return Entry;\r
44}\r
45\r
46/**\r
47 Search a IP6_MLD_GROUP list entry node from a list array.\r
48\r
49 @param[in] IpSb Points to IP6 service binding instance.\r
50 @param[in] MulticastAddr The IPv6 multicast address to be searched.\r
51\r
52 @return The found IP6_ML_GROUP list entry or NULL.\r
53\r
54**/\r
55IP6_MLD_GROUP *\r
56Ip6FindMldEntry (\r
d1050b9d
MK
57 IN IP6_SERVICE *IpSb,\r
58 IN EFI_IPv6_ADDRESS *MulticastAddr\r
a3bcde70
HT
59 )\r
60{\r
d1050b9d
MK
61 LIST_ENTRY *Entry;\r
62 IP6_MLD_GROUP *Group;\r
a3bcde70
HT
63\r
64 NET_CHECK_SIGNATURE (IpSb, IP6_SERVICE_SIGNATURE);\r
65 ASSERT (MulticastAddr != NULL && IP6_IS_MULTICAST (MulticastAddr));\r
66\r
67 NET_LIST_FOR_EACH (Entry, &IpSb->MldCtrl.Groups) {\r
68 Group = NET_LIST_USER_STRUCT (Entry, IP6_MLD_GROUP, Link);\r
69 if (EFI_IP6_EQUAL (MulticastAddr, &Group->Address)) {\r
70 return Group;\r
71 }\r
72 }\r
73\r
74 return NULL;\r
75}\r
76\r
77/**\r
78 Count the number of IP6 multicast groups that are mapped to the\r
79 same MAC address. Several IP6 multicast address may be mapped to\r
80 the same MAC address.\r
81\r
82 @param[in] MldCtrl The MLD control block to search in.\r
83 @param[in] Mac The MAC address to search.\r
84\r
85 @return The number of the IP6 multicast group that mapped to the same\r
86 multicast group Mac.\r
87\r
88**/\r
89INTN\r
90Ip6FindMac (\r
d1050b9d
MK
91 IN IP6_MLD_SERVICE_DATA *MldCtrl,\r
92 IN EFI_MAC_ADDRESS *Mac\r
a3bcde70
HT
93 )\r
94{\r
d1050b9d
MK
95 LIST_ENTRY *Entry;\r
96 IP6_MLD_GROUP *Group;\r
97 INTN Count;\r
a3bcde70
HT
98\r
99 Count = 0;\r
100\r
101 NET_LIST_FOR_EACH (Entry, &MldCtrl->Groups) {\r
102 Group = NET_LIST_USER_STRUCT (Entry, IP6_MLD_GROUP, Link);\r
103\r
104 if (NET_MAC_EQUAL (&Group->Mac, Mac, sizeof (EFI_MAC_ADDRESS))) {\r
105 Count++;\r
106 }\r
107 }\r
108\r
109 return Count;\r
110}\r
111\r
112/**\r
113 Generate MLD report message and send it out to MulticastAddr.\r
114\r
115 @param[in] IpSb The IP service to send the packet.\r
116 @param[in] Interface The IP interface to send the packet.\r
117 If NULL, a system interface will be selected.\r
118 @param[in] MulticastAddr The specific IPv6 multicast address to which\r
119 the message sender is listening.\r
120\r
121 @retval EFI_OUT_OF_RESOURCES There are not sufficient resources to complete the\r
122 operation.\r
123 @retval EFI_SUCCESS The MLD report message was successfully sent out.\r
124\r
125**/\r
126EFI_STATUS\r
127Ip6SendMldReport (\r
d1050b9d
MK
128 IN IP6_SERVICE *IpSb,\r
129 IN IP6_INTERFACE *Interface OPTIONAL,\r
130 IN EFI_IPv6_ADDRESS *MulticastAddr\r
a3bcde70
HT
131 )\r
132{\r
d1050b9d
MK
133 IP6_MLD_HEAD *MldHead;\r
134 NET_BUF *Packet;\r
135 EFI_IP6_HEADER Head;\r
136 UINT16 PayloadLen;\r
137 UINTN OptionLen;\r
138 UINT8 *Options;\r
139 EFI_STATUS Status;\r
140 UINT16 HeadChecksum;\r
141 UINT16 PseudoChecksum;\r
a3bcde70
HT
142\r
143 NET_CHECK_SIGNATURE (IpSb, IP6_SERVICE_SIGNATURE);\r
144 ASSERT (MulticastAddr != NULL && IP6_IS_MULTICAST (MulticastAddr));\r
145\r
146 //\r
147 // Generate the packet to be sent\r
148 // IPv6 basic header + Hop by Hop option + MLD message\r
149 //\r
150\r
151 OptionLen = 0;\r
d1050b9d 152 Status = Ip6FillHopByHop (NULL, &OptionLen, IP6_ICMP);\r
a3bcde70
HT
153 ASSERT (Status == EFI_BUFFER_TOO_SMALL);\r
154\r
d1050b9d
MK
155 PayloadLen = (UINT16)(OptionLen + sizeof (IP6_MLD_HEAD));\r
156 Packet = NetbufAlloc (sizeof (EFI_IP6_HEADER) + (UINT32)PayloadLen);\r
a3bcde70
HT
157 if (Packet == NULL) {\r
158 return EFI_OUT_OF_RESOURCES;\r
159 }\r
160\r
161 //\r
162 // Create the basic IPv6 header.\r
163 // RFC3590: Use link-local address as source address if it is available,\r
164 // otherwise use the unspecified address.\r
165 //\r
d1050b9d
MK
166 Head.FlowLabelL = 0;\r
167 Head.FlowLabelH = 0;\r
168 Head.PayloadLength = HTONS (PayloadLen);\r
169 Head.NextHeader = IP6_HOP_BY_HOP;\r
170 Head.HopLimit = 1;\r
a3bcde70
HT
171 IP6_COPY_ADDRESS (&Head.DestinationAddress, MulticastAddr);\r
172\r
173 //\r
174 // If Link-Local address is not ready, we use unspecified address.\r
175 //\r
176 IP6_COPY_ADDRESS (&Head.SourceAddress, &IpSb->LinkLocalAddr);\r
177\r
178 NetbufReserve (Packet, sizeof (EFI_IP6_HEADER));\r
179\r
180 //\r
181 // Fill a IPv6 Router Alert option in a Hop-by-Hop Options Header\r
182 //\r
d1050b9d 183 Options = NetbufAllocSpace (Packet, (UINT32)OptionLen, FALSE);\r
a3bcde70
HT
184 ASSERT (Options != NULL);\r
185 Status = Ip6FillHopByHop (Options, &OptionLen, IP6_ICMP);\r
186 if (EFI_ERROR (Status)) {\r
187 NetbufFree (Packet);\r
188 Packet = NULL;\r
189 return Status;\r
190 }\r
191\r
192 //\r
193 // Fill in MLD message - Report\r
194 //\r
d1050b9d 195 MldHead = (IP6_MLD_HEAD *)NetbufAllocSpace (Packet, sizeof (IP6_MLD_HEAD), FALSE);\r
a3bcde70
HT
196 ASSERT (MldHead != NULL);\r
197 ZeroMem (MldHead, sizeof (IP6_MLD_HEAD));\r
198 MldHead->Head.Type = ICMP_V6_LISTENER_REPORT;\r
199 MldHead->Head.Code = 0;\r
200 IP6_COPY_ADDRESS (&MldHead->Group, MulticastAddr);\r
201\r
d1050b9d 202 HeadChecksum = NetblockChecksum ((UINT8 *)MldHead, sizeof (IP6_MLD_HEAD));\r
a3bcde70
HT
203 PseudoChecksum = NetIp6PseudoHeadChecksum (\r
204 &Head.SourceAddress,\r
205 &Head.DestinationAddress,\r
206 IP6_ICMP,\r
207 sizeof (IP6_MLD_HEAD)\r
208 );\r
209\r
210 MldHead->Head.Checksum = (UINT16) ~NetAddChecksum (HeadChecksum, PseudoChecksum);\r
211\r
212 //\r
213 // Transmit the packet\r
214 //\r
215 return Ip6Output (IpSb, Interface, NULL, Packet, &Head, NULL, 0, Ip6SysPacketSent, NULL);\r
216}\r
217\r
218/**\r
219 Generate MLD Done message and send it out to MulticastAddr.\r
220\r
221 @param[in] IpSb The IP service to send the packet.\r
222 @param[in] MulticastAddr The specific IPv6 multicast address to which\r
223 the message sender is ceasing to listen.\r
224\r
225 @retval EFI_OUT_OF_RESOURCES There are not sufficient resources to complete the\r
226 operation.\r
227 @retval EFI_SUCCESS The MLD report message was successfully sent out.\r
228\r
229**/\r
230EFI_STATUS\r
231Ip6SendMldDone (\r
d1050b9d
MK
232 IN IP6_SERVICE *IpSb,\r
233 IN EFI_IPv6_ADDRESS *MulticastAddr\r
a3bcde70
HT
234 )\r
235{\r
d1050b9d
MK
236 IP6_MLD_HEAD *MldHead;\r
237 NET_BUF *Packet;\r
238 EFI_IP6_HEADER Head;\r
239 UINT16 PayloadLen;\r
240 UINTN OptionLen;\r
241 UINT8 *Options;\r
242 EFI_STATUS Status;\r
243 EFI_IPv6_ADDRESS Destination;\r
244 UINT16 HeadChecksum;\r
245 UINT16 PseudoChecksum;\r
a3bcde70
HT
246\r
247 NET_CHECK_SIGNATURE (IpSb, IP6_SERVICE_SIGNATURE);\r
248 ASSERT (MulticastAddr != NULL && IP6_IS_MULTICAST (MulticastAddr));\r
249\r
250 //\r
251 // Generate the packet to be sent\r
252 // IPv6 basic header + Hop by Hop option + MLD message\r
253 //\r
254\r
255 OptionLen = 0;\r
d1050b9d 256 Status = Ip6FillHopByHop (NULL, &OptionLen, IP6_ICMP);\r
a3bcde70
HT
257 ASSERT (Status == EFI_BUFFER_TOO_SMALL);\r
258\r
d1050b9d
MK
259 PayloadLen = (UINT16)(OptionLen + sizeof (IP6_MLD_HEAD));\r
260 Packet = NetbufAlloc (sizeof (EFI_IP6_HEADER) + (UINT32)PayloadLen);\r
a3bcde70
HT
261 if (Packet == NULL) {\r
262 return EFI_OUT_OF_RESOURCES;\r
263 }\r
264\r
265 //\r
266 // Create the basic IPv6 header.\r
267 //\r
d1050b9d
MK
268 Head.FlowLabelL = 0;\r
269 Head.FlowLabelH = 0;\r
270 Head.PayloadLength = HTONS (PayloadLen);\r
271 Head.NextHeader = IP6_HOP_BY_HOP;\r
272 Head.HopLimit = 1;\r
a3bcde70
HT
273\r
274 //\r
275 // If Link-Local address is not ready, we use unspecified address.\r
276 //\r
277 IP6_COPY_ADDRESS (&Head.SourceAddress, &IpSb->LinkLocalAddr);\r
278\r
279 Ip6SetToAllNodeMulticast (TRUE, IP6_LINK_LOCAL_SCOPE, &Destination);\r
280 IP6_COPY_ADDRESS (&Head.DestinationAddress, &Destination);\r
281\r
282 NetbufReserve (Packet, sizeof (EFI_IP6_HEADER));\r
283\r
284 //\r
285 // Fill a IPv6 Router Alert option in a Hop-by-Hop Options Header\r
286 //\r
d1050b9d 287 Options = NetbufAllocSpace (Packet, (UINT32)OptionLen, FALSE);\r
a3bcde70
HT
288 ASSERT (Options != NULL);\r
289 Status = Ip6FillHopByHop (Options, &OptionLen, IP6_ICMP);\r
290 if (EFI_ERROR (Status)) {\r
291 NetbufFree (Packet);\r
292 Packet = NULL;\r
293 return Status;\r
294 }\r
295\r
296 //\r
297 // Fill in MLD message - Done\r
298 //\r
d1050b9d 299 MldHead = (IP6_MLD_HEAD *)NetbufAllocSpace (Packet, sizeof (IP6_MLD_HEAD), FALSE);\r
a3bcde70
HT
300 ASSERT (MldHead != NULL);\r
301 ZeroMem (MldHead, sizeof (IP6_MLD_HEAD));\r
302 MldHead->Head.Type = ICMP_V6_LISTENER_DONE;\r
303 MldHead->Head.Code = 0;\r
304 IP6_COPY_ADDRESS (&MldHead->Group, MulticastAddr);\r
305\r
d1050b9d 306 HeadChecksum = NetblockChecksum ((UINT8 *)MldHead, sizeof (IP6_MLD_HEAD));\r
a3bcde70
HT
307 PseudoChecksum = NetIp6PseudoHeadChecksum (\r
308 &Head.SourceAddress,\r
309 &Head.DestinationAddress,\r
310 IP6_ICMP,\r
311 sizeof (IP6_MLD_HEAD)\r
312 );\r
313\r
314 MldHead->Head.Checksum = (UINT16) ~NetAddChecksum (HeadChecksum, PseudoChecksum);\r
315\r
316 //\r
317 // Transmit the packet\r
318 //\r
319 return Ip6Output (IpSb, NULL, NULL, Packet, &Head, NULL, 0, Ip6SysPacketSent, NULL);\r
320}\r
321\r
322/**\r
323 Init the MLD data of the IP6 service instance. Configure\r
324 MNP to receive ALL SYSTEM multicast.\r
325\r
326 @param[in] IpSb The IP6 service whose MLD is to be initialized.\r
327\r
328 @retval EFI_OUT_OF_RESOURCES There are not sufficient resourcet to complete the\r
329 operation.\r
330 @retval EFI_SUCCESS The MLD module successfully initialized.\r
331\r
332**/\r
333EFI_STATUS\r
334Ip6InitMld (\r
d1050b9d 335 IN IP6_SERVICE *IpSb\r
a3bcde70
HT
336 )\r
337{\r
d1050b9d
MK
338 EFI_IPv6_ADDRESS AllNodes;\r
339 IP6_MLD_GROUP *Group;\r
340 EFI_STATUS Status;\r
a3bcde70
HT
341\r
342 //\r
343 // Join the link-scope all-nodes multicast address (FF02::1).\r
344 // This address is started in Idle Listener state and never transitions to\r
345 // another state, and never sends a Report or Done for that address.\r
346 //\r
347\r
348 Ip6SetToAllNodeMulticast (FALSE, IP6_LINK_LOCAL_SCOPE, &AllNodes);\r
349\r
d1050b9d 350 Group = Ip6CreateMldEntry (IpSb, &AllNodes, (UINT32)IP6_INFINIT_LIFETIME);\r
a3bcde70
HT
351 if (Group == NULL) {\r
352 return EFI_OUT_OF_RESOURCES;\r
353 }\r
354\r
355 Status = Ip6GetMulticastMac (IpSb->Mnp, &AllNodes, &Group->Mac);\r
356 if (EFI_ERROR (Status)) {\r
357 goto ERROR;\r
358 }\r
359\r
360 //\r
361 // Configure MNP to receive all-nodes multicast\r
362 //\r
363 Status = IpSb->Mnp->Groups (IpSb->Mnp, TRUE, &Group->Mac);\r
364 if (EFI_ERROR (Status) && (Status != EFI_ALREADY_STARTED)) {\r
365 goto ERROR;\r
366 }\r
367\r
368 return EFI_SUCCESS;\r
369\r
370ERROR:\r
371 RemoveEntryList (&Group->Link);\r
372 FreePool (Group);\r
373 return Status;\r
374}\r
375\r
376/**\r
377 Add a group address to the array of group addresses.\r
378 The caller should make sure that no duplicated address\r
379 existed in the array.\r
380\r
381 @param[in, out] IpInstance Points to an IP6_PROTOCOL instance.\r
382 @param[in] Group The IP6 multicast address to add.\r
383\r
384 @retval EFI_OUT_OF_RESOURCES There are not sufficient resources to complete\r
385 the operation.\r
7de8045a 386 @retval EFI_SUCCESS The address is added to the group address array.\r
a3bcde70
HT
387\r
388**/\r
389EFI_STATUS\r
390Ip6CombineGroups (\r
d1050b9d
MK
391 IN OUT IP6_PROTOCOL *IpInstance,\r
392 IN EFI_IPv6_ADDRESS *Group\r
a3bcde70
HT
393 )\r
394{\r
d1050b9d 395 EFI_IPv6_ADDRESS *GroupList;\r
a3bcde70
HT
396\r
397 NET_CHECK_SIGNATURE (IpInstance, IP6_PROTOCOL_SIGNATURE);\r
398 ASSERT (Group != NULL && IP6_IS_MULTICAST (Group));\r
399\r
400 IpInstance->GroupCount++;\r
401\r
402 GroupList = AllocatePool (IpInstance->GroupCount * sizeof (EFI_IPv6_ADDRESS));\r
403 if (GroupList == NULL) {\r
404 return EFI_OUT_OF_RESOURCES;\r
405 }\r
406\r
407 if (IpInstance->GroupCount > 1) {\r
408 ASSERT (IpInstance->GroupList != NULL);\r
409\r
410 CopyMem (\r
411 GroupList,\r
412 IpInstance->GroupList,\r
413 (IpInstance->GroupCount - 1) * sizeof (EFI_IPv6_ADDRESS)\r
414 );\r
415\r
416 FreePool (IpInstance->GroupList);\r
417 }\r
418\r
419 IP6_COPY_ADDRESS (GroupList + (IpInstance->GroupCount - 1), Group);\r
420\r
421 IpInstance->GroupList = GroupList;\r
422\r
423 return EFI_SUCCESS;\r
424}\r
425\r
426/**\r
427 Remove a group address from the array of group addresses.\r
428 Although the function doesn't assume the byte order of Group,\r
429 the network byte order is used by the caller.\r
430\r
431 @param[in, out] IpInstance Points to an IP6_PROTOCOL instance.\r
432 @param[in] Group The IP6 multicast address to remove.\r
433\r
434 @retval EFI_NOT_FOUND Cannot find the to be removed group address.\r
435 @retval EFI_SUCCESS The group address was successfully removed.\r
436\r
437**/\r
438EFI_STATUS\r
439Ip6RemoveGroup (\r
d1050b9d
MK
440 IN OUT IP6_PROTOCOL *IpInstance,\r
441 IN EFI_IPv6_ADDRESS *Group\r
a3bcde70
HT
442 )\r
443{\r
d1050b9d
MK
444 UINT32 Index;\r
445 UINT32 Count;\r
a3bcde70
HT
446\r
447 Count = IpInstance->GroupCount;\r
448\r
449 for (Index = 0; Index < Count; Index++) {\r
450 if (EFI_IP6_EQUAL (IpInstance->GroupList + Index, Group)) {\r
451 break;\r
452 }\r
453 }\r
454\r
455 if (Index == Count) {\r
456 return EFI_NOT_FOUND;\r
457 }\r
458\r
459 while (Index < Count - 1) {\r
460 IP6_COPY_ADDRESS (IpInstance->GroupList + Index, IpInstance->GroupList + Index + 1);\r
461 Index++;\r
462 }\r
463\r
464 ASSERT (IpInstance->GroupCount > 0);\r
465 IpInstance->GroupCount--;\r
466\r
467 return EFI_SUCCESS;\r
468}\r
469\r
470/**\r
471 Join the multicast group on behalf of this IP6 service binding instance.\r
472\r
473 @param[in] IpSb The IP6 service binding instance.\r
474 @param[in] Interface Points to an IP6_INTERFACE structure.\r
475 @param[in] Address The group address to join.\r
476\r
477 @retval EFI_SUCCESS Successfully join the multicast group.\r
478 @retval EFI_OUT_OF_RESOURCES Failed to allocate resources.\r
479 @retval Others Failed to join the multicast group.\r
480\r
481**/\r
482EFI_STATUS\r
483Ip6JoinGroup (\r
d1050b9d
MK
484 IN IP6_SERVICE *IpSb,\r
485 IN IP6_INTERFACE *Interface,\r
486 IN EFI_IPv6_ADDRESS *Address\r
a3bcde70
HT
487 )\r
488{\r
d1050b9d
MK
489 IP6_MLD_GROUP *Group;\r
490 EFI_STATUS Status;\r
a3bcde70
HT
491\r
492 Group = Ip6FindMldEntry (IpSb, Address);\r
493 if (Group != NULL) {\r
494 Group->RefCnt++;\r
495 return EFI_SUCCESS;\r
496 }\r
497\r
498 //\r
7de8045a
AC
499 // Repeat the report once or twice after short delays [Unsolicited Report Interval] (default:10s)\r
500 // Simulate this operation as a Multicast-Address-Specific Query was received for that address.\r
a3bcde70
HT
501 //\r
502 Group = Ip6CreateMldEntry (IpSb, Address, IP6_UNSOLICITED_REPORT_INTERVAL);\r
503 if (Group == NULL) {\r
504 return EFI_OUT_OF_RESOURCES;\r
505 }\r
506\r
507 Group->SendByUs = TRUE;\r
508\r
509 Status = Ip6GetMulticastMac (IpSb->Mnp, Address, &Group->Mac);\r
510 if (EFI_ERROR (Status)) {\r
511 return Status;\r
512 }\r
513\r
514 Status = IpSb->Mnp->Groups (IpSb->Mnp, TRUE, &Group->Mac);\r
515 if (EFI_ERROR (Status) && (Status != EFI_ALREADY_STARTED)) {\r
516 goto ERROR;\r
517 }\r
518\r
519 //\r
520 // Send unsolicited report when a node starts listening to a multicast address\r
521 //\r
522 Status = Ip6SendMldReport (IpSb, Interface, Address);\r
523 if (EFI_ERROR (Status)) {\r
524 goto ERROR;\r
525 }\r
526\r
527 return EFI_SUCCESS;\r
528\r
529ERROR:\r
530 RemoveEntryList (&Group->Link);\r
531 FreePool (Group);\r
532 return Status;\r
533}\r
534\r
535/**\r
536 Leave the IP6 multicast group.\r
537\r
538 @param[in] IpSb The IP6 service binding instance.\r
539 @param[in] Address The group address to leave.\r
540\r
541 @retval EFI_NOT_FOUND The IP6 service instance isn't in the group.\r
542 @retval EFI_SUCCESS Successfully leave the multicast group..\r
543 @retval Others Failed to leave the multicast group.\r
544\r
545**/\r
546EFI_STATUS\r
547Ip6LeaveGroup (\r
d1050b9d
MK
548 IN IP6_SERVICE *IpSb,\r
549 IN EFI_IPv6_ADDRESS *Address\r
a3bcde70
HT
550 )\r
551{\r
d1050b9d
MK
552 IP6_MLD_GROUP *Group;\r
553 EFI_STATUS Status;\r
a3bcde70
HT
554\r
555 Group = Ip6FindMldEntry (IpSb, Address);\r
556 if (Group == NULL) {\r
557 return EFI_NOT_FOUND;\r
558 }\r
559\r
560 //\r
561 // If more than one instance is in the group, decrease\r
562 // the RefCnt then return.\r
563 //\r
564 if ((Group->RefCnt > 0) && (--Group->RefCnt > 0)) {\r
565 return EFI_SUCCESS;\r
566 }\r
567\r
568 //\r
569 // If multiple IP6 group addresses are mapped to the same\r
570 // multicast MAC address, don't configure the MNP to leave\r
571 // the MAC.\r
572 //\r
573 if (Ip6FindMac (&IpSb->MldCtrl, &Group->Mac) == 1) {\r
574 Status = IpSb->Mnp->Groups (IpSb->Mnp, FALSE, &Group->Mac);\r
575 if (EFI_ERROR (Status) && (Status != EFI_NOT_FOUND)) {\r
576 return Status;\r
577 }\r
578 }\r
579\r
580 //\r
581 // Send a leave report if we are the last node to report\r
582 //\r
583 if (Group->SendByUs) {\r
584 Status = Ip6SendMldDone (IpSb, Address);\r
585 if (EFI_ERROR (Status)) {\r
586 return Status;\r
587 }\r
588 }\r
589\r
590 RemoveEntryList (&Group->Link);\r
591 FreePool (Group);\r
592\r
593 return EFI_SUCCESS;\r
594}\r
595\r
596/**\r
597 Worker function for EfiIp6Groups(). The caller\r
598 should make sure that the parameters are valid.\r
599\r
600 @param[in] IpInstance The IP6 child to change the setting.\r
601 @param[in] JoinFlag TRUE to join the group, otherwise leave it.\r
602 @param[in] GroupAddress The target group address. If NULL, leave all\r
603 the group addresses.\r
604\r
605 @retval EFI_ALREADY_STARTED Wants to join the group, but is already a member of it\r
606 @retval EFI_OUT_OF_RESOURCES Failed to allocate sufficient resources.\r
7de8045a 607 @retval EFI_DEVICE_ERROR Failed to set the group configuration.\r
a3bcde70
HT
608 @retval EFI_SUCCESS Successfully updated the group setting.\r
609 @retval EFI_NOT_FOUND Try to leave the group which it isn't a member.\r
610\r
611**/\r
612EFI_STATUS\r
613Ip6Groups (\r
d1050b9d
MK
614 IN IP6_PROTOCOL *IpInstance,\r
615 IN BOOLEAN JoinFlag,\r
616 IN EFI_IPv6_ADDRESS *GroupAddress OPTIONAL\r
a3bcde70
HT
617 )\r
618{\r
d1050b9d
MK
619 EFI_STATUS Status;\r
620 IP6_SERVICE *IpSb;\r
621 UINT32 Index;\r
622 EFI_IPv6_ADDRESS *Group;\r
a3bcde70
HT
623\r
624 IpSb = IpInstance->Service;\r
625\r
626 if (JoinFlag) {\r
627 ASSERT (GroupAddress != NULL);\r
628\r
629 for (Index = 0; Index < IpInstance->GroupCount; Index++) {\r
630 if (EFI_IP6_EQUAL (IpInstance->GroupList + Index, GroupAddress)) {\r
631 return EFI_ALREADY_STARTED;\r
632 }\r
633 }\r
634\r
635 Status = Ip6JoinGroup (IpSb, IpInstance->Interface, GroupAddress);\r
636 if (!EFI_ERROR (Status)) {\r
637 return Ip6CombineGroups (IpInstance, GroupAddress);\r
638 }\r
639\r
640 return Status;\r
641 }\r
642\r
643 //\r
644 // Leave the group. Leave all the groups if GroupAddress is NULL.\r
645 //\r
646 for (Index = IpInstance->GroupCount; Index > 0; Index--) {\r
647 Group = IpInstance->GroupList + (Index - 1);\r
648\r
649 if ((GroupAddress == NULL) || EFI_IP6_EQUAL (Group, GroupAddress)) {\r
650 Status = Ip6LeaveGroup (IpInstance->Service, Group);\r
651 if (EFI_ERROR (Status)) {\r
652 return Status;\r
653 }\r
654\r
655 Ip6RemoveGroup (IpInstance, Group);\r
656\r
657 if (IpInstance->GroupCount == 0) {\r
658 ASSERT (Index == 1);\r
659 FreePool (IpInstance->GroupList);\r
660 IpInstance->GroupList = NULL;\r
661 }\r
662\r
663 if (GroupAddress != NULL) {\r
664 return EFI_SUCCESS;\r
665 }\r
666 }\r
667 }\r
668\r
669 return ((GroupAddress != NULL) ? EFI_NOT_FOUND : EFI_SUCCESS);\r
670}\r
671\r
672/**\r
673 Set a random value of the delay timer for the multicast address from the range\r
674 [0, Maximum Response Delay]. If a timer for any address is already\r
675 running, it is reset to the new random value only if the requested\r
676 Maximum Response Delay is less than the remaining value of the\r
677 running timer. If the Query packet specifies a Maximum Response\r
678 Delay of zero, each timer is effectively set to zero, and the action\r
679 specified below for timer expiration is performed immediately.\r
680\r
681 @param[in] IpSb The IP6 service binding instance.\r
682 @param[in] MaxRespDelay The Maximum Response Delay, in milliseconds.\r
683 @param[in] MulticastAddr The multicast address.\r
684 @param[in, out] Group Points to a IP6_MLD_GROUP list entry node.\r
685\r
686 @retval EFI_SUCCESS The delay timer is successfully updated or\r
687 timer expiration is performed immediately.\r
688 @retval Others Failed to send out MLD report message.\r
689\r
690**/\r
691EFI_STATUS\r
692Ip6UpdateDelayTimer (\r
d1050b9d
MK
693 IN IP6_SERVICE *IpSb,\r
694 IN UINT16 MaxRespDelay,\r
695 IN EFI_IPv6_ADDRESS *MulticastAddr,\r
696 IN OUT IP6_MLD_GROUP *Group\r
a3bcde70
HT
697 )\r
698{\r
d1050b9d 699 UINT32 Delay;\r
a3bcde70
HT
700\r
701 //\r
702 // If the Query packet specifies a Maximum Response Delay of zero, perform timer\r
703 // expiration immediately.\r
704 //\r
705 if (MaxRespDelay == 0) {\r
706 Group->DelayTimer = 0;\r
707 return Ip6SendMldReport (IpSb, NULL, MulticastAddr);\r
708 }\r
709\r
d1050b9d 710 Delay = (UINT32)(MaxRespDelay / 1000);\r
a3bcde70
HT
711\r
712 //\r
713 // Sets a delay timer to a random value selected from the range [0, Maximum Response Delay]\r
714 // If a timer is already running, resets it if the request Maximum Response Delay\r
715 // is less than the remaining value of the running timer.\r
716 //\r
d1050b9d 717 if ((Group->DelayTimer == 0) || (Delay < Group->DelayTimer)) {\r
a3bcde70
HT
718 Group->DelayTimer = Delay / 4294967295UL * NET_RANDOM (NetRandomInitSeed ());\r
719 }\r
720\r
721 return EFI_SUCCESS;\r
722}\r
723\r
724/**\r
725 Process the Multicast Listener Query message.\r
726\r
727 @param[in] IpSb The IP service that received the packet.\r
728 @param[in] Head The IP head of the MLD query packet.\r
729 @param[in] Packet The content of the MLD query packet with IP head\r
730 removed.\r
731\r
732 @retval EFI_SUCCESS The MLD query packet processed successfully.\r
733 @retval EFI_INVALID_PARAMETER The packet is invalid.\r
734 @retval Others Failed to process the packet.\r
735\r
736**/\r
737EFI_STATUS\r
738Ip6ProcessMldQuery (\r
d1050b9d
MK
739 IN IP6_SERVICE *IpSb,\r
740 IN EFI_IP6_HEADER *Head,\r
741 IN NET_BUF *Packet\r
a3bcde70
HT
742 )\r
743{\r
d1050b9d
MK
744 EFI_IPv6_ADDRESS AllNodes;\r
745 IP6_MLD_GROUP *Group;\r
746 IP6_MLD_HEAD MldPacket;\r
747 LIST_ENTRY *Entry;\r
748 EFI_STATUS Status;\r
a3bcde70
HT
749\r
750 Status = EFI_INVALID_PARAMETER;\r
751\r
752 //\r
753 // Check the validity of the packet, generic query or specific query\r
754 //\r
755 if (!NetIp6IsUnspecifiedAddr (&Head->SourceAddress) && !NetIp6IsLinkLocalAddr (&Head->SourceAddress)) {\r
756 goto Exit;\r
757 }\r
758\r
d1050b9d 759 if ((Head->HopLimit != 1) || !IP6_IS_MULTICAST (&Head->DestinationAddress)) {\r
a3bcde70
HT
760 goto Exit;\r
761 }\r
762\r
763 //\r
764 // The Packet points to MLD report raw data without Hop-By-Hop option.\r
765 //\r
d1050b9d 766 NetbufCopy (Packet, 0, sizeof (IP6_MLD_HEAD), (UINT8 *)&MldPacket);\r
a3bcde70
HT
767 MldPacket.MaxRespDelay = NTOHS (MldPacket.MaxRespDelay);\r
768\r
769 Ip6SetToAllNodeMulticast (FALSE, IP6_LINK_LOCAL_SCOPE, &AllNodes);\r
770 if (!EFI_IP6_EQUAL (&Head->DestinationAddress, &AllNodes)) {\r
771 //\r
772 // Receives a Multicast-Address-Specific Query, check it firstly\r
773 //\r
774 if (!EFI_IP6_EQUAL (&Head->DestinationAddress, &MldPacket.Group)) {\r
775 goto Exit;\r
776 }\r
d1050b9d 777\r
a3bcde70
HT
778 //\r
779 // The node is not listening but it receives the specific query. Just return.\r
780 //\r
781 Group = Ip6FindMldEntry (IpSb, &MldPacket.Group);\r
782 if (Group == NULL) {\r
783 Status = EFI_SUCCESS;\r
784 goto Exit;\r
785 }\r
786\r
787 Status = Ip6UpdateDelayTimer (\r
788 IpSb,\r
789 MldPacket.MaxRespDelay,\r
790 &MldPacket.Group,\r
791 Group\r
792 );\r
793 goto Exit;\r
794 }\r
795\r
796 //\r
797 // Receives a General Query, sets a delay timer for each multicast address it is listening\r
798 //\r
799 NET_LIST_FOR_EACH (Entry, &IpSb->MldCtrl.Groups) {\r
800 Group = NET_LIST_USER_STRUCT (Entry, IP6_MLD_GROUP, Link);\r
801 Status = Ip6UpdateDelayTimer (IpSb, MldPacket.MaxRespDelay, &Group->Address, Group);\r
802 if (EFI_ERROR (Status)) {\r
803 goto Exit;\r
804 }\r
805 }\r
806\r
807 Status = EFI_SUCCESS;\r
808\r
809Exit:\r
810 NetbufFree (Packet);\r
811 return Status;\r
812}\r
813\r
814/**\r
815 Process the Multicast Listener Report message.\r
816\r
817 @param[in] IpSb The IP service that received the packet.\r
818 @param[in] Head The IP head of the MLD report packet.\r
819 @param[in] Packet The content of the MLD report packet with IP head\r
820 removed.\r
821\r
822 @retval EFI_SUCCESS The MLD report packet processed successfully.\r
823 @retval EFI_INVALID_PARAMETER The packet is invalid.\r
824\r
825**/\r
826EFI_STATUS\r
827Ip6ProcessMldReport (\r
d1050b9d
MK
828 IN IP6_SERVICE *IpSb,\r
829 IN EFI_IP6_HEADER *Head,\r
830 IN NET_BUF *Packet\r
a3bcde70
HT
831 )\r
832{\r
d1050b9d
MK
833 IP6_MLD_HEAD MldPacket;\r
834 IP6_MLD_GROUP *Group;\r
835 EFI_STATUS Status;\r
a3bcde70
HT
836\r
837 Status = EFI_INVALID_PARAMETER;\r
838\r
839 //\r
840 // Validate the incoming message, if invalid, drop it.\r
841 //\r
842 if (!NetIp6IsUnspecifiedAddr (&Head->SourceAddress) && !NetIp6IsLinkLocalAddr (&Head->SourceAddress)) {\r
843 goto Exit;\r
844 }\r
845\r
d1050b9d 846 if ((Head->HopLimit != 1) || !IP6_IS_MULTICAST (&Head->DestinationAddress)) {\r
a3bcde70
HT
847 goto Exit;\r
848 }\r
849\r
850 //\r
851 // The Packet points to MLD report raw data without Hop-By-Hop option.\r
852 //\r
d1050b9d 853 NetbufCopy (Packet, 0, sizeof (IP6_MLD_HEAD), (UINT8 *)&MldPacket);\r
a3bcde70
HT
854 if (!EFI_IP6_EQUAL (&Head->DestinationAddress, &MldPacket.Group)) {\r
855 goto Exit;\r
856 }\r
857\r
858 Group = Ip6FindMldEntry (IpSb, &MldPacket.Group);\r
859 if (Group == NULL) {\r
860 goto Exit;\r
861 }\r
862\r
863 //\r
864 // The report is sent by another node, stop its own timer relates to the multicast address and clear\r
865 //\r
866\r
867 if (!Group->SendByUs) {\r
868 Group->DelayTimer = 0;\r
869 }\r
870\r
871 Status = EFI_SUCCESS;\r
872\r
873Exit:\r
874 NetbufFree (Packet);\r
875 return Status;\r
876}\r
877\r
878/**\r
879 The heartbeat timer of MLD module. It sends out a solicited MLD report when\r
880 DelayTimer expires.\r
881\r
882 @param[in] IpSb The IP6 service binding instance.\r
883\r
884**/\r
885VOID\r
886Ip6MldTimerTicking (\r
d1050b9d 887 IN IP6_SERVICE *IpSb\r
a3bcde70
HT
888 )\r
889{\r
d1050b9d
MK
890 IP6_MLD_GROUP *Group;\r
891 LIST_ENTRY *Entry;\r
a3bcde70
HT
892\r
893 //\r
894 // Send solicited report when timer expires\r
895 //\r
896 NET_LIST_FOR_EACH (Entry, &IpSb->MldCtrl.Groups) {\r
897 Group = NET_LIST_USER_STRUCT (Entry, IP6_MLD_GROUP, Link);\r
898 if ((Group->DelayTimer > 0) && (--Group->DelayTimer == 0)) {\r
899 Ip6SendMldReport (IpSb, NULL, &Group->Address);\r
900 }\r
901 }\r
902}\r