]> git.proxmox.com Git - mirror_edk2.git/blame - MdeModulePkg/Universal/Network/IScsiDxe/IScsiIbft.c
sync comments, fix function header, rename variable name to follow coding style.
[mirror_edk2.git] / MdeModulePkg / Universal / Network / IScsiDxe / IScsiIbft.c
CommitLineData
12618416 1/** @file\r
2 Implementation for iSCSI Boot Firmware Table publication.\r
6a690e23 3\r
12618416 4Copyright (c) 2004 - 2008, Intel Corporation\r
7a444476 5All rights reserved. This program and the accompanying materials\r
6are licensed and made available under the terms and conditions of the BSD License\r
7which accompanies this distribution. The full text of the license may be found at\r
8http://opensource.org/licenses/bsd-license.php\r
9\r
10THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,\r
11WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.\r
6a690e23 12\r
13Module Name:\r
14\r
15 IScsiIbft.c\r
16\r
17Abstract:\r
18\r
19 Implementation for iSCSI Boot Firmware Table publication.\r
20\r
12618416 21**/\r
6a690e23 22\r
23#include "IScsiImpl.h"\r
24\r
12618416 25/**\r
26 Initialize the header of the iSCSI Boot Firmware Table.\r
27 \r
28 @param Header[in] The header of the iSCSI Boot Firmware Table.\r
29\r
30 @retval None.\r
31\r
32**/\r
6a690e23 33VOID\r
34IScsiInitIbfTableHeader (\r
35 IN EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER *Header\r
36 )\r
6a690e23 37{\r
e48e37fc 38 ZeroMem (Header, sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER));\r
6a690e23 39\r
40 Header->Signature = EFI_ACPI_3_0_ISCSI_BOOT_FIRMWARE_TABLE_SIGNATURE;\r
41 Header->Length = IBFT_HEAP_OFFSET;\r
42 Header->Revision = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_REVISION;\r
43 Header->Checksum = 0;\r
44\r
45 Header->OemId[0] = 'I';\r
46 Header->OemId[1] = 'N';\r
47 Header->OemId[2] = 'T';\r
48 Header->OemId[3] = 'E';\r
49 Header->OemId[4] = 'L';\r
50}\r
51\r
12618416 52/**\r
53 Initialize the control section of the iSCSI Boot Firmware Table.\r
54 \r
55 @param Table[in] The ACPI table.\r
56\r
57 @param HandleCount[in] The number of the handles associated with iSCSI sessions, it's\r
58 equal to the number of iSCSI sessions.\r
59\r
60 @retval None.\r
61\r
62**/\r
6a690e23 63VOID\r
64IScsiInitControlSection (\r
65 IN EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER *Table,\r
66 IN UINTN HandleCount\r
67 )\r
6a690e23 68{\r
69 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE *Control;\r
70 UINTN NumOffset;\r
71\r
72 Control = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE *) (Table + 1);\r
73\r
e48e37fc 74 ZeroMem (Control, sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE));\r
6a690e23 75\r
76 Control->Header.StructureId = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE_ID;\r
77 Control->Header.Version = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE_VERSION;\r
78 Control->Header.Length = sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE);\r
79\r
80 //\r
81 // Each session occupies two offsets, one for the NIC section,\r
82 // the other for the Target section.\r
83 //\r
84 NumOffset = 2 * HandleCount;\r
85 if (NumOffset > 4) {\r
86 //\r
87 // Need expand the control section if more than 2 NIC/Target sections\r
88 // exist.\r
89 //\r
69b0882d 90 Control->Header.Length = (UINT16) (Control->Header.Length + (NumOffset - 4) * sizeof (UINT16));\r
6a690e23 91 }\r
92}\r
93\r
12618416 94/**\r
95 Add one item into the heap.\r
96\r
97 @param Heap[in][out] On input, the current address of the heap; On output, the address of\r
98 the heap after the item is added.\r
99\r
100 @param Data[in] The data to add into the heap.\r
101\r
102 @param Len[in] Length of the Data in byte.\r
103\r
104 @retval None.\r
105\r
106**/\r
6a690e23 107VOID\r
108IScsiAddHeapItem (\r
109 IN OUT UINT8 **Heap,\r
110 IN VOID *Data,\r
111 IN UINTN Len\r
112 )\r
6a690e23 113{\r
114 //\r
115 // Add one byte for the NULL delimiter.\r
116 //\r
117 *Heap -= Len + 1;\r
118\r
e48e37fc 119 CopyMem (*Heap, Data, Len);\r
6a690e23 120 *(*Heap + Len) = 0;\r
121}\r
122\r
12618416 123/**\r
124 Fill the Initiator section of the iSCSI Boot Firmware Table.\r
125\r
126 @param Table[in] The ACPI table.\r
127\r
128 @param Heap[in][out] The heap.\r
129\r
130 @param Handle[in] The handle associated with the iSCSI session.\r
131\r
132 @retval None.\r
133\r
134**/\r
6a690e23 135VOID\r
136IScsiFillInitiatorSection (\r
137 IN EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER *Table,\r
138 IN OUT UINT8 **Heap,\r
139 IN EFI_HANDLE Handle\r
140 )\r
6a690e23 141{\r
142 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE *Control;\r
143 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE *Initiator;\r
144 ISCSI_DRIVER_DATA *DriverData;\r
145 ISCSI_SESSION *Session;\r
146 ISCSI_PRIVATE_PROTOCOL *IScsiIdentifier;\r
147 EFI_STATUS Status;\r
148\r
149 Control = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE *) (Table + 1);\r
150\r
151 //\r
152 // Initiator section immediately follows the control section.\r
153 //\r
154 Initiator = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE *) ((UINT8 *) Control + IBFT_ROUNDUP (Control->Header.Length));\r
155\r
156 Control->InitiatorOffset = (UINT16) ((UINTN) Initiator - (UINTN) Table);\r
157\r
e48e37fc 158 ZeroMem (Initiator, sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE));\r
6a690e23 159\r
160 Initiator->Header.StructureId = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE_ID;\r
161 Initiator->Header.Version = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE_VERSION;\r
162 Initiator->Header.Length = sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE);\r
163 Initiator->Header.Flags = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE_FLAG_BLOCK_VALID | EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE_FLAG_BOOT_SELECTED;\r
164\r
165 //\r
166 // Get the identifier from the handle.\r
167 //\r
69b0882d 168 Status = gBS->HandleProtocol (Handle, &mIScsiPrivateGuid, (VOID **) &IScsiIdentifier);\r
6a690e23 169 if (EFI_ERROR (Status)) {\r
170 ASSERT (FALSE);\r
171 return ;\r
172 }\r
173\r
174 DriverData = ISCSI_DRIVER_DATA_FROM_IDENTIFIER (IScsiIdentifier);\r
175 Session = &DriverData->Session;\r
176\r
177 //\r
178 // Fill the iSCSI Initiator Name into the heap.\r
179 //\r
180 IScsiAddHeapItem (Heap, Session->InitiatorName, Session->InitiatorNameLength - 1);\r
181\r
182 Initiator->IScsiNameLength = (UINT16) (Session->InitiatorNameLength - 1);\r
183 Initiator->IScsiNameOffset = (UINT16) ((UINTN) *Heap - (UINTN) Table);\r
184}\r
185\r
12618416 186/**\r
187 Map the v4 IP address into v6 IP address.\r
188\r
189 @param V4 The v4 IP address.\r
190\r
191 @param V6 The v6 IP address.\r
192\r
193 @retval None.\r
194\r
195**/\r
6a690e23 196VOID\r
197IScsiMapV4ToV6Addr (\r
198 IN EFI_IPv4_ADDRESS *V4,\r
199 OUT EFI_IPv6_ADDRESS *V6\r
200 )\r
6a690e23 201{\r
202 UINTN Index;\r
203\r
e48e37fc 204 ZeroMem (V6, sizeof (EFI_IPv6_ADDRESS));\r
6a690e23 205\r
206 V6->Addr[10] = 0xff;\r
207 V6->Addr[11] = 0xff;\r
208\r
209 for (Index = 0; Index < 4; Index++) {\r
210 V6->Addr[12 + Index] = V4->Addr[Index];\r
211 }\r
212}\r
213\r
12618416 214/**\r
6a690e23 215 Get the NIC's PCI location and return it accroding to the composited\r
216 format defined in iSCSI Boot Firmware Table.\r
6a690e23 217\r
12618416 218 @param Controller[in] The handle of the controller.\r
6a690e23 219\r
12618416 220 @retval UINT16 The composited representation of the NIC PCI location.\r
6a690e23 221\r
12618416 222**/\r
12618416 223UINT16\r
224IScsiGetNICPciLocation (\r
225 IN EFI_HANDLE Controller\r
226 )\r
6a690e23 227{\r
228 EFI_STATUS Status;\r
229 EFI_DEVICE_PATH_PROTOCOL *DevicePath;\r
230 EFI_HANDLE PciIoHandle;\r
231 EFI_PCI_IO_PROTOCOL *PciIo;\r
232 UINTN Segment;\r
233 UINTN Bus;\r
234 UINTN Device;\r
235 UINTN Function;\r
236\r
237 Status = gBS->HandleProtocol (\r
238 Controller,\r
239 &gEfiDevicePathProtocolGuid,\r
69b0882d 240 (VOID **)&DevicePath\r
6a690e23 241 );\r
242 if (EFI_ERROR (Status)) {\r
243 return 0;\r
244 }\r
245\r
246 Status = gBS->LocateDevicePath (\r
247 &gEfiPciIoProtocolGuid,\r
248 &DevicePath,\r
249 &PciIoHandle\r
250 );\r
251 if (EFI_ERROR (Status)) {\r
252 return 0;\r
253 }\r
254\r
69b0882d 255 Status = gBS->HandleProtocol (PciIoHandle, &gEfiPciIoProtocolGuid, (VOID **)&PciIo);\r
6a690e23 256 if (EFI_ERROR (Status)) {\r
257 return 0;\r
258 }\r
259\r
260 Status = PciIo->GetLocation (PciIo, &Segment, &Bus, &Device, &Function);\r
261 if (EFI_ERROR (Status)) {\r
262 return 0;\r
263 }\r
264\r
265 return (UINT16) ((Bus << 8) | (Device << 3) | Function);\r
266}\r
267\r
12618416 268/**\r
269 Get the MAC address of the controller.\r
270 \r
271 @param Controller[in] The handle of the controller.\r
272\r
273 @retval EFI_MAC_ADDRESS * The mac address.\r
274\r
275**/\r
6a690e23 276EFI_MAC_ADDRESS *\r
277IScsiGetMacAddress (\r
278 IN EFI_HANDLE Controller\r
279 )\r
6a690e23 280{\r
281 EFI_STATUS Status;\r
282 EFI_SIMPLE_NETWORK_PROTOCOL *Snp;\r
283\r
284 Status = gBS->HandleProtocol (\r
285 Controller,\r
286 &gEfiSimpleNetworkProtocolGuid,\r
69b0882d 287 (VOID **) &Snp\r
6a690e23 288 );\r
289 ASSERT_EFI_ERROR (Status);\r
290\r
291 return &Snp->Mode->PermanentAddress;\r
292}\r
293\r
12618416 294/**\r
295 Fill the NIC and target sections in iSCSI Boot Firmware Table.\r
296\r
297 @param Table[in] The buffer of the ACPI table.\r
298\r
299 @param Heap[in][out] The heap buffer used to store the variable length parameters such as iSCSI name.\r
300\r
301 @param HandleCount[in] The number of handles having iSCSI private protocol installed.\r
302\r
303 @param Handles[in] The handle buffer.\r
304\r
305 @retval None.\r
306\r
307**/\r
6a690e23 308VOID\r
309IScsiFillNICAndTargetSections (\r
310 IN EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER *Table,\r
311 IN OUT UINT8 **Heap,\r
312 IN UINTN HandleCount,\r
313 IN EFI_HANDLE *Handles\r
314 )\r
6a690e23 315{\r
316 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE *Control;\r
317 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE *Nic;\r
318 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE *Target;\r
319 ISCSI_DRIVER_DATA *DriverData;\r
320 ISCSI_SESSION_CONFIG_DATA *SessionConfigData;\r
321 ISCSI_CHAP_AUTH_CONFIG_NVDATA *AuthConfig;\r
322 UINT16 *SectionOffset;\r
323 UINTN Index;\r
324 UINT16 Length;\r
325 EFI_MAC_ADDRESS *Mac;\r
326 ISCSI_PRIVATE_PROTOCOL *IScsiIdentifier;\r
327 EFI_STATUS Status;\r
328\r
329 //\r
330 // Get the offset of the first Nic and Target section.\r
331 //\r
332 Control = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_CONTROL_STRUCTURE *) (Table + 1);\r
333 Nic = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE *) ((UINTN) Table +\r
334 Control->InitiatorOffset + IBFT_ROUNDUP (sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_INITIATOR_STRUCTURE)));\r
335 Target = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE *) ((UINTN) Nic +\r
336 IBFT_ROUNDUP (sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE)));\r
337\r
338 SectionOffset = &Control->NIC0Offset;\r
339\r
340 for (Index = 0; Index < HandleCount; Index++) {\r
69b0882d 341 Status = gBS->HandleProtocol (Handles[Index], &mIScsiPrivateGuid, (VOID **)&IScsiIdentifier);\r
6a690e23 342 if (EFI_ERROR (Status)) {\r
343 ASSERT (FALSE);\r
344 return ;\r
345 }\r
346\r
347 DriverData = ISCSI_DRIVER_DATA_FROM_IDENTIFIER (IScsiIdentifier);\r
348 SessionConfigData = &DriverData->Session.ConfigData;\r
349 AuthConfig = &DriverData->Session.AuthData.AuthConfig;\r
350\r
351 //\r
352 // Fill the Nic section.\r
353 //\r
e48e37fc 354 ZeroMem (Nic, sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE));\r
6a690e23 355\r
356 Nic->Header.StructureId = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE_ID;\r
357 Nic->Header.Version = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE_VERSION;\r
358 Nic->Header.Length = sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE);\r
359 Nic->Header.Index = (UINT8) Index;\r
360 Nic->Header.Flags = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE_FLAG_BLOCK_VALID |\r
361 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE_FLAG_BOOT_SELECTED |\r
362 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE_FLAG_GLOBAL;\r
363\r
364 //\r
365 // Get the subnet mask prefix length.\r
366 //\r
367 Nic->SubnetMaskPrefixLength = IScsiGetSubnetMaskPrefixLength (&SessionConfigData->NvData.SubnetMask);\r
368\r
369 if (SessionConfigData->NvData.InitiatorInfoFromDhcp) {\r
370 Nic->Origin = IpPrefixOriginDhcp;\r
371 } else {\r
372 Nic->Origin = IpPrefixOriginManual;\r
373 }\r
374 //\r
375 // Map the various v4 addresses into v6 addresses.\r
376 //\r
377 IScsiMapV4ToV6Addr (&SessionConfigData->NvData.LocalIp, &Nic->Ip);\r
378 IScsiMapV4ToV6Addr (&SessionConfigData->NvData.Gateway, &Nic->Gateway);\r
379 IScsiMapV4ToV6Addr (&SessionConfigData->PrimaryDns, &Nic->PrimaryDns);\r
380 IScsiMapV4ToV6Addr (&SessionConfigData->SecondaryDns, &Nic->SecondaryDns);\r
381 IScsiMapV4ToV6Addr (&SessionConfigData->DhcpServer, &Nic->DhcpServer);\r
382\r
383 Mac = IScsiGetMacAddress (DriverData->Controller);\r
e48e37fc 384 CopyMem (Nic->Mac, Mac, sizeof (Nic->Mac));\r
6a690e23 385\r
386 //\r
387 // Get the PCI location of the Nic.\r
388 //\r
389 Nic->PciLocation = IScsiGetNICPciLocation (DriverData->Controller);\r
390\r
391 *SectionOffset = (UINT16) ((UINTN) Nic - (UINTN) Table);\r
392 SectionOffset++;\r
393\r
394 //\r
395 // Fill the Target section.\r
396 //\r
e48e37fc 397 ZeroMem (Target, sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE));\r
6a690e23 398\r
399 Target->Header.StructureId = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE_ID;\r
400 Target->Header.Version = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE_VERSION;\r
401 Target->Header.Length = sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE);\r
402 Target->Header.Index = (UINT8) Index;\r
403 Target->Header.Flags = EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE_FLAG_BLOCK_VALID | EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE_FLAG_BOOT_SELECTED;\r
404 Target->Port = SessionConfigData->NvData.TargetPort;\r
405 Target->CHAPType = AuthConfig->CHAPType;\r
406 Target->NicIndex = (UINT8) Index;\r
407\r
408 IScsiMapV4ToV6Addr (&SessionConfigData->NvData.TargetIp, &Target->Ip);\r
e48e37fc 409 CopyMem (Target->BootLun, SessionConfigData->NvData.BootLun, sizeof (Target->BootLun));\r
6a690e23 410\r
411 //\r
412 // Target iSCSI Name, CHAP name/secret, reverse CHAP name/secret.\r
413 //\r
414 Length = (UINT16) AsciiStrLen (SessionConfigData->NvData.TargetName);\r
415 IScsiAddHeapItem (Heap, SessionConfigData->NvData.TargetName, Length);\r
416\r
417 Target->IScsiNameLength = Length;\r
418 Target->IScsiNameOffset = (UINT16) ((UINTN) *Heap - (UINTN) Table);\r
419\r
420 if (Target->CHAPType != ISCSI_CHAP_NONE) {\r
421 //\r
422 // CHAP Name\r
423 //\r
424 Length = (UINT16) AsciiStrLen (AuthConfig->CHAPName);\r
425 IScsiAddHeapItem (Heap, AuthConfig->CHAPName, Length);\r
426 Target->CHAPNameLength = Length;\r
427 Target->CHAPNameOffset = (UINT16) ((UINTN) *Heap - (UINTN) Table);\r
428\r
429 //\r
430 // CHAP Secret\r
431 //\r
432 Length = (UINT16) AsciiStrLen (AuthConfig->CHAPSecret);\r
433 IScsiAddHeapItem (Heap, AuthConfig->CHAPSecret, Length);\r
434 Target->CHAPSecretLength = Length;\r
435 Target->CHAPSecretOffset = (UINT16) ((UINTN) *Heap - (UINTN) Table);\r
436\r
437 if (Target->CHAPType == ISCSI_CHAP_MUTUAL) {\r
438 //\r
439 // Reverse CHAP Name\r
440 //\r
441 Length = (UINT16) AsciiStrLen (AuthConfig->ReverseCHAPName);\r
442 IScsiAddHeapItem (Heap, AuthConfig->ReverseCHAPName, Length);\r
443 Target->ReverseCHAPNameLength = Length;\r
444 Target->ReverseCHAPNameOffset = (UINT16) ((UINTN) *Heap - (UINTN) Table);\r
445\r
446 //\r
447 // Reverse CHAP Secret\r
448 //\r
449 Length = (UINT16) AsciiStrLen (AuthConfig->ReverseCHAPSecret);\r
450 IScsiAddHeapItem (Heap, AuthConfig->ReverseCHAPSecret, Length);\r
451 Target->ReverseCHAPSecretLength = Length;\r
452 Target->ReverseCHAPSecretOffset = (UINT16) ((UINTN) *Heap - (UINTN) Table);\r
453 }\r
454 }\r
455\r
456 *SectionOffset = (UINT16) ((UINTN) Target - (UINTN) Table);\r
457 SectionOffset++;\r
458\r
459 //\r
460 // Advance to the next NIC/Target pair\r
461 //\r
462 Nic = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE *) ((UINTN) Target +\r
463 IBFT_ROUNDUP (sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE)));\r
464 Target = (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_TARGET_STRUCTURE *) ((UINTN) Nic +\r
465 IBFT_ROUNDUP (sizeof (EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_NIC_STRUCTURE)));\r
466 }\r
467}\r
468\r
12618416 469/**\r
6a690e23 470 Publish and remove the iSCSI Boot Firmware Table according to the iSCSI\r
471 session status.\r
472\r
12618416 473 @param None.\r
6a690e23 474\r
12618416 475 @retval None.\r
6a690e23 476\r
12618416 477**/\r
478VOID\r
479IScsiPublishIbft (\r
480 IN VOID\r
481 )\r
6a690e23 482{\r
483 EFI_STATUS Status;\r
484 UINTN TableHandle;\r
485 EFI_ACPI_SUPPORT_PROTOCOL *AcpiSupport;\r
486 EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER *Table;\r
487 UINTN HandleCount;\r
488 EFI_HANDLE *HandleBuffer;\r
489 UINT8 *Heap;\r
490 INTN Index;\r
491 EFI_ACPI_TABLE_VERSION Version;\r
492 UINT32 Signature;\r
493\r
69b0882d 494 Status = gBS->LocateProtocol (&gEfiAcpiSupportProtocolGuid, NULL, (VOID **)&AcpiSupport);\r
6a690e23 495 if (EFI_ERROR (Status)) {\r
496 return ;\r
497 }\r
498 //\r
499 // Try to remove the old iSCSI Boot Firmware Table.\r
500 //\r
501 for (Index = 0;; Index++) {\r
502 Status = AcpiSupport->GetAcpiTable (\r
503 AcpiSupport,\r
504 Index,\r
69b0882d 505 (VOID **)&Table,\r
6a690e23 506 &Version,\r
507 &TableHandle\r
508 );\r
509 if (EFI_ERROR (Status)) {\r
510 break;\r
511 }\r
512\r
513 Signature = Table->Signature;\r
e48e37fc 514 gBS->FreePool (Table);\r
6a690e23 515\r
516 if (Signature == EFI_ACPI_3_0_ISCSI_BOOT_FIRMWARE_TABLE_SIGNATURE) {\r
517 //\r
518 // Remove the table.\r
519 //\r
520 Status = AcpiSupport->SetAcpiTable (\r
521 AcpiSupport,\r
522 NULL,\r
523 FALSE,\r
524 Version,\r
525 &TableHandle\r
526 );\r
527 if (EFI_ERROR (Status)) {\r
528 return ;\r
529 }\r
530\r
531 break;\r
532 }\r
533 }\r
534 //\r
535 // Get all iSCSI private protocols.\r
536 //\r
537 Status = gBS->LocateHandleBuffer (\r
538 ByProtocol,\r
539 &mIScsiPrivateGuid,\r
540 NULL,\r
541 &HandleCount,\r
542 &HandleBuffer\r
543 );\r
544 if (EFI_ERROR (Status)) {\r
545 return ;\r
546 }\r
547 //\r
548 // Allocate 4k bytes to hold the ACPI table.\r
549 //\r
e48e37fc 550 Table = AllocatePool (IBFT_MAX_SIZE);\r
6a690e23 551 if (Table == NULL) {\r
552 return ;\r
553 }\r
554\r
69b0882d 555 Heap = (UINT8 *) Table + IBFT_HEAP_OFFSET;\r
6a690e23 556\r
557 //\r
558 // Fill in the various section of the iSCSI Boot Firmware Table.\r
559 //\r
560 IScsiInitIbfTableHeader (Table);\r
561 IScsiInitControlSection (Table, HandleCount);\r
562 IScsiFillInitiatorSection (Table, &Heap, HandleBuffer[0]);\r
563 IScsiFillNICAndTargetSections (Table, &Heap, HandleCount, HandleBuffer);\r
564\r
e48e37fc 565 gBS->FreePool (HandleBuffer);\r
6a690e23 566\r
567 TableHandle = 0;\r
568\r
569 //\r
570 // Install or update the iBFT table.\r
571 //\r
572 Status = AcpiSupport->SetAcpiTable (\r
573 AcpiSupport,\r
574 Table,\r
575 TRUE,\r
576 EFI_ACPI_TABLE_VERSION_3_0,\r
577 &TableHandle\r
578 );\r
579 if (!EFI_ERROR (Status)) {\r
580 AcpiSupport->PublishTables (AcpiSupport, EFI_ACPI_TABLE_VERSION_3_0);\r
581 }\r
582\r
e48e37fc 583 gBS->FreePool (Table);\r
6a690e23 584}\r