Your IP : 216.73.216.52


Current Path : /snap/lxd/current/lib/python3/dist-packages/pyuefivars/
Upload File :
Current File : //snap/lxd/current/lib/python3/dist-packages/pyuefivars/efivarfs.py

#!/usr/bin/env python3
#
# Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
# SPDX-License-Identifier: MIT

import os
from .varstore import UEFIVar, UEFIVarStore
import uuid
import sys

EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS = 0x20


class EFIVARFSUEFIVarStore(UEFIVarStore):
    """
    Varstore class to ingest an efivarfs directory as UEFI variables.

    Please beware that efivarfs only exposes UEFI variables that execute
    at runtime, so any boot time variables are unavailable. Also, additional
    metadata such as secure variable details are not readable through efivarfs.
    """

    def __init__(self, path):
        super().__init__()
        self.is_empty = False

        if not path:
            path = '/sys/firmware/efi/efivars'

        if not os.path.isdir(path):
            raise Exception(f'"{path}" is not a valid efivarfs path')

        var_names = os.listdir(path)
        # example:  'X-Nitro-BootServicesExited-8be4df61-93ca-11d2-aa0d-00e098032b8c'

        for var_name in var_names:
            # efivarfs file name are f'{name}-{guid}'
            s = var_name.split('-')

            try:
                # The last 5 elements make up the GUID
                guid = uuid.UUID('-'.join(s[-5:])).bytes_le
            except ValueError:
                raise Exception(f'Invalid efivarfs file "{var_name}"')

            # Anything before that is the name of the variable
            name = '-'.join(s[:-5])

            # efivarfs file contents are f'{attr.le32}{data}'
            content = open(os.path.join(path, var_name), 'rb').read()
            data = content[4:]
            attr = int.from_bytes(content[:4], "little")

            # UEFI Secure Boot variables are special; they contain hidden
            # key material that we can not extract from efivarfs. Skip them.
            if attr & EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS:
                print(f'Skipping authenticated variable "{var_name}"', file=sys.stderr)
                continue

            # Now that we reassembled everything, remember the variable
            self.vars.append(UEFIVar(name, data, guid, attr))

    def __str__(self) -> str:
        raise Exception('Unable to serialize efivarfs into a file')