Home | History | Annotate | Download | only in SectionExtractionDxe
      1 /** @file
      2  Section Extraction DXE Driver
      3 
      4 Copyright (c) 2013 - 2014, Intel Corporation. All rights reserved.<BR>
      5 This program and the accompanying materials
      6 are licensed and made available under the terms and conditions of the BSD License
      7 which accompanies this distribution.  The full text of the license may be found at
      8 http://opensource.org/licenses/bsd-license.php
      9 
     10 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
     11 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
     12 
     13 **/
     14 
     15 #include <PiDxe.h>
     16 #include <Protocol/GuidedSectionExtraction.h>
     17 #include <Library/DebugLib.h>
     18 #include <Library/ExtractGuidedSectionLib.h>
     19 #include <Library/MemoryAllocationLib.h>
     20 #include <Library/BaseMemoryLib.h>
     21 #include <Library/UefiBootServicesTableLib.h>
     22 
     23 /**
     24   The ExtractSection() function processes the input section and
     25   allocates a buffer from the pool in which it returns the section
     26   contents. If the section being extracted contains
     27   authentication information (the section's
     28   GuidedSectionHeader.Attributes field has the
     29   EFI_GUIDED_SECTION_AUTH_STATUS_VALID bit set), the values
     30   returned in AuthenticationStatus must reflect the results of
     31   the authentication operation. Depending on the algorithm and
     32   size of the encapsulated data, the time that is required to do
     33   a full authentication may be prohibitively long for some
     34   classes of systems. To indicate this, use
     35   EFI_SECURITY_POLICY_PROTOCOL_GUID, which may be published by
     36   the security policy driver (see the Platform Initialization
     37   Driver Execution Environment Core Interface Specification for
     38   more details and the GUID definition). If the
     39   EFI_SECURITY_POLICY_PROTOCOL_GUID exists in the handle
     40   database, then, if possible, full authentication should be
     41   skipped and the section contents simply returned in the
     42   OutputBuffer. In this case, the
     43   EFI_AUTH_STATUS_PLATFORM_OVERRIDE bit AuthenticationStatus
     44   must be set on return. ExtractSection() is callable only from
     45   TPL_NOTIFY and below. Behavior of ExtractSection() at any
     46   EFI_TPL above TPL_NOTIFY is undefined. Type EFI_TPL is
     47   defined in RaiseTPL() in the UEFI 2.0 specification.
     48 
     49 
     50   @param This         Indicates the
     51                       EFI_GUIDED_SECTION_EXTRACTION_PROTOCOL instance.
     52   @param InputSection Buffer containing the input GUIDed section
     53                       to be processed. OutputBuffer OutputBuffer
     54                       is allocated from boot services pool
     55                       memory and contains the new section
     56                       stream. The caller is responsible for
     57                       freeing this buffer.
     58   @param OutputBuffer *OutputBuffer is allocated from boot services
     59                       pool memory and contains the new section stream.
     60                       The caller is responsible for freeing this buffer.
     61   @param OutputSize   A pointer to a caller-allocated UINTN in
     62                       which the size of OutputBuffer allocation
     63                       is stored. If the function returns
     64                       anything other than EFI_SUCCESS, the value
     65                       of OutputSize is undefined.
     66 
     67   @param AuthenticationStatus A pointer to a caller-allocated
     68                               UINT32 that indicates the
     69                               authentication status of the
     70                               output buffer. If the input
     71                               section's
     72                               GuidedSectionHeader.Attributes
     73                               field has the
     74                               EFI_GUIDED_SECTION_AUTH_STATUS_VAL
     75                               bit as clear, AuthenticationStatus
     76                               must return zero. Both local bits
     77                               (19:16) and aggregate bits (3:0)
     78                               in AuthenticationStatus are
     79                               returned by ExtractSection().
     80                               These bits reflect the status of
     81                               the extraction operation. The bit
     82                               pattern in both regions must be
     83                               the same, as the local and
     84                               aggregate authentication statuses
     85                               have equivalent meaning at this
     86                               level. If the function returns
     87                               anything other than EFI_SUCCESS,
     88                               the value of AuthenticationStatus
     89                               is undefined.
     90 
     91 
     92   @retval EFI_SUCCESS          The InputSection was successfully
     93                                processed and the section contents were
     94                                returned.
     95 
     96   @retval EFI_OUT_OF_RESOURCES The system has insufficient
     97                                resources to process the
     98                                request.
     99 
    100   @retval EFI_INVALID_PARAMETER The GUID in InputSection does
    101                                 not match this instance of the
    102                                 GUIDed Section Extraction
    103                                 Protocol.
    104 
    105 **/
    106 EFI_STATUS
    107 EFIAPI
    108 CustomGuidedSectionExtract (
    109   IN CONST  EFI_GUIDED_SECTION_EXTRACTION_PROTOCOL *This,
    110   IN CONST  VOID                                   *InputSection,
    111   OUT       VOID                                   **OutputBuffer,
    112   OUT       UINTN                                  *OutputSize,
    113   OUT       UINT32                                 *AuthenticationStatus
    114   );
    115 
    116 //
    117 // Module global for the Section Extraction Protocol handle
    118 //
    119 EFI_HANDLE mSectionExtractionHandle = NULL;
    120 
    121 //
    122 // Module global for the Section Extraction Protocol instance
    123 //
    124 EFI_GUIDED_SECTION_EXTRACTION_PROTOCOL mCustomGuidedSectionExtractionProtocol = {
    125   CustomGuidedSectionExtract
    126 };
    127 
    128 /**
    129   The ExtractSection() function processes the input section and
    130   allocates a buffer from the pool in which it returns the section
    131   contents. If the section being extracted contains
    132   authentication information (the section's
    133   GuidedSectionHeader.Attributes field has the
    134   EFI_GUIDED_SECTION_AUTH_STATUS_VALID bit set), the values
    135   returned in AuthenticationStatus must reflect the results of
    136   the authentication operation. Depending on the algorithm and
    137   size of the encapsulated data, the time that is required to do
    138   a full authentication may be prohibitively long for some
    139   classes of systems. To indicate this, use
    140   EFI_SECURITY_POLICY_PROTOCOL_GUID, which may be published by
    141   the security policy driver (see the Platform Initialization
    142   Driver Execution Environment Core Interface Specification for
    143   more details and the GUID definition). If the
    144   EFI_SECURITY_POLICY_PROTOCOL_GUID exists in the handle
    145   database, then, if possible, full authentication should be
    146   skipped and the section contents simply returned in the
    147   OutputBuffer. In this case, the
    148   EFI_AUTH_STATUS_PLATFORM_OVERRIDE bit AuthenticationStatus
    149   must be set on return. ExtractSection() is callable only from
    150   TPL_NOTIFY and below. Behavior of ExtractSection() at any
    151   EFI_TPL above TPL_NOTIFY is undefined. Type EFI_TPL is
    152   defined in RaiseTPL() in the UEFI 2.0 specification.
    153 
    154 
    155   @param This         Indicates the
    156                       EFI_GUIDED_SECTION_EXTRACTION_PROTOCOL instance.
    157   @param InputSection Buffer containing the input GUIDed section
    158                       to be processed. OutputBuffer OutputBuffer
    159                       is allocated from boot services pool
    160                       memory and contains the new section
    161                       stream. The caller is responsible for
    162                       freeing this buffer.
    163   @param OutputBuffer *OutputBuffer is allocated from boot services
    164                       pool memory and contains the new section stream.
    165                       The caller is responsible for freeing this buffer.
    166   @param OutputSize   A pointer to a caller-allocated UINTN in
    167                       which the size of OutputBuffer allocation
    168                       is stored. If the function returns
    169                       anything other than EFI_SUCCESS, the value
    170                       of OutputSize is undefined.
    171 
    172   @param AuthenticationStatus A pointer to a caller-allocated
    173                               UINT32 that indicates the
    174                               authentication status of the
    175                               output buffer. If the input
    176                               section's
    177                               GuidedSectionHeader.Attributes
    178                               field has the
    179                               EFI_GUIDED_SECTION_AUTH_STATUS_VAL
    180                               bit as clear, AuthenticationStatus
    181                               must return zero. Both local bits
    182                               (19:16) and aggregate bits (3:0)
    183                               in AuthenticationStatus are
    184                               returned by ExtractSection().
    185                               These bits reflect the status of
    186                               the extraction operation. The bit
    187                               pattern in both regions must be
    188                               the same, as the local and
    189                               aggregate authentication statuses
    190                               have equivalent meaning at this
    191                               level. If the function returns
    192                               anything other than EFI_SUCCESS,
    193                               the value of AuthenticationStatus
    194                               is undefined.
    195 
    196 
    197   @retval EFI_SUCCESS          The InputSection was successfully
    198                                processed and the section contents were
    199                                returned.
    200 
    201   @retval EFI_OUT_OF_RESOURCES The system has insufficient
    202                                resources to process the
    203                                request.
    204 
    205   @retval EFI_INVALID_PARAMETER The GUID in InputSection does
    206                                 not match this instance of the
    207                                 GUIDed Section Extraction
    208                                 Protocol.
    209 
    210 **/
    211 EFI_STATUS
    212 EFIAPI
    213 CustomGuidedSectionExtract (
    214   IN CONST  EFI_GUIDED_SECTION_EXTRACTION_PROTOCOL *This,
    215   IN CONST  VOID                                   *InputSection,
    216   OUT       VOID                                   **OutputBuffer,
    217   OUT       UINTN                                  *OutputSize,
    218   OUT       UINT32                                 *AuthenticationStatus
    219   )
    220 {
    221   EFI_STATUS      Status;
    222   VOID            *ScratchBuffer;
    223   VOID            *AllocatedOutputBuffer;
    224   UINT32          OutputBufferSize;
    225   UINT32          ScratchBufferSize;
    226   UINT16          SectionAttribute;
    227 
    228   //
    229   // Init local variable
    230   //
    231   ScratchBuffer         = NULL;
    232   AllocatedOutputBuffer = NULL;
    233 
    234   //
    235   // Call GetInfo to get the size and attribute of input guided section data.
    236   //
    237   Status = ExtractGuidedSectionGetInfo (
    238              InputSection,
    239              &OutputBufferSize,
    240              &ScratchBufferSize,
    241              &SectionAttribute
    242              );
    243 
    244   if (EFI_ERROR (Status)) {
    245     DEBUG ((DEBUG_ERROR, "GetInfo from guided section Failed - %r\n", Status));
    246     return Status;
    247   }
    248 
    249   if (ScratchBufferSize > 0) {
    250     //
    251     // Allocate scratch buffer
    252     //
    253     ScratchBuffer = AllocatePool (ScratchBufferSize);
    254     if (ScratchBuffer == NULL) {
    255       return EFI_OUT_OF_RESOURCES;
    256     }
    257   }
    258 
    259   if (OutputBufferSize > 0) {
    260     //
    261     // Allocate output buffer
    262     //
    263     AllocatedOutputBuffer = AllocatePool (OutputBufferSize);
    264     if (AllocatedOutputBuffer == NULL) {
    265       FreePool (ScratchBuffer);
    266       return EFI_OUT_OF_RESOURCES;
    267     }
    268     *OutputBuffer = AllocatedOutputBuffer;
    269   }
    270 
    271   //
    272   // Call decode function to extract raw data from the guided section.
    273   //
    274   Status = ExtractGuidedSectionDecode (
    275              InputSection,
    276              OutputBuffer,
    277              ScratchBuffer,
    278              AuthenticationStatus
    279              );
    280   if (EFI_ERROR (Status)) {
    281     //
    282     // Decode failed
    283     //
    284     if (AllocatedOutputBuffer != NULL) {
    285       FreePool (AllocatedOutputBuffer);
    286     }
    287     if (ScratchBuffer != NULL) {
    288       FreePool (ScratchBuffer);
    289     }
    290     DEBUG ((DEBUG_ERROR, "Extract guided section Failed - %r\n", Status));
    291     return Status;
    292   }
    293 
    294   if (*OutputBuffer != AllocatedOutputBuffer) {
    295     //
    296     // OutputBuffer was returned as a different value,
    297     // so copy section contents to the allocated memory buffer.
    298     //
    299     CopyMem (AllocatedOutputBuffer, *OutputBuffer, OutputBufferSize);
    300     *OutputBuffer = AllocatedOutputBuffer;
    301   }
    302 
    303   //
    304   // Set real size of output buffer.
    305   //
    306   *OutputSize = (UINTN) OutputBufferSize;
    307 
    308   //
    309   // Free unused scratch buffer.
    310   //
    311   if (ScratchBuffer != NULL) {
    312     FreePool (ScratchBuffer);
    313   }
    314 
    315   return EFI_SUCCESS;
    316 }
    317 
    318 /**
    319   Main entry for the Section Extraction DXE module.
    320 
    321   This routine registers the Section Extraction Protocols that have been registered
    322   with the Section Extraction Library.
    323 
    324   @param[in] ImageHandle    The firmware allocated handle for the EFI image.
    325   @param[in] SystemTable    A pointer to the EFI System Table.
    326 
    327   @retval EFI_SUCCESS       The entry point is executed successfully.
    328   @retval other             Some error occurs when executing this entry point.
    329 
    330 **/
    331 EFI_STATUS
    332 EFIAPI
    333 SectionExtractionDxeEntry (
    334   IN EFI_HANDLE        ImageHandle,
    335   IN EFI_SYSTEM_TABLE  *SystemTable
    336   )
    337 {
    338   EFI_STATUS  Status;
    339   EFI_GUID    *ExtractHandlerGuidTable;
    340   UINTN       ExtractHandlerNumber;
    341 
    342   //
    343   // Get custom extract guided section method guid list
    344   //
    345   ExtractHandlerNumber = ExtractGuidedSectionGetGuidList (&ExtractHandlerGuidTable);
    346 
    347   //
    348   // Install custom guided extraction protocol
    349   //
    350   while (ExtractHandlerNumber-- > 0) {
    351     Status = gBS->InstallMultipleProtocolInterfaces (
    352                     &mSectionExtractionHandle,
    353                     &ExtractHandlerGuidTable [ExtractHandlerNumber], &mCustomGuidedSectionExtractionProtocol,
    354                     NULL
    355                     );
    356     ASSERT_EFI_ERROR (Status);
    357   }
    358 
    359   return EFI_SUCCESS;
    360 }
    361