[PATCH v3 2/2] efi_loader: efi_net: let efi_net_set_dp properly update the device path
Adriano Córdova
adrianox at gmail.com
Thu Feb 20 14:24:47 CET 2025
El jue, 20 feb 2025 a las 10:12, Adriano Córdova (<adrianox at gmail.com>)
escribió:
>
>
> El jue, 20 feb 2025 a las 4:54, Heinrich Schuchardt (<xypron.glpk at gmx.de>)
> escribió:
>
>> On 1/27/25 13:34, Adriano Cordova wrote:
>> > This commit fixes an use after free introduced in Commit e55a4acb54
>> > (" efi_loader: net: set EFI bootdevice device path to HTTP when loaded
>> > from wget"). The logic in efi_net_set_dp is reworked so that when the
>> > function is invoked it not only changes the value of the static variable
>> > net_dp (this is how the function was implemented in e55a4acb54) but also
>> > updates the protocol interface of the device path protocol in case efi
>> > has started.
>> >
>> > Fixes: e55a4acb54e8 ("efi_loader: net: set EFI bootdevice device path
>> to HTTP when loaded from wget")
>> > Signed-off-by: Adriano Cordova <adriano.cordova at canonical.com>
>> > ---
>> > Changes in v3: initialize ret variable in efi_net_set_dp
>> > Changes in v2: use reinstall_protocol_interface() and new_net_dp in the
>> body of efi_net_set_dp()
>> >
>> > lib/efi_loader/efi_net.c | 61 ++++++++++++++++++++++++++++++++++------
>> > 1 file changed, 52 insertions(+), 9 deletions(-)
>> >
>> > diff --git a/lib/efi_loader/efi_net.c b/lib/efi_loader/efi_net.c
>> > index 67593ef50c0..f696a47060c 100644
>> > --- a/lib/efi_loader/efi_net.c
>> > +++ b/lib/efi_loader/efi_net.c
>> > @@ -925,12 +925,15 @@ efi_status_t efi_net_register(void)
>> > &netobj->net);
>> > if (r != EFI_SUCCESS)
>> > goto failure_to_add_protocol;
>> > - if (!net_dp)
>> > - efi_net_set_dp("Net", NULL);
>> > - r = efi_add_protocol(&netobj->header, &efi_guid_device_path,
>> > - net_dp);
>> > +
>> > + if (net_dp)
>> > + r = efi_add_protocol(&netobj->header,
>> &efi_guid_device_path,
>> > + net_dp);
>> > + else
>> > + r = efi_net_set_dp("Net", NULL);
>>
>> Hello Adriano,
>>
>> thanks for spotting and fixing the issue.
>>
>> Patch 1 of the series is already merged.
>>
>> Why would we create a device-path here and not install it?
>>
>> If this is really intended, please, add a comment.
>>
>
> Hi Heinrich,
>
> efi_net_set_dp is installing it (only if EFI is already up, otherwise it
> saves it in net_dp).
>
>>
>> > if (r != EFI_SUCCESS)
>> > goto failure_to_add_protocol;
>> > +
>> > r = efi_add_protocol(&netobj->header,
>> &efi_pxe_base_code_protocol_guid,
>> > &netobj->pxe);
>> > if (r != EFI_SUCCESS)
>> > @@ -1055,18 +1058,58 @@ out_of_resources:
>> > */
>> > efi_status_t efi_net_set_dp(const char *dev, const char *server)
>> > {
>> > - efi_free_pool(net_dp);
>> > + efi_status_t ret = EFI_SUCCESS;
>> > + struct efi_handler *phandler;
>> > + struct efi_device_path *old_net_dp, *new_net_dp;
>> >
>> > - net_dp = NULL;
>> > + old_net_dp = net_dp;
>> > + new_net_dp = NULL;
>> > if (!strcmp(dev, "Net"))
>> > - net_dp = efi_dp_from_eth();
>> > + new_net_dp = efi_dp_from_eth();
>> > else if (!strcmp(dev, "Http"))
>> > - net_dp = efi_dp_from_http(server);
>> > + new_net_dp = efi_dp_from_http(server);
>> >
>> > - if (!net_dp)
>> > + if (!new_net_dp) {
>> > return EFI_OUT_OF_RESOURCES;
>> > + }
>> > +
>> > + // If netobj is not started yet, end here.
>> > + if (!netobj) {
>> > + goto exit;
>> > + }
>> > +
>> > + phandler = NULL;
>> > + efi_search_protocol(&netobj->header, &efi_guid_device_path,
>> &phandler);
>> > +
>> > + // If the device path protocol is not yet installed, install it
>> > + if (!phandler)
>> > + goto add;
>> > +
>> > + // If it is already installed, try to update it
>> > + ret = efi_reinstall_protocol_interface(&netobj->header,
>> &efi_guid_device_path,
>> > + old_net_dp, new_net_dp);
>> > + if (ret != EFI_SUCCESS)
>> > + goto error;
>> > +
>> > + net_dp = new_net_dp;
>> > + efi_free_pool(old_net_dp);
>>
>> How can we be sure that old_net_dp is a device-tree that we installed
>> and was not installed by an EFI application?
>>
>> Best regards
>>
>> Heinrich
>>
>
> Because it comes from the net_dp variable, which is not accessible to EFI
> applications, and is always allocated by U-Boot.
>
> Best,
> Adriano
>
Although it could be teoretically freed by an application. In that case we
would be freeing it twice.
>
>> >
>> > return EFI_SUCCESS;
>> > +add:
>> > + ret = efi_add_protocol(&netobj->header, &efi_guid_device_path,
>> > + new_net_dp);
>> > + if (ret != EFI_SUCCESS)
>> > + goto error;
>> > +exit:
>> > + net_dp = new_net_dp;
>> > + efi_free_pool(old_net_dp);
>> > +
>> > + return ret;
>> > +error:
>> > + // Failed, restore
>> > + efi_free_pool(new_net_dp);
>> > +
>> > + return ret;
>> > }
>> >
>> > /**
>>
>>
More information about the U-Boot
mailing list